2025-03-04 [22:23:54.0082] ljharb: With regard to https://github.com/tc39/proposal-pipeline-operator/issues/310#issuecomment-2696324408, is there an actual “Tennent’s correspondence principle” of PL design that we’re trying not to break? The best answer I can find defining that phrase is https://softwareengineering.stackexchange.com/questions/116395/what-is-the-good-explanation-of-tennents-correspondence-principle, which seems to be some descriptive advice regarding correspondence between variable definitions and producer parameters. Right now, under https://github.com/tc39/proposal-pipeline-operator/wiki/Bikeshedding-the-Hack-topic-token#disqualified-tokens, variable identifiers are disqualified because variable tokens “may unexpectedly shadow outer variables”. Which is true. (I do agree that we should do our best to avoid any variables, for what it’s worth.) [22:24:08.0830] * ljharb: With regard to https://github.com/tc39/proposal-pipeline-operator/issues/310#issuecomment-2696324408, is there an actual “Tennent’s correspondence principle” of PL design that we’re trying not to break? The best answer I can find defining that phrase is https://softwareengineering.stackexchange.com/questions/116395/what-is-the-good-explanation-of-tennents-correspondence-principle, which seems to be some descriptive advice regarding correspondence between variable definitions and producer parameters. (I do agree that we should do our best to avoid any variables, for what it’s worth. Right now, under https://github.com/tc39/proposal-pipeline-operator/wiki/Bikeshedding-the-Hack-topic-token#disqualified-tokens, variable identifiers are disqualified because variable tokens “may unexpectedly shadow outer variables”. Which is true.) [22:24:18.0691] * ljharb: With regard to https://github.com/tc39/proposal-pipeline-operator/issues/310#issuecomment-2696324408, is there an actual “Tennent’s correspondence principle” of PL design that we’re trying not to break? The best answer I can find defining that phrase is https://softwareengineering.stackexchange.com/questions/116395/what-is-the-good-explanation-of-tennents-correspondence-principle, which seems to be some descriptive advice regarding correspondence between variable definitions and producer parameters. (I do still agree that we should do our best to avoid any variables, for what it’s worth. Right now, under https://github.com/tc39/proposal-pipeline-operator/wiki/Bikeshedding-the-Hack-topic-token#disqualified-tokens, variable identifiers are disqualified because variable tokens “may unexpectedly shadow outer variables”. Which is true.) [22:31:38.0812] it's a TODO item in the terminology doc https://github.com/tc39/how-we-work/blob/main/terminology.md#todo-considerations [22:33:39.0824] rkirsling: Nice, thanks for pointing me there. It's been a while. [22:34:13.0517] but yeah it's a deep cut that seems to lack a primary source(??) yet https://fanf.livejournal.com/118421.html gives a clear sentence of "the meaning of an expression or statement should not change when an extra level of block structure is added around it" [22:36:04.0505] It's a pretty good goal. Even if it doesn't seem to actually have been named by Tennent—at least according to Crockford. We might want to give it a better name… [22:36:39.0687] In general, the goal should be to reduce modality and contextuality of code as much as possible, which imposes cognitive burden. [22:36:41.0240] honestly agree [22:36:56.0834] * It's a manifestation of the general goal to reduce modality and contextuality of code as much as possible, which imposes cognitive burden. [22:37:00.0745] * It's a manifestation of the general goal to reduce modality and contextuality of code as much as possible, to reduce cognitive burden. [22:37:02.0370] just give it our own name and relay credit for the idea where it's due [22:39:19.0906] ('cause hell, even the acronym is terrible -- who wants to overload TCP :joy:) [22:39:29.0354] * ('cause hell, even the acronym is terrible -- who wants to overload TCP 😂) [23:29:32.0854] definitely - https://yehudakatz.com/2012/01/10/javascript-needs-blocks/ https://www.tomshouseofdollars.com/Tennent/ [23:30:01.0528] it’s been on tc39 bingo for a long time :-p [23:30:38.0295] yeah but not like, in a continuous way [23:31:17.0026] it's one of those "you had to be there, otherwise for years you're just like _what's the deal with that_ anytime it comes up" phrases [23:32:25.0700] like it always looks like a misspelling of David Tennant's name and I'm like, what's Dr. Who got to do with it [00:03:17.0478] It really does need a better name than a false eponym. Like “block-wrapped equivalence” or “in-block identity”. [00:19:19.0903] or something like "invariance under rescoping" [00:21:16.0352] * or something like "invariance under rescoping" or "scope context (in)sensitivity" [00:27:40.0179] Single Semantic Scope [00:31:38.0086] “Stable Under Scope”. “That variable is SUS.” [00:33:46.0537] (I kind of suspect that that new #310 issue was LLM generated; it has that feeling…) [00:34:38.0912] * “Stable Under Scope”. E.g., “That variable is SUS.” [00:35:31.0555] * “Stable Under Scopes”: “That variable is SUS.” “Stable Between Scopes”: “That variable is SBS.” [00:36:15.0692] * “Stable Under Scope”: “That variable is SUS.”🔪
“Stable Between Scopes”: “That variable is SBS.” [09:10:54.0064] Hm, yeah, a better name is indeed needed. We have a similar policy in CSS that adding an unstyled inline between some text and an ancestor should not cause observable behavior changes in any inherited properties; we refer to it informally as "inline transparency" [09:11:15.0855] * Hm, yeah, a better name is indeed needed. We have a similar policy in CSS that adding an unstyled inline between some text and an ancestor should not cause observable behavior changes in any inherited properties specified on the ancestor; we refer to it informally as "inline transparency" [09:12:01.0315] Additional Scope Transparency, I don't think there's any other PL-related AST acronym ^_^ [13:51:03.0866] * “Stable Under Scope”: “That variable is SUS.” 🔪 “Stable Between Scopes”: “That variable is SBS.” 2025-03-05 [16:39:58.0311] It’s been a long time, and yet I’m realizing now that nobody seems to have proposed `..` as the topic reference. ```js x = grunt.config('uglify.all.files') |> Object.keys(..)[0]; x = pkgs[0] |> npa(..).escapedName |> await npmFetch.json(.., opts); x = cb(predicate) |> _.negate(..) |> _.filter(obj, .., context); Object.keys(envars) .map(envar => `${envar}=${envars[envar]}`) .join(' ') |> `$ ${..}` |> chalk.dim(.., 'node', args.join(' ')) |> console.log(..); context |> (.. && (..).nodeType ? (..).ownerDocument || .. : document) |> jQuery.parseHTML(match[1], .., true) |> jQuery.merge(..); ``` `..` as the topic doesn’t seem bad. It only really clashes with property accesses—which are already postfix, so you would rarely use pipe on them anyway. And spread syntax, I guess, but doing that on the topic would probably be even rarer. `..` is even more visually lightweight than `^^`, and it’s easier to type than `^^` and `%%`. Plus, `..` usually gets its own typographic ligatures in fancy code fonts, for those who are into those. [16:48:05.0977] why would it be rare to have an iterable topic? [16:52:58.0162] I don’t think having an iterable topic would be rare, but I do think that spreading an iterable topic into a function call or array/object literal would be pretty rare. [16:55:26.0352] * I don’t think having an iterable topic would be rare, but I do think that spreading an iterable topic into a function call or array/object literal in a pipe would be pretty rare. ```js arr |> f(...(..)) // How often would this happen? ``` Though, even when it does happen, `...(..)` isn't the end of the world either. I think that might be worth the tradeoff versus `%%` or `^^`. And my intuition is that it’d be significantly easier to parse, too. [16:56:21.0363] * I don’t think having an iterable topic would be rare, but I do think that spreading an iterable topic into a function call or array/object literal in a pipe would be pretty rare. ```js x |> createIterable(..) |> variadicFn(...(..)) |> g(.., 0) // How often would this happen? ``` Though, even when it does happen, `...(..)` isn't the end of the world either. I think that might be worth the tradeoff versus `%%` or `^^`. And my intuition is that it’d be significantly easier to parse, too. [16:56:50.0451] * I don’t think having an iterable topic would be rare, but I do think that spreading an iterable topic into a function call or array/object literal in a pipe would be pretty rare. ```js // How often would spreading a topic into a variadic function happen? x |> createIterable(..) |> variadicFn(...(..)) |> g(.., 0) ``` Though, even when it does happen, `...(..)` isn't the end of the world either. I think that might be worth the tradeoff versus `%%` or `^^`. And my intuition is that it’d be significantly easier to parse, too. [16:57:21.0086] * I don’t think having an iterable topic would be rare, but I do think that spreading an iterable topic into a function call or array/object literal in a pipe would be pretty rare. ```js // How often would you need to spread a topic // into a variadic function? x |> createIterable(..) |> variadicFn(...(..)) |> g(.., 0) ``` Though, even when it does happen, `...(..)` isn't the end of the world either. I think that might be worth the tradeoff versus `%%` or `^^`. And my intuition is that it’d be significantly easier to parse, too. [17:00:52.0864] * I don’t think having an iterable topic would be rare, but I do think that spreading an iterable topic into a function call or array/object literal in a pipe would be pretty rare. ```js // How often would you need to spread a topic // into a variadic function? x |> createIterable(..) |> variadicFn(...(..)) |> g(.., 0) ``` Though, even when it does happen, `...(..)` isn't the end of the world either. I think that might be worth the tradeoff versus `%%` or `^^`. And my intuition is that `..` would be significantly easier to parse than `%%` or `^^`, too. [17:01:42.0301] * I don’t think having an iterable topic would be rare, but I do think that *spreading* an iterable topic into a function call or array/object literal within a pipe would be pretty rare. ```js // How often would you need to spread a topic // into a variadic function? x |> createIterable(..) |> variadicFn(...(..)) |> g(.., 0) ``` Though, even when it does happen, `...(..)` isn't the end of the world either. I think that might be worth the tradeoff versus `%%` or `^^`. And my intuition is that `..` would be significantly easier to parse than `%%` or `^^`, too. [17:04:45.0648] * I don’t think having an iterable topic would be rare, but I do think that _spreading_ an iterable topic into a function call or array/object literal within a pipe would be pretty rare. ```js // How often would you need to spread a topic // into a variadic function? x |> createIterable(..) |> variadicFn(...(..)) |> g(.., 0) // Maybe "appending" a value into an array copy is more common: x |> fnThatReturnsArray(..) |> [ ...(..), 0 ] ``` Though, even when it does happen, `f(...(..))` or `[ ...(..) ]` wouldn't be the end of the world either. I think that they might be worth the tradeoff versus the disadvantages of `%%` or `^^`. And my intuition is that `..` would be significantly easier to parse than `%%` or `^^`, too. [17:05:06.0938] * I don’t think having an iterable topic would be rare, but I do think that _spreading_ an iterable topic into a function call or array/object literal within a pipe would be pretty rare. ```js // How often would you need to spread a topic // into a variadic function? x |> fnThatReturnsIter(..) |> variadicFn(...(..)) |> g(.., 0) // Maybe "appending" a value into an array copy is more common: x |> fnThatReturnsIter(..) |> [ ...(..), 0 ] ``` Though, even when it does happen, `f(...(..))` or `[ ...(..) ]` wouldn't be the end of the world either. I think that they might be worth the tradeoff versus the disadvantages of `%%` or `^^`. And my intuition is that `..` would be significantly easier to parse than `%%` or `^^`, too. [21:00:04.0004] `arrayOfNumbers |> Math.max(...(..))`? [21:01:43.0055] ^ this seems like a nice syntactic "test case" [21:02:12.0884] * ^ this seems like it'd be a nice syntactic "test case" for considering the various options against [21:02:23.0255] * ^ this seems like a nice syntactic "test case" to consider the various options against [21:04:28.0032] * ^ this seems like a nice syntactic "test case" for evaluating the various options [22:09:18.0331] Yeah, I’ll acquiesce that flattening into Math.max and min does happen. Though I’d be a bit surprised if such calls exceeded 0.1%, let alone 1%, of all JS function calls. And just a space separating `...` and `..` should work, without parentheses, right? ```js arr |> Math.min(0, ... ..) |> numFormat.format(..) |> console.log(`Max is {..}`); arr |> Math.min(0, ...^^) |> numFormat.format(^^) |> console.log(`Max is {^^}`); arr |> Math.min(0, ...%%) |> numFormat.format(^^) |> console.log(`Max is {%%}`); arr |> Math.min(0, ...#_) |> numFormat.format(#_) |> console.log(`Max is {#_}`); ``` I’m actually kind of excited about `..` as another candidate topic token. It’s so “lightweight” compared to the other choices left. [22:10:08.0767] * Yeah, I’ll acquiesce that flattening into Math.max and min does happen. Though I’d be a bit surprised if such calls exceeded 0.1%, let alone 1%, of all JS function calls. And just a space separating `...` and `..` should work, without parentheses, right? ```js arr |> Math.min(0, ... ..) |> console.log(`Max is {..}`); arr |> Math.min(0, ...^^) |> console.log(`Max is {^^}`); arr |> Math.min(0, ...%%) |> console.log(`Max is {%%}`); arr |> Math.min(0, ...#_) |> console.log(`Max is {#_}`); ``` I’m actually kind of excited about `..` as another candidate topic token. It’s so “lightweight” compared to the other choices left. [22:11:22.0741] * Yeah, I’ll acquiesce that flattening into Math.max and min does happen. Though I’d be a bit surprised if such calls exceeded 0.1%, let alone 1%, of all JS function calls that would benefit from piping. And just a space separating `...` and `..` should work, without parentheses, right? ```js arr |> Math.min(0, ... ..) |> console.log(`Max is {..}`); arr |> Math.min(0, ...^^) |> console.log(`Max is {^^}`); arr |> Math.min(0, ...%%) |> console.log(`Max is {%%}`); arr |> Math.min(0, ...#_) |> console.log(`Max is {#_}`); ``` I’m actually kind of excited about `..` as another candidate topic token. It’s so “lightweight” compared to the other choices left. [22:11:52.0069] * Yeah, I’ll acquiesce that flattening into Math.max and min does happen. Though I’d be a bit surprised if such calls exceeded 0.1%, let alone 1%, of all JS function calls that would benefit from piping. And just a space separating `...` and `..` should work, without parentheses, right? `[ ... .. ]` rather than `[ ...(..) ]`. ```js arr |> Math.min(0, ... ..) |> console.log(`Max is {..}`); arr |> Math.min(0, ...^^) |> console.log(`Max is {^^}`); arr |> Math.min(0, ...%%) |> console.log(`Max is {%%}`); arr |> Math.min(0, ...#_) |> console.log(`Max is {#_}`); ``` I’m actually kind of excited about `..` as another candidate topic token. It’s so “lightweight” compared to the other choices left. [22:16:00.0984] * Yeah, I’ll acquiesce that flattening into Math.max and min does happen. Though I’d be a bit surprised if such calls exceeded 0.1%, let alone 1%, of all JS function calls that would benefit from piping. And just a space separating `...` and `..` should work, without parentheses, right? `x |> [... ..]` rather than `x |> [...(..)]`. ```js arr |> Math.min(0, ... ..) |> console.log(`Max is {..}`); arr |> Math.min(0, ...^^) |> console.log(`Max is {^^}`); arr |> Math.min(0, ...%%) |> console.log(`Max is {%%}`); arr |> Math.min(0, ...#_) |> console.log(`Max is {#_}`); ``` I’m actually kind of excited about `..` as another candidate topic token. It’s so “lightweight” compared to the other choices left. [22:37:20.0067] I can't say I'm thrilled with how `...x` looks for property access. [22:42:29.0878] I'm still of the opinion that F#-style pipes would have been better. `arr |> (x) => Math.min(0, ...x)` means no special topic is needed, PFA or no PFA. [23:05:01.0779] that's a nonstarter tho based on browser feedback [23:05:10.0209] personally i think `..` is way worse than `^^` or `&&` [23:22:23.0144] I mostly like the `..` except that I think there are so many cases where parens would be needed (at least from a lint perspective) that one would almost need to consider it a four-character thing `(..)` [23:24:26.0480] * I mostly like `..` except that I think there are so many cases where parens would be needed (at least from a lint perspective) that one would almost need to consider it a four-character thing `(..)`, which might in itself rule it out [23:26:26.0853] I hate %% but I think this discussion makes me come around to ^^ a bit [23:26:32.0729] * I hate %% but I think this discussion actually makes me come around to ^^ a bit [00:12:59.0102] the main thing I like about ^^ is the shape resemblance with |> [00:27:08.0429] interesting [00:27:47.0121] I don't know about that, but in this context, I was appreciating how ^^ can sit next to . and ... nicely [00:30:18.0248] if we could type dots above like ˙˙ then that'd be cool but we can't so ^^ stands as a typable alternative [01:05:03.0089] > <@ljharb:matrix.org> that's a nonstarter tho based on browser feedback F# was stalled due to browser feedback about too many closures? Couldn't we define something that looks 100% like an arrow but doesn't actually introduce a function, it just binds some local variable) the "parameters" evaluates its body? Since it's an iife anyway  [01:05:24.0094] * F# was stalled due to browser feedback about too many closures? Couldn't we define something that looks 100% like an arrow but doesn't actually introduce a function, it just binds some local variable (the "parameters") and evaluates its body? Since it's an iife anyway  [01:09:54.0114] Maybe even syntactically limiting it to 1 parameters, like we do for setters [01:10:05.0460] * Maybe even syntactically limiting it to 1 parameter, like we do for setters [01:48:23.0513] Something that looks exactly like a function but isn't a function sounds not great [02:00:22.0290] the main advantage for the reader of f# is that it probably wouldn't be arrow functions, and would instead be curried functions - so looks almost like method calls. If every f# pipe was an arrow that sounds verbose ``` getData() |> _ => Object.entries(_) |> _ => handleEntries(_) |> _ => Object.fromEntries(_); ``` vs ``` getData() |> Object.entries(^^) |> handleEntries(^^) |> Object.fromEntries(^^); ``` [02:01:34.0022] if we can't find a token, then maybe the pseudo arrow is a good compromise [02:03:53.0830] I mean, either an arrow-like thing to give a name, or an expression that just gets called, with a cover grammar  [02:04:16.0793] If it starts with `( identifier ) =>` it's one, otherwise its the other [02:05:11.0600] > <@aclaymore:matrix.org> Something that looks exactly like a function but isn't a function sounds not great You wouldn't really be able to observe that other than the restriction in the parameters, because you can't get a reference to it (since it's basically an iife) [07:16:19.0006] there have been suggestions to embed an identifier inside the pipe syntax like e.g. `getData() ~x> Object.entries(x) ~a> handleEntries(a) ~a> Object.fromEntries(a)` (_although I think they were `|identifier>`, which wouldn't work because `getData() |x> Object.entries(x) |a> handleEntries(a) |a> Object.fromEntries(a)` already has a valid parse_) [07:29:18.0948] thanks I hate it [07:30:18.0805] (I mean l do find `|x>` amusing à la bra-ket notation...but also that's not what this is) [07:32:15.0744] > <@rbuckton:matrix.org> I can't say I'm thrilled with how `...x` looks for property access. For what it’s worth, `...x` would always tokenize as spread-x and never topic-member-x, and `a |> ...x` would always be a syntax error (no topic in pipe body). And you would nearly always want to do `a.x` instead of `a |> .. .x` anyway… [07:33:28.0743] > <@gibson042:matrix.org> there have been suggestions to embed an identifier inside the pipe syntax like e.g. `getData() ~x> Object.entries(x) ~a> handleEntries(a) ~a> Object.fromEntries(a)` (_although I think they were `|identifier>`, which wouldn't work because `getData() |x> Object.entries(x) |a> handleEntries(a) |a> Object.fromEntries(a)` already has a valid parse_) Speaking of `~`, I don’t think `~~` as topic has been proposed either. It’s not bad either, maybe more of a pain to type. [07:34:43.0447] Though it may be a problem with `~~~`… [07:35:32.0899] ASI in ``` x |> ~~ f() ``` is bad though [07:35:32.0951] * Though it may be a problem with XOR in ~~~… [07:47:27.0736] took me a sec to decipher what you meant but oof yeah that's a non-starter [09:46:56.0507] yeah i think that makes the syntax a non-starter, fwiw [09:48:06.0435] We've avoided unary operators for the ASI reasons Nicolo just brought up. If we're using an existing operator it has to be binary, so them being adjacent is impossible. [10:23:59.0472] The committee temperature for `..` seems quite cold, alas. I suppose it’s understandable. I’ll add `..` and `~~` to the wiki table of rejected topic tokens later with reasons. I also need to post an update in #232 saying basically that the champions have been busy proposing lower-hanging fruit over the past two years. [10:29:54.0187] There *was* some talk at the last plenary’s TG5 about maybe doing another pipe usability study, sparked by a suggestion from Michael Ficarra. At the time, there was some surprising new positive enthusiasm about Hack pipe from Mark Miller, too. If a new TG5 study demonstrates usability benefits from the pipe operator, when compared to deeply nested expressions or temporary variables, then other committee members may become more amenable to pipe. The study could also evaluate the usability of the remaining candidate topic tokens. (The TG5 meeting notes aren’t up yet. If anyone knows how best to contact Mikbar Barash, I’d like to ask him to let me look at the meeting notes before he posts them.) [10:37:44.0754] @mikbar-uib:matrix.org ? [15:50:41.0422] note that it's Mik-*hail* Barash, abbreviated to mikbar 😅 [15:50:48.0479] * note that it's Mik_hail_ Barash, abbreviated to mikbar 😅 [15:50:59.0224] * note that it's _Mikhail_ Barash, abbreviated to mikbar 😅