<html><head><meta http-equiv="Content-Type" content="text/html charset=utf-8"></head><body style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><div class=""><br class=""></div><div><blockquote type="cite" class=""><div class="">On Jan 9, 2018, at 8:28 AM, Nate Cook &lt;<a href="mailto:natecook@apple.com" class="">natecook@apple.com</a>&gt; wrote:</div><br class="Apple-interchange-newline"><div class=""><meta http-equiv="Content-Type" content="text/html; charset=utf-8" class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class=""><div class=""><blockquote type="cite" class=""><div class="">On Jan 9, 2018, at 4:12 AM, Jonathan Hull &lt;<a href="mailto:jhull@gbis.com" class="">jhull@gbis.com</a>&gt; wrote:</div><br class="Apple-interchange-newline"><div class=""><meta http-equiv="Content-Type" content="text/html charset=utf-8" class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class="">Some thoughts:<div class=""><br class=""></div><div class="">- How do I randomly select an enum?<br class=""></div></div></div></blockquote><div class=""><br class=""></div><div class="">Vote for SE-0194! :)</div><br class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><div class=""><div class="">- I like that RandomNumberGenerator doesn’t have an associated type. I agree that we should just spit out UInt64s for simplicity.</div></div></div></div></blockquote><div class=""><br class=""></div><div class="">It simplifies things a lot, at some performance cost. For example, the LCRNG type really produces UInt32s, so I have to use two of its outputs to generate one value, even if I don’t need that many bits.</div></div></div></div></blockquote><div><br class=""></div><div>I agree with you, but just to play devil’s advocate, why not have two outputs: one for UInt32 and another for UInt64? &nbsp;The generator itself would only have to provide one of the UInt types and the other would be provided by default implementation (either by stacking 2 UInt32s or discarding half of a UInt64). &nbsp;I know this is less minimal, but I don’t think it is actually complicated or hard to use. You just ask for the type that is most useful for you.</div><div><br class=""></div><br class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class=""><div class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><div class=""><div class="">- I don’t like how it is so closely tied with Range. &nbsp;I realize that both Int and Float work with Ranges, but other random types do not (e.g. CGVectors). &nbsp;You are special casing FixedWidthInteger and BinaryFloatingPoint, which are very important… but we lose the ability to deal with other randomly generated types.</div><div class=""><br class=""></div><div class="">- Following on the previous point, I don’t like that the code for dealing with Integers/Floats is in Range. &nbsp;It feels like things aren’t properly encapsulated.&nbsp;</div></div></div></div></blockquote><div class=""><br class=""></div><div class="">I actually agree with you, and for getting individual values prefer the form `let x = Int.random(in: 1…10)`. Here’s how I got to what’s in the playground:</div><div class=""><br class=""></div><div class="">1) We definitely want to be able to select a random element from a collection.</div><div class="">2) Given that, we’ll have (1…10).random() and (0..&lt;10).random() even if those aren’t what we prefer, and people will use them.</div><div class="">3) If people use that construction for integers, it will be strange to not have the same facility for floating-point numbers.</div><div class="">4) Once we have the range-based capability for both, the type-based versions are redundant (i.e., they can be added in the future if we decide we made the wrong decision by excluding them).</div><div class=""><br class=""></div><div class="">You’re of course correct that a pattern of range-based random functions doesn’t extend well to other types. I show on the last page a couple different ways of writing those, for Bool and Data. Most of the other types you’d want to create lie outside the Swift standard library, so we can’t address really those here.</div></div></div></div></blockquote><div><br class=""></div><div>Right. I guess my thought is that I would like them to be able to use a standard creation pattern so it doesn’t vary from type to type (that is the whole point of “unification” in my mind). &nbsp;In my own code, I have a concept of constraint, of which a set are passed to the object being created. This allows me to random create colors which look good together, etc…. &nbsp;I then have some convenience methods which just automatically create an appropriate constraint from a range where appropriate. &nbsp;I’d really like to see something standard which allows for constraints other than simple ranges.</div><div><br class=""></div><div>I think I would feel at least a little better if the range stuff worked for any type which can be put in a range…</div><div><br class=""></div><div>I also feel like the focus on ranges to the exclusion of everything else is one of those cute things that will come back to bite us later. &nbsp;My main focus is on random things which are presented to the user in some way, and I realize that is a different use-case than most programmers. Even for things like pure number generation, I am worried about things like sig-figs and not just range.</div><br class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class=""><div class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><div class=""><div class="">- Why bother supporting non-closed Ranges at all? &nbsp;If you only allow closed ranges, then you can’t end up with an empty range. The only difference in behavior I can think of is on floating point, but I can’t think of a use-case where excluding the supremum is actually useful in any real world way.</div></div></div></div></blockquote><div class=""><br class=""></div><div class="">Half-open ranges are a major use case for generating random numbers, particularly when working with collections. Whenever you see that someone’s written `random() % n`, that’s the half-open range 0..&lt;n.</div></div></div></div></blockquote><div><br class=""></div>Ok, you convinced me on the half-open ranges.</div><div><br class=""></div><div>I still think the best way to handle the 'random() % n' issue is to look for it as a pattern and then just issue a warning/fixit.</div><div><br class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class=""><div class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><div class=""><div class="">- This may sound strange, but I would <i class="">really</i> like to see Bool handled as a default implementation on the generator protocol itself. &nbsp;On my own version of this I have both the ‘coinFlip()’ and ‘oneIn(_ num:Int)’ methods which I find extremely useful. &nbsp;CoinFlip just gives you a random bool, whereas you can say things like oneIn(100) to get ‘true’ roughly 1 out of every 100 times you call it. &nbsp;These are useful for branching randomly. &nbsp;They are most useful on the source/generator itself because it is ergonomic when you need to rewind the source.</div></div></div></div></blockquote><div class=""><br class=""></div><div class="">Bool is certainly a very important type to be able to randomly generate. I’m not opposed to it being included in a proposal, but it’s simple enough to do on your own that it didn’t pass the “minimal” test that I was using in the playground. You could use something like this static method:</div><div class=""><br class=""></div><div class=""><div class=""><div class="">&nbsp; &nbsp; extension Bool {</div><div class="">&nbsp; &nbsp; &nbsp; &nbsp; static func random(</div><div class="">&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; probability: Double = 0.5,&nbsp;</div><div class="">&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; using generator: RandomNumberGenerator = Random.default</div><div class="">&nbsp; &nbsp; &nbsp; &nbsp; ) {</div><div class="">&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; return (0.0 ..&lt; 1.0).random(using: generator) &lt; probability</div><div class="">&nbsp; &nbsp; &nbsp; &nbsp; }</div><div class="">&nbsp; &nbsp; }</div><div class=""><br class=""></div><div class="">I don’t think there should be any value-producing methods on generators—most users shouldn’t need to think about generators at all, and the ones who have a specific need (repeatability, rewinding, etc) should be able to use the same APIs as the people who aren’t thinking about them.</div></div></div></div></div></div></blockquote><div><br class=""></div><div>I can add them easily myself. That said, it is very nice ergonomically for branching randomly.</div><div><br class=""></div><div><span class="Apple-tab-span" style="white-space:pre">        </span>if myGenerator.oneIn(3) {</div><div><span class="Apple-tab-span" style="white-space:pre">                </span>//This happens 1/3 of the time</div><div><span class="Apple-tab-span" style="white-space:pre">        </span>}else{</div><div><span class="Apple-tab-span" style="white-space:pre">                </span>//This happens 2/3 of the time</div><div><span class="Apple-tab-span" style="white-space:pre">        </span>}</div><div><br class=""></div><br class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class=""><div class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><div class=""><div class="">- IMO distributions should be sources/generators themselves which just wrap another source. &nbsp;We could have a subprotocol of RandomNumberGenerator which just semantically guarantees uniform distribution, and then distributions that need it could be sure of the input distribution. &nbsp;Notice this doesn’t limit the distribution to only be used for Integers as they are in the demo. They can be used anywhere a source can be used.</div></div></div></div></blockquote><div class=""><br class=""></div><div class="">I’d really like to maintain a clear line between generators and distributions (which I don’t think we need an additional protocol for). Distributions create values of a specific kind of type, with a particular distribution, which usually isn't suitable to use as the input for another algorithm that needs random data. Generators just pump out (hopefully) uniformly distributed bits, which distributions and other algorithms can then interpret and shape.</div></div></div></div></blockquote><div><br class=""></div><div>This gets to the heart of my main concern with the approach from the playground. &nbsp;Everything is special-cased. &nbsp;If the distribution only works for FixedWidthIntegers, then I can’t use it to create colors with that distribution, for example.</div><div><br class=""></div><div>What I like about wrappers is that they fit anywhere a generator fits, so the receiver doesn’t really have to know/think about it at all. &nbsp;You don’t have to rewrite it for each type you want it to work for.</div><br class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class=""><div class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><div class=""><div class="">- Having a subprotocol for generators which can be rewound is extremely important for entire classes of real-world problems. &nbsp;I have spent a lot of time using this and it solves a LOT of problems. For example, I have a Lorem Ipsum Generator which takes Attributes and a CGSize to fill. &nbsp;It works by branching (using the Bool methods above) and then rewinding bits which don’t fit (If you just futz with the last part instead of generating appropriate clauses, it won’t look right). &nbsp;I also have a bunch of backtracking algorithms which rely on this rewind ability. &nbsp;Plus numerous visual effects which rely on a repeatable rewindable source.</div><div class=""><span class="Apple-tab-span" style="white-space:pre">        </span><i class="">- Tl;dr: It isn’t enough to just have a seed, you need to be able to mark a state of a generator and return to that state later.</i></div><div class=""><br class=""></div><div class=""><span class="Apple-tab-span" style="white-space:pre">        </span>My RepeatableRandomSource Protocol has 3 extra methods:</div><div class=""><span class="Apple-tab-span" style="white-space:pre">        </span>- It takes a seed</div><div class=""><span class="Apple-tab-span" style="white-space:pre">        </span>- It has a mark() method which returns a token</div><div class=""><span class="Apple-tab-span" style="white-space:pre">        </span>- It has a returnToMark(_ mark:Mark) method which takes a token and restores the appropriate state&nbsp;</div></div></div></div></blockquote><div class=""><br class=""></div><div class="">That’s neat, I’ve never used a generator like this! The RandomNumberGenerator protocol would certainly allow for a generator with these features, and libraries or future additions to the standard library could include that kind of protocol.</div></div></div></div></blockquote><div><br class=""></div><div>Thanks. &nbsp;I am coming to that conclusion. &nbsp;I can build most of what I need on a simple base. &nbsp;I guess I am sad that others won’t be able to share in it as easily, since it is a really nice/powerful way to code. I see <i class="">so many</i>&nbsp;visual bugs in UIs which have generated things randomly (e.g. things dance around or change when you resize them), and it is like this wouldn’t have happened if they had easy access to a repeatable/rewindable source.</div><div><br class=""></div><div><br class=""></div><div>One additional question. &nbsp;How do you ergonomically get a Double which doesn’t have a range, but also isn’t NaN?</div><div><br class=""></div><div>Thanks,</div><div>Jon</div><div><br class=""></div><br class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class=""><div class=""><blockquote type="cite" class=""><div class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class=""><div class=""><div class=""><br class=""></div><div class=""><br class=""></div><div class=""><div class=""><div class=""><div class=""><blockquote type="cite" class=""><div class="">On Jan 8, 2018, at 11:02 AM, Nate Cook via swift-evolution &lt;<a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a>&gt; wrote:</div><br class="Apple-interchange-newline"><div class=""><meta http-equiv="Content-Type" content="text/html; charset=utf-8" class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class="">I created a playground to explore this question, starting with a minimal subset of the proposal’s additions and building from there. The attached playground demonstrates what’s possible with this subset on the first page, then uses subsequent pages to explore how the main random facilities of the C++ STL work under this model. (In my opinion, they work pretty well!)<div class=""><br class=""></div><div class="">The subset in the playground has three main differences from the proposal:</div><div class="">&nbsp;- It doesn't include a <font face="Menlo" class="">Randomizable</font> protocol or a <font face="Menlo" class="">random</font> property on numeric&nbsp;types.<br class="">&nbsp;- It doesn't include the static <font face="Menlo" class="">random(in:)</font> methods on numeric types, either.<br class="">&nbsp;- The <font face="Menlo" class="">RandomNumberGenerator</font> protocol doesn't have an associated type. Instead, it&nbsp;requires all conforming types to produce <font face="Menlo" class="">UInt64</font> values.<br class=""><br class=""></div><div class="">I’ve tried to include a bit of real-world usage in the playground to demonstrate what writing code would look like with these additions. Please take a look!</div><div class=""><br class=""></div><div class="">Nate</div><div class=""><br class=""></div><div class=""></div></div><span id="cid:ABC18921-24CA-48D8-BC24-C1F445F8C252@hsd1.ca.comcast.net." class="">&lt;Random.playground.zip&gt;</span><meta http-equiv="Content-Type" content="text/html; charset=utf-8" class=""><div style="word-wrap: break-word; -webkit-nbsp-mode: space; line-break: after-white-space;" class=""><div class=""></div><div class=""><br class=""></div><div class=""><div class=""><blockquote type="cite" class=""><div class="">On Dec 2, 2017, at 9:50 PM, Dave Abrahams via swift-evolution &lt;<a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a>&gt; wrote:</div><br class="Apple-interchange-newline"><div class=""><meta http-equiv="content-type" content="text/html; charset=utf-8" class=""><div dir="auto" class="">I don’t have much to say about this other than that I think the discussion seems way too narrow, focusing on spelling rather than on functionality and composability. &nbsp;I consider the “generic random number library” design to be a mostly-solved problem, in&nbsp;<span style="background-color: rgba(255, 255, 255, 0);" class="">the C++ standard library (<a href="http://en.cppreference.com/w/cpp/numeric/random" class="">http://en.cppreference.com/w/cpp/numeric/random</a>). &nbsp;</span>Whatever goes into the Swift standard library does not need to have all those features right away, but should support being extended into something having the same general shape. IMO the right design strategy is to <u class="">implement and use</u> a Swift version of C++’s facilities and only then consider proposing [perhaps a subset of] that design for standardization in Swift.<div class=""><div class=""><br class=""></div><div class="">Sent from my iPad<div class=""><br class="">On Dec 2, 2017, at 5:12 PM, Kyle Murray via swift-evolution &lt;<a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a>&gt; wrote:<br class=""><br class=""></div><blockquote type="cite" class=""><div class=""><meta http-equiv="Content-Type" content="text/html; charset=us-ascii" class=""><br class=""><div class=""><blockquote type="cite" class=""><div class="">On Dec 2, 2017, at 6:02 PM, Xiaodi Wu via swift-evolution &lt;<a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a>&gt; wrote:</div><br class="Apple-interchange-newline"><div class=""><span style="font-family: Helvetica; font-size: 12px; font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; text-align: start; text-indent: 0px; text-transform: none; white-space: normal; word-spacing: 0px; -webkit-text-stroke-width: 0px; float: none; display: inline !important;" class="">Instead, we ought to make clear to users both the features and the limitations of this API, to encourage use where suitable and to discourage use where unsuitable.</span></div></blockquote></div><br class=""><div class="">I like that you're considering the balance here. I've been lightly following this thread and want to add my thoughts on keeping crypto and pseudorandomness out of the name of at least one <font face="Menlo" class="">random</font> API intended for general use.</div><div class=""><br class=""></div><div class="">For someone who doesn't know or care about the subtleties of insecure or pseudorandom numbers, I'm not sure that the name <font face="Menlo" class="">insecureRandom</font> is effectively much different than <font face="Menlo" class="">badRandom</font>, at least in terms of the information it conveys to non-experts. To Greg's point, that's the opposite of the signal that the API name should suggest because it's what most people should use most of the time. As you say, this API is being designed for general use.</div><div class=""><br class=""></div><div class="">There's a cost to adding extra complexity to names, too. I don't think it's far-fetched to suspect that people who find <font face="Menlo" class="">insecureRandom</font> in an autocomplete listing or search will think "Where's the plain random function?"... and then go looking for a community extension that will inevitably provide a trivial alias:&nbsp;<font face="Menlo" class="">func random() { return insecureRandom() }</font>. That's the sort of adoption I'd expect from something for new programmers, like Swift Playgrounds. Someone's introduction to randomness in programming should probably involve no more than a straightforward mapping from the elementary definition, rather than forcing a teaching moment from more advanced math.</div><div class=""><br class=""></div><div class="">I think there are better places for caveat information than in the API names themselves; documentation being one clear destination. This is in contrast with&nbsp;<font face="Menlo" class="">Unsafe*Pointer</font>, where the safety element is critical enough to be elevated to be more than caveat-level information. You can go really far and create really cool things before these caveats start to apply. Using randomness as a black box in an intro programming environment seems like a much more common scenario than someone attempting to roll their first crypto by only reading API names and hoping for the best.</div><div class=""><br class=""></div><div class="">-Kyle</div></div></blockquote><blockquote type="cite" class=""><div class=""><span class="">_______________________________________________</span><br class=""><span class="">swift-evolution mailing list</span><br class=""><span class=""><a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a></span><br class=""><span class=""><a href="https://lists.swift.org/mailman/listinfo/swift-evolution" class="">https://lists.swift.org/mailman/listinfo/swift-evolution</a></span><br class=""></div></blockquote></div></div></div>_______________________________________________<br class="">swift-evolution mailing list<br class=""><a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a><br class=""><a href="https://lists.swift.org/mailman/listinfo/swift-evolution" class="">https://lists.swift.org/mailman/listinfo/swift-evolution</a><br class=""></div></blockquote></div><br class=""></div></div>_______________________________________________<br class="">swift-evolution mailing list<br class=""><a href="mailto:swift-evolution@swift.org" class="">swift-evolution@swift.org</a><br class=""><a href="https://lists.swift.org/mailman/listinfo/swift-evolution" class="">https://lists.swift.org/mailman/listinfo/swift-evolution</a><br class=""></div></blockquote></div><br class=""></div></div></div></div></div></div></blockquote></div><br class=""></div></div></blockquote></div><br class=""></body></html>