From 38c1ecd72a72069bdb310b7f996d6d46801bf827 Mon Sep 17 00:00:00 2001 From: Aki Date: Sun, 3 Nov 2024 22:18:46 -0800 Subject: [PATCH] 104th Meeting of TC39, notes (#349) * DRAFT: 104th Meeting of TC39, notes * Add conclusion to decimal presentation * Update RKG notes * Add conclusions and key points to JSSugar discussion * Update AsyncContext notes * Add summary to Secretary's Report, thank you Samina --------- Co-authored-by: Jesse Alama Co-authored-by: Ross Kirsling Co-authored-by: Shu-yu Guo Co-authored-by: Andreu Botella --- meetings/2024-10/october-08.md | 1382 +++++++++++++++++++++++++++++ meetings/2024-10/october-09.md | 1483 ++++++++++++++++++++++++++++++++ meetings/2024-10/october-10.md | 1189 +++++++++++++++++++++++++ 3 files changed, 4054 insertions(+) create mode 100644 meetings/2024-10/october-08.md create mode 100644 meetings/2024-10/october-09.md create mode 100644 meetings/2024-10/october-10.md diff --git a/meetings/2024-10/october-08.md b/meetings/2024-10/october-08.md new file mode 100644 index 00000000..2892b9d1 --- /dev/null +++ b/meetings/2024-10/october-08.md @@ -0,0 +1,1382 @@ +# 104th TC39 Meeting | 8th October 2024 + +----- + +**Attendees:** + +| Name | Abbreviation | Organization | +|-------------------|--------------|-----------------| +| Linus Groh | LGH | Bloomberg | +| Oliver Medhurst | OMT | IE (Porffor) | +| Waldemar Horwat | WH | Invited Expert | +| Chengzhong Wu | CZW | Bloomberg | +| Ron Buckton | RBN | Microsoft | +| Daniel Minor | DLM | Mozilla | +| Ashley Claymore | ACE | Bloomberg | +| Andreu Botella | ABO | Igalia | +| Ross Kirsling | RKG | Sony | +| Devin Rousso | DRO | Invited Expert | +| Ujjwal Sharma | USA | Igalia | +| Dmitry Makhnev | DJM | JetBrains | +| Bradford Smith | BSH | Google | +| Chip Morningstar | CM | Consensys | +| Daniel Ehrenberg | DE | Bloomberg | +| Chris de Almeida | CDA | IBM | +| Mikhail Barash | MBH | Univ. of Bergen | +| Nicolò Ribaudo | NRO | Igalia | +| Philip Chimento | PFC | Igalia | +| Mark Miller | MM | Agoric | +| Tom Kopp | TKP | Zalari | +| Ben Allen | BAN | Igalia | +| Jordan Harband | JHD | HeroDevs | +| Sergey Rubanov | SRV | Invited Expert | +| Istvan Sebestyen | IS | Ecma | +| Yulia Startsev | YSV | Mozilla | +| Marja Hölttä | MHA | Google | +| Yusuke Suzuki | YSZ | Apple | +| Keith Miller | KM | Apple | +| Michael Saboff | MLS | Apple | +| Justin Ridgewell | JRL | Google | +| Andrew Paprocki | API | Bloomberg | +| Jesse Alama | JMN | | +| Jonathan Kuperman | JKP | Bloomberg | +| Shu-yu Guo | SYG | Google | +| Yilong Li | LYL | Alibaba | +| Jack Works | JWK | Sujitech | + +## Approval of previous minutes and adoption of current agenda + +Presenter: Rob Palmer (RPR) + +RPR: Can I check, do we have—check the consensus appraisal for the previous minutes? No objections. + +RPR: The previous minutes are approved. + +RPR: Okay. And then for our upcoming meeting. So we have the current agenda published. So do we have consensus to adopt the current agenda? + +YSV: We have added extra time to this meeting. Is that right? + +YSV: I have a concern about the agenda and how much we have in it. As well as, the fact that a few things were added quite close to the deadline, which resulted in us being unable to fully review everything. So we will have to hold back on consensus on a couple of items because we weren’t able to appropriately review. I would like to highlight this: so that maybe we can discuss how to do this in the future in terms of ahead of time, recognizing that a meeting is too full, rather than extending the time. + +RPR: Okay. Thank you, YSV. + +RPR: All right. So I believe that we are adopting the current agenda and we will have those conversations later. + +RPR: So for now, let’s move ahead to the secretary’s report. Given today by Aki + +## Secretary's Report + +Presenter: Aki Braun (AKI) + +- [slides](https://github.com/tc39/agendas/blob/main/2024/tc39-2024-042.pdf) + +AKI: SHN had planned to be here, something came up that prevented her attendance and she’s heartbroken. She really likes attending these meetings. I am presenting the secretary's report, but only because SHN is unable to. + +AKI: Quickly, we will go through new ECMA projects, exe-com what is coming up there and I will chat with you about TPAC. We have a few things to review of the annex slides to make sure we are aware of. + +AKI: If anybody has heard anything about WinterCG or TC55, this is super exciting. The Web Interoperable Runtimes Community Group from W3C, Luca and Oliver and a handful of TG delegates along with some other people, discovered that they were not able to publish a standard as a community group. And trying to figure out the best way to move forward, the group decided that it made sense to move to ECMA. So we have a scope to describe the group will be standardizing and maintaining a minimum API for web servers to support APIs that allow JavaScript developers to write basically isomorphic JavaScript, write once, use in client, use on server. And there’s more to it than this. I think this is a very reductive description, but it’s a different type… They will be presenting to the ExeCom in a few weeks or SHN will be presenting the next ExeCom and we look forward to moving forward and publishing standards or technical reports. + +AKI: We have a TC56 proposal. A high-level shading language. HLSL which is currently in a—in the perfect place for standardization is what it is. So we are looking at using the royalty free patent policy that ECMA developed to create this technical committee and publish both the standard and a standard library for HLSL and also do a lot of liaison with other bodies to make sure as adopted and distributed and used. I don’t know how close that is to being concrete. I think—it seems very promising. + +AKI: All right. The ExeCom. This is in a couple of weeks. The 22 and 23 October in Geneva. The TC39 chairs will be preparing a report for the ExeCom that they go through every time. They are—there will be the confirmation of chairs and all of the different leadership throughout ECMA. And there will be elections for the ExeCom. So if you work for an ordinary member and you want to go for president or vice-president, consider doing that. ECMA needs leadership to understand the breadth that we do and a TC39 delegate is the perfect person for that. If you don’t work for an ordinary member, you can still participate in the ECMA ExeCom, there are 4 positions available. I strongly encourage everyone to throw their name in for the ExeCom. It’s good to see how ECMA works and that your dues money is invested responsibly. + +AKI: A week before last I went to W3C TPAC 2024. Went on a massive charm offensive. Talked to a lot of different working groups about all sorts of things we can be collaborating on as standards bodies without competing or stepping on toes. There’s a lot of that—we have a Venn diagram for us to be make to talk to each other. The obvious one is the i18n working group and TG2. There’s not enough overlapping participants. Right now, we are seeing how we can get together and make sure that we are communicating and we are sorting plans together and not trying to standardize something that maybe gets in the way of the other group’s plans or understanding. That may look like joint meetings. It may look like some sort of regular communication. I am not sure. It is something that I am actively working on. If anybody has opinions on that, I would love to hear them. Beyond just the I18N working group and the TG2, there’s also other sorts of impacts we could be having on W3C. They want to hear about security, work with us, on security concerns. They want to hear from us on our opinions on sustainability, environmental sustainability. Think, preventing E waste and power consumption and sort of all sorts of stuff we have the able to influence, at our level. There’s a lot of people who are really curious and excited by TG5. Every time I mentioned it to somebody, they said tell me more. So I look forward to seeing what TG5 does and finding ways for us to get there successes and projects back to W3C and see if there’s collaboration we should be doing there + +AKI: One last thing that I am going to save talking longer about for a future plenary is that W3C would like very, very much for us to consider horizontal reviews. That means, when they are publishing a standard that has any potential overlap with ECMA’s work, they want us to take a look at it and say, "that doesn't make sense" or "I have a concern here." They will reserve the right to say, "I am glad you said that, but that is something we can’t change". They would love the opportunity to review our standards before we publish them as well. And if they bring up a concern they have, we may wish to integrate their technical contributions (after signing the appropriate RFTG paperwork), or we may wish to say "thank you very much. We will be moving forward". Horizontal reviews would contain no obligations, contractual or otherwise. Basically, we don’t want or need to conflict with the web platform. We can discuss this further in a future plenary. + +AKI: Some reminders: the invited expert policy is pretty clear, but I want to make sure that everyone knows what we want the process to look like in the future. It’s very important to us that we are able to invite experts when appropriate, and it is something we’re timeboxing. Timeboxing so we don’t forget and end up with an IE representing a company that should be a full member. So there’s no reason to think that we are actually going to change allowing invited experts in any sense. But it’s going to start being an annual thing, where at the end of the year, Samina will just chat with people and say, hey. Are you still active? Yeah? Is your company interested in being a member? No. All right. Having those conversations. It will be an annual review, but only in the sense of making sure that people are actually still engaged. + +AKI: RPR touched on this, but a reminder of the code of conduct. I think really importantly, when we disagree, try to understand why. Code of conduct exists for when things go wrong. Right? I don’t doubt anybody’s desire to make sure that we have a professional and pleasant experience. Sometimes—sometimes disagreements happen. Let’s make sure we are not afraid to bring in the code of conduct committee to work things out. + +AKI: There’s some updates to the list of ECMA documents that you all may share about. If there’s a GA document that you want to access, we should be able to get it for you. We are looking at finding ways to make access a little bit more smooth, a little bit more fluid. + +AKI: TC39 specific documents that have been published recently, there’s the secretary report from the last meeting. Some contributor license agreement responses. We have minutes and slides from the last meeting and the venue and agenda from this meeting + +AKI: In the GA documents, there are the results from the postal vote on clarifying bylaws and approving TC55. The initial vote was a no. The TC55 has updated their scope and updated the program of work, and I think that—I am pretty sure we have resolved any concerns that were there. There’s also highlights to—if you want to see how ECMA works with our standards bodies and especially beyond W3C, which a lot of you understand, you want to check that out, and the next GA is going to be in New York at Bloomberg. + +AKI: I think that’s probably all I want to chat with you. RPR already covered the next slide, which is the next TC39 meeting. And the slide after, the upcoming GA, which is in New York in December and the upcoming ExeCom meeting in Geneva. Any questions for ECMA? + +### Speaker's Summary of Key Points + +1. New ECMA Projects: + +- A collaboration with the W3C Web Interoperable Runtimes Community Group WinterCG is underway. +- Proposal for a TC56 to establish standards for a high-level shading language (HLSL) has been initiated + +1. Upcoming ExeCom Meeting: + +- Scheduled for 22-23 October in Geneva + +1. Collaboration with W3C: + +- Ongoing efforts to build cooperative relationships with various W3C working groups. +- Consider "horizontal reviews" where ECMA and W3C review each other's standards to prevent conflicts and promote sustainability in technology. + +1. Invited Experts Policy: + +- Annual review to confirm engagement and membership interest of invited experts. + +1. Code of Conduct and Documentation: + +- A reminder of the code of conduct, particularly the importance of constructive disagreement. +- Emphasis on documenting summaries and conclusions for agenda items to facilitate future reference and accurate minute-keeping. + +1. Upcoming Events Dates Reviewed + +## 262 Editor’s Update + +Presenter: Shu-yu Guo (SYG) + +- [slides](https://docs.google.com/presentation/d/1HUMRtYebcfUI6wAIyVokeYxCRz6CEgJkkt2jvNPoRLA) + +SYG: We merged the fully-defined `Math.sqrt` which reached consensus at the previous meeting, to make it not implementation approximated anymore, but to fully define it. We had two normative fixes merged. One was the spec security bug in async generator. #535 in the reflector, it was private, to fix a bug in the TLA machine, that fixes a hang. That’s also merged. + +SYG: The only editorial changes is we removed the notion of a language value type as a first class thing. You should not talk about language value types and pass them around. If you need to do things with them, we did a look over the document, and most of the operation that is done to type is to compare two things that have the same type. There’s an AO for it called SameType, use that. Do not use the phrase "the type of". Instead, directly say, "is an X" or "is not an X". + +SYG: And we removed the AO that reflects the type because the type is no longer reflected. We just directly use the type. For spec authors, if you need to update any of your spec drafts, please do so. + +SYG: We integrated spell checks into the PR process and I will turn it over to MF to talk more about this. + +MF: Yes, so you may notice, if you submit a pull request to the spec, that there’s now a GitHub action that runs spell checking on the pull request. And it will look like this. In line in the diff view, it has detected a spelling error. I want to go over how this works to help you understand and resolve it. + +MF: So the way this works is that the action checks out the previous commit, and looks at all of the words that are used in the document. And by words, I mean it also splits apart things that are in camel case into its own word. That’s basically its dictionary. That together with the standard Oxford English dictionary. + +MF: It uses that as its dictionary to check against your change. So if you are using a new word that has never been used before, or has identified a problem that maybe is, like, an abbreviation that has not been used before, it’s a hint to how that word is used elsewhere in the document. It doesn’t mean it’s wrong, but you should take a second look at it to see if maybe there’s a convention we use or a different spelling. + +MF: I think that’s all you need to know. + +SYG: Okay. This is upcoming work that we carry over from slide deck to slide deck. No additions here. + +SYG: And that’s it. Thanks. + +### Summary/Conclusion + +In addition to ongoing editorial and infrastructure work, the editors have landed the following normative changes: + +- [#3354](https://github.com/tc39/ecma262/pull/3354): fully define Math.sqrt +- [Reflector#535](https://github.com/tc39/Reflector/issues/535): guard against reentrancy in AsyncGenerator.prototype.return +- Landed as https://github.com/tc39/ecma262/commit/4cb5a6980e20be76c648f113c4cc762342172df3 +- [#3357](https://github.com/tc39/ecma262/pull/3357): Fix TLA hang (obvious spec bug) + +## 402 Editor’s Update + +Presenter: Ben Allen (BAN) + +- [slides](https://docs.google.com/presentation/d/1Fx6ikx0rAb3XMos4fjA8QJ-8QD2yjQJOW3YDCGXu91k) + +BAN: So some—we have a number of editorial fixes. These are the closest to interesting. No, actually interesting. + +BAN: We have been doing work to update the style guide and stuff to better adhere to the style guide. So there’s a lot of behavior in 402 that is implementation and locale-dependent. And so previously, the phase “implementation defined and locale-dependent” was defined in several different places. It’s mostly we use the "ILD" initialization which was defined in several places. Also implementation and locale system-dependent behavior that was also the ILND was also defined in several places. We are now defining it in one place and linking it, which improves readability a great deal. + +BAN: This is a small one. Most of the Intl objects had a slot named relevant extension keys and `Intl.locale` which is not an Intl service constructor, which until recently had a slot named relevant extension keys that behaved differently from all the other Intl objects. ABL has written a PR which renamed that slot to locale extension keys. Because it kept catching people out, including me, so I am really glad about this one. + +BAN: This is a smaller one. So previously, 262 had standardized on using HTML character entities for goodly number of symbols. So Laguo and M dashes and forth, 262 it used those. A couple of years ago 262 used the unicode characters. We updated 402 to that standard. We have a number of PR coming down the pike. In the interest of time, I will go through them quickly. + +BAN: We have a spec bug, whereas specified, RelativeTimeFormat, and DatetimeFormat ignored the numberingSystem used. If you used this, even if you did, it's the Latin character 2. + +BAN: There’s a spec bug, we fixed that, no engine is affected. Some work with the style guide preferring—normative PR is related to that. And a couple related to how currency is formatted. The sort of interesting one is one that got more time to discuss about what our source for data on the minor units should be. Spec currently says ISO-4217. Engines vary. Some use that, and some use CLDR. The two data sources are used for different things. We have a PR. We will use web reality as a source for how many digits after this for implementing currency. That is all. Thank you. + +### Speaker's Summary of Key Points + +- Several spec bugs fixed, +- Updates to style guide and editorial changes to adhere to style guide +- Editorial changes to match norms established by 262. PR to make ISO-4217 a recommended rather than normative data source. + +## 404 Editor's Update + +Presenter: Chip Morningstar (CM) + +CM: JSON, represent. + +RPR: Thank you, CM. The first two-word status update. + +## Test262 Status Update + +Presenter: Jordan Harband (JHD) + +JDH: So all of the resizable ArrayBuffer tests are imported out of the staging directory in the main test corpus. We are working on reviewing more than 400 tests for explicit resource management. And `Atomics.pause` has test coverage. + +JHD: So that’s the only specific thing I had to say. In general, please help write tests and review them. + +RPR: Thank you, JHD. + +## TG3 Report + +Presenter: Chris de Almeida (CDA) + +CDA: TG3 is meeting about security things. Please join us. We don’t have anything to request consensus for. There's another thing on the plenary that SYG will be talking about. + +RPR: Thank you, CDA. + +## TG4 Report + +Presenter: Jonathan Kuperman (JKP) + +- [slides](https://docs.google.com/presentation/d/147UbvCUAoW07vkQjYKJrf8SqqmCnmicvVBwEu__lOu4/) + +JKP: But so last time I presented, I talked quite a bit about big plans we had for the future, new features like the scopes proposals. This talk is not about that. This is the goal we had for 2024. We’re not shaking things up or adding any big features but to add a foundation for the layers. We took the living Google document for source maps and started really going through and networking with a lot of the source map generators and consumers and having meetings and reading through code and figuring out what we can and can’t say. What I am getting at here is, this goal for 2024 and the thing we’re asking approval for is not new or breaking changes. It is like a lot of specification text documenting the current reality of source maps. Another goal we had outside of the specification itself was robust test coverage and validation. We will talk about it. We have made progress on automated tests running against all of the devtools and source map validator against source man generators. The big thing with the test, we have the TG4 test suite, on the map tests. We broke everything out in the testable things and created a test case for each one of those things + +JKP: This automated suite is integrated in Firefox and runs part of the CI. We have a branch into Chrome which is running, waiting on legal approval for this. But this has been already integrated we have got local patches in the repo, which make all devtools have the test passing or a few expected failures + +JKP: We have been using this as part of our—like part of—requirement for all new features. We work on the scopes feature before we present that to plenary, we will have full test coverage, mirroringing what we do in TG1 + +JKP: So, yeah. For the milestones, we have gone through working with the devtools to make sure they implement 100% of the feature. An example is we have had help from a number of folks getting WebKit to support the ignore list. This was the first TG4 feature, it used to be the—used x_google_ignoreList and we moved it over. That’s really gone well. We had a stable point where we just wrote everything down. Tried to get everybody all on the same page. + +JKP: So the final specification, we have this link here. This is the snapshot we are asking for approval on. This snapshot will point to the living draft, which is the main branch of the—I don’t know if I am saying that right. It will have a recommendation at the top of it that—for people looking to implement a source map to check out the living specification. We will update throughout the year. The ignore list, we marked it as deprecated, but not removed it. And suggested it be removed to the new ignore list. + +JKP: We have one PR pending—I will skip the second one. A lot of editorial improvements, being more clear in our descriptions and write specifications-style algorithms for how to do things. One example is that we have specified how to decode source map and how to extract the source mapping URL. + +JKP: With the mappings and annotations, that one has not yet merged. It’s pending a final review right now. Part of this request would be assuming reviews go through, we would like to include it. This is the 122 PR that I have linked here, yeah. So what happens after the approval? To get approval, notify the beginning of the 60-day opt-out period. Members have a chance to review the contributions and opt out, if they decide to do so. It’s under a different license and licensing the original work done by Google and Mozilla over into ECMA’s modified BSD. If we get both of those, we ask the Ecma GA for the final approval, and then if we got that, publish the edition so it might be 2025 instead of 2024, depending on how that works. And also working with AKI to take our generated bikeshed files and turn them into the PDF generation for the spec. + +JKP: Looking forward, a bunch of great work on the scopes proposal. It’s in a good state. Looking for final implementor approval and get the specification text written. It should merge with Chrome and hopefully Firefox devtools implementations on plenary. We want the TG4 test suite to be, legal approval from Chrome and the upstream implementation for WebKit and adding more support and other featuresThat is kind of my update. Seeking approval on the linked thing above, which does not add any large new features, but is a lot of text documenting the current state of text maps. + +RPR: Okay. There are any questions for JKP? + +MM: So I am unclear about what the—what the normative impact of the approval is that you are asking for. Does that mean that the—first of all, I like the sequence you are going through. It’s a very nice process. I was unprepared to make a normative approval decision on this, at this meeting. Certainly have not reviewed it. Are you asking for an approval that the process you have outlined moves forward, as you have stated it, or is there content such as the written down codification of existing practice that you are seeking to make normative? + +JKP: I am not 100% sure I understand and maybe if NRO is on the call could respond. The thing that we are seeking today would be to start the 60-day opt-out period. Saying we have cut this branch. This branch is what we, like, you know, to be approved by the GA. I think my struggle to answer is being new to the TC39 process. But my understanding we need to present something at some point that would get consensus approved from TG1, which starts off this process saying we have this artifact to become the official specification, and to start this legal process of 60-day opt out and and that goes out, taking T we are looking to start this—start this process. + +MM: So specifically, the approval is not asking that any technical content to be approved to become normative today? It’s that the process you have outlined is a good process to go forward, which I am very happy. + +NRO: Okay. So the list has had normative changes. The last primitive change was many months ago. And the rest of work we have been doing is just clarifying previously unclear parts of the spec. This is similar to the approval we usually seek around—for ECMA262, we needed to say this is done. Let’s now, as a committee, take to ECMA GA for publishing. So we need now to understand saying TC39 is happy with this document being published as a spec. + +RPR: So to be clear here, Mark, John is proposing the proposal, but this is also TC39’s confirmation point that this specification is ready to proceed. + +MM: I am not ready to approve that the specification itself is ready to proceed. I was not aware that there was going to be a normative approval request on this asked for, at this meeting. + +DE: MM, this has been mentioned at several past meetings, asking for this normative approval. We have been working towards this goal for, like, 6 months this year. Still member organizations will vote on whether to adopt this standard at the Ecma GA [General Assembly] meeting. This is not the final adoption of it. But it is TC39 recommending it. If we don’t meet this goal, then we are delayed by 6 months. I wonder if you have suggestions for how we could have explained this more clearly? + +MM: You said that you did state this previously, so I’m—I apologize for having missed that. That’s my fault. The—it doesn’t—so let me just clarify. It does not need to be approved today to make that goal, it just needs to be approved at this meeting. So you could give they -- + +DE: That’s right. + +MM: A couple of days to look it over, and then voice an opinion on Thursday, is that correct? + +DE: Yeah, that sounds great. + +MM: Okay. + +NRO: Yes. Specifically the last day we have to send the notice to the GA to a six-day period I think Thursday evening, because the GA will be on December 11th and we want to do it two months in advance. + +MM: Okay, okay, thank you. I’ll have something to say on Thursday, thank you. + +RPR: Thank you. So one other clarification is that this is all about tooling and things that does not happen at run time. + +MM: There are interesting security issues that have been pointed out about source maps. I want to look it over with regard to some of those, and in particular, the ours source—I mean, it’s been a very long time since I looked at the technical contents of source maps, so what I’m about to say might no longer even make sense. But the last I looked at it, it could include URLs to fetch more source maps, and where there are those URLs, somebody could be operating a debugger that was expected to deference to URL from its location from behind a firewall runtime, and local debugger behind a firewall is forced to—is caused to fetch from. Now, you know, with many other things that fall into that, it’s an HTTP get, not a post, but still, i’d like to just take a look at the whole thing organically with regard to various security issues. + +NRO: That specific thing has been removed. So you probably don’t have to worry about that. + +MM: Okay, good. That was actually my main worry. Okay, great. Okay. So without that to worry about, I’m much more confident I’ll have good news on Thursday, but I’d still like to wait until Thursday. + +CM: So I think I also missed some stuff here. And this is more of a question about process, I don’t have any particular concerns or objections about these specific standards efforts. But it’s just interestingly different from the way the TG2 process works, where essentially all of the internationalization stuff ends up getting run in excruciating detail through the plenary, and I don’t think we’ve seen much, if any, of the detail of the source maps stuff running through plenary. I don’t have any objection to the way you’ve done it, I actually prefer it. But I’m curious why the process is different and if this different process is acceptable to everyone where the people are actually interested in the source map stuff just engage directly with the TG4 process, why the same logic is not applied equally to TG2? + +NRO: We discussed this when TG4 was created and the reason for deferred process and it’s not something you can do within code. It’s just something that is implemented in, like, in develop tools that live in JavaScript execution, and there was—the committee was happy with giving more freedom for this reason to the TG4, also because most delegates were not as, I’d say, interested in source map as what TG2 does, so this has the result of not taking too much committee time. + +CDP: Yeah, I just wanted to mention, these are all fair comments, but they did specifically bring this to committee and get consensus for this method of working. + +CM: Yeah, I’m not sure how I missed that. As I say, I don’t have any objection, it’s more of a, wow, couldn’t we do that for internationalization too? + +DE: For internationalization [ECMA-402/TC39 TG2], we discussed this as well at previous meetings when this was on the agenda and we decided that, because it has this normative content for JavaScript itself, and it’s often good to get the reviews in plenary. So for TG2, we seek consensus for all stage advancement and normative changes in plenary. + +DE: I wanted to ask Jon, do we need to block on recommending this to the GA for standardization until the tests are relicensed? I mean, not all Ecma standards have conformance tests. + +JKP: Sorry, can you say that again? Do we need to block on GA until? + +DE: Until the tests are relicensed? I mean, making sure we have the tests before standardization is good, but it’s not, as far as I know, a general requirement. + +NRO: License problems is not about the test, it’s about the spec itself. + +DE: I see. Yeah, sorry. + +NRO: Because it’s the comments and we are trying to re-license it to BSD, which Ecma uses. I—given that there is the 60-day time period in which company cans say we are not okay with content being contributed by us, I don’t think we need to block on that, but I’m not super sure on how these rules work. + +DE: I think Ecma’s BSD license is usually used just for software, and not for the spec text. So let’s be intentional about this. Yeah, I’m hopeful that it’s, as you say, because the contributing organizations are all present, then it should be okay with respect to the spec text. + +JKP: Yeah, I think that was why I was confused. It’s not the testing, it’s the spec text. As to not take up the whole committee’s time, we can follow up. It seems okay and while we’re here, I’m happy to speak with people from Google and Mozilla and make sure we’re doing the right thing here. + +KG: (from TCQ) suggestion for the future: bold/otherwise call out normative TG updates on the agenda + +RPR: That seems easy to address. And that’s the end of the queue. Just given that we’re still in this time slot, do we want to call for consensus modular Mark’s check the next time? We may be able to address this without even coming back. + +JKP: Yeah, that would be great for me. At least either way to know where we stand on it, what’s left to do. That sounds great. + +RPR: So this ask for continuing is subject to the PR that still needs to be merged. And is also conditional on Mark’s consent later this week. + +JKP: Yeah, that sounds great. So, yeah, I guess in light of that, I would like to ask for consensus on this modulo, PR and Mark being able to approve it during this two-day plenary. + +RPR: You have positive support from Chris. And no objections. So, yeah, congratulations, you have conditional consensus. Thanks. + +### Speaker's Summary of Key Points + +- JKP presented the work done over the course of 2024 for source maps including enumerating all of the changes and improvements made to the specification. +- JKP asked for consensus approval and received feedback that committee members like MM would like a little bit of extra time to look through the specification. +- Going forward, TG4 will explicitly call out every normative change they make to the specification during their regular plenary status updates +- At the end, JKP asked for conditional approval modulo the one pending PR, giving us a chance to work with MM on an editorial fix and then will come back at the end of plenary and ask for official consensus then. + +### Conclusion + +- Consensus on changes and improvements to the spec, pending one [outstanding PR](https://github.com/tc39/source-map/pull/122) and MM's approval. +- In the continuation we received MM’s approval, added an [editorial note](https://github.com/tc39/source-map/pull/137) to the [specification](https://github.com/tc39/source-map/tree/2024). +- TC39 will refer the source map specification to the Ecma GA for standardization, beginning the 60-day opt-out period + +## TG5 Report + +Presenter: Yulia Startsev (YSV) + +- [slides](https://docs.google.com/presentation/d/156wJbnrIEt-hbkhh0paVAIjrG9L7oe_R9z211yS0bIA/edit?usp=sharing) + +YSV: Okay, so TG5 is our TG dedicated to experimentation. So we have regular meetings every month, and well, we’re trying at every TC39 plenary, we’re trying to have a colocated workshop on the topic of language research. We have a workshop co-located with this meeting, it will be at the University of Tokyo. If you’re interested in participating or observing, please come and let MBH or myself know. This is the agenda. We have a couple of topics on syntax and interactive programming. And there will be a keynote and panel discussion with TC39 members. It’s exciting because we’re going to have the folks from KAIST also come and present the work that they’ve been working on the WebAssembly standard with this new project they have called SpecTec, so that’s going to be quite exciting. + +YSV: Now, in case you’re unfamiliar with TG5, what we do is we try to bring work from academia to the industrial context of TC39. So we have scientific studies and proposals, we have development of methods, approaches and tools to aid understanding, design and specification of the language, and also to consider TC39 and JavaScript within the larger ecosystem of programming language design, specification and standardization. We have an issue list, and I’d like to invite everyone here who is a TC39 delegate to populate our issue list. The issue list is important because we have collaborations with universities and the researchers and students working at those universities are unsure how to get started with helping TC39 do research and good scientific work. So if you add things to our list, then a researcher might come along and pick them up. Now, what you see here is my brain dump of stuff that I know has been going on. It’s incomplete, and I think that having the input of the people in this room would really help us bring more interesting projects to, you know, bachelor students, masters students, Ph.D. students, postdocs, professors. And it will also help us build a collaboration with those academic institutions, which are interested in working with us. + +YSV: One that I’ll call out in particular is the usability studies. I think this is what most people here are familiar with in terms of TG5’s work, looking at, for example, syntax. That’s a common one. We currently have a study that is being run by a professor from UCSD. A study is currently in planning and just gotten approval, like, ethics board approval. I forget exactly the name of that for MessageFormat, so in the past, we’ve had concern about the maturity of MessageFormat 2.0. So they are going to do a study to validate whether or not the MessageFormat 2.0 specification does what it needs to do before we look at integrating it into the web platform, so that’s exciting. We had a study being run on Do Expressions and it was paused because the undergraduate student left. And that does happen sometimes. So we are working with students, and the work sometimes doesn’t get finished. But I think that this is something that can be resumed by another student. And we have several studies that we’ve run ourselves in the past. And we can use that as a model for guiding future work done by students. So this is something that I think everyone is familiar with from the side of research, but that research space is much, much larger. And, again, take a look at our list and come talk to us if you’ve got any questions. That’s the issue list. And that’s it. + +### Speaker's Summary of Key Points + +- TG5 workshop will be this Friday, see slides for the agenda +- Please add to our [issue list](https://github.com/tc39/tg5/issues), so that researchers will be aware of areas of interest where they can collaborate with TC39 + +## Code of Conduct updates + +Chris de Almeida (CDA) + +CDA: We do not have updates from the code of conduct committee, which is exactly the way that we like it. So, yeah. Very quiet. Just a reminder for anybody who is interested in joining us on the code of conduct committee, please reach out to one of us. Thank you. + +## Iterator sequencing for stage 2.7 + +Presenter: Michael Ficarra (MF) + +- [proposal](https://github.com/tc39/proposal-iterator-sequencing) +- [slides](https://docs.google.com/presentation/d/1Z5Bz_4xpwRX7tjwmrMakrj3_II8Qy40fnAq-TrPdt0U) + +MF: We are asking for Stage 2.7 for iterator sequencing. I typically start with a reminder of what we’re trying to do here. With iterator sequencing, we are trying to give a way to create an iterator from 0 or more iterables where that iterator yields each of the values that each of those iterables would yield in the order that they would yield them. So we’re trying to concatenate iterable values together, essentially. The last time this was discussed, I don’t remember when that was, we ended up choosing from a variety of solutions. We chose this one, which is an `Iterator.concat` function, that takes an iterator of iterables as a parameter. I’m going to go over some of the design decisions that we made during this process. + +MF: Okay. In line with our new normative conventions, we reject all iterable primitives. If you recall, we recently added that to our how-we-work document of our normative conventions, so only non-primitive iterables are acceptable parameters. We also, throughout this process it was our goal to allow you to conveniently interleave non-iterable values into your iterable—to kind of append just that value. And we talked about it in the last meeting. When we came to the decision to go with Iterator.concat, we simultaneously decided that it was sufficiently usable to just put that value in square brackets to interleave it. This creates an array, and that array is iterated with the value in line with the rest of them. We decided to drop the infinite inputs use case. It was not really as common, and there are solutions for that, if you really need to. You can flatMap the identity function and that was good enough for something that is so uncommon. + +MF: A change since the last time that this was discussed, though, is that we now eagerly validate all parameters. It was the case that if you had some number of iterables given as parameters, and if you don’t reach a parameter that was not iterable, you would just not get an error. For instance, if you’re concatenating an iterable before something that’s not iterable or if you just don’t advance it far enough to reach there. Now it is the case that all of the parameters are checked to be iterable before we return your iterator to you. So invalid inputs show up immediately. And here is just a reminder of some of the other solutions we considered. + +MF: So we have spec text. This is what it looks like. I’m not going to go over it right now, but you can view it on the repo. So there is, to my knowledge, a single remaining point of contention. And that is the handling of non-iterable values. So I just explained that we do this eager parameter validation, and when something is not iterable, we throw before we return an iterator to you. I’ve gotten feedback from one delegate that they would like this to not throw. The behavior that’s expected is the value—that the second line, basically acts like the first line. I don’t like this suggestion. I think that having the second line behave like the first line is not a reasonable behavior, and I’ll get into why in a moment. I would think a possible okay behavior would be to have it just be treated as an empty iterable. You know, adding no value—contributing no values to the resulting iterator. But that’s not what’s being suggested. + +MF: We have a lot of precedent for this. I think the strongest argument is that we already have a lot of existing places that accept iterables. We have for-of, we have positional destructuring, we have spread, `yield*`, probably others, and those all throw when they see non-iterables. So I think that argument alone should be sufficient to make this decision. But I’ll go over more reasons why. So if we went this route, you know, ignoring precedent, I think that it would be problematic because we would run into cases like this that say somebody defines some type of their own, some custom type. This one is a Maybe type. You can ignore those first three lines. We just have some Maybe type. And on that last line, that person is trying to append it to some existing iterator. And that’s—that works. That’s how the language would work in this scenario. If the author of that Maybe type later makes the Maybe iterable, this does a different thing. You no longer are appending that maybe value to this iterator. You’re doing whatever that iterator value is, which is probably producing a value if it’s Just or producing no values if it’s Nothing. I think this is a serious foot gun. The behavior that this author was looking for was to just append the Maybe. They could have put it in square brackets, but this worked consistently until it didn’t. And I think that that’s not the way we should design the language. + +MF: It’s been suggested that there is precedent for treating—for having an API in this style. One suggestion is Array.prototype.concat. I think this is not actually a valid precedent. We don’t have the same issues with Array.prototype.concat because while non-iterable things can get `Symbol.iterator` from an update, non-arrays don’t become arrays. We’re not going to talk about Symbol.isConcatSpreadable. We ignore that for the most part, even within this committee. Nobody knows about it. So non-arrays just don’t become arrays, so you don’t run into this same foot gun with code changing later. It’s still not suggested that you should write your code like this, but you don’t run into those updating kind of errors. I’m going to skip this slide for now, I don’t think we need to go into thenables just for time. Something I do think is precedent other than all the language features I mentioned that take iterables, the strongest precedent is that, you know, we’re basically defining iterator concat like this. It’s just, you know, the mental model should be that it’s just looping over all of the passed parameters and doing `yield*` on them. We have `yield*` and yield. They are separate things. We didn’t add just `yield` to the language and have it take non-iterables and just produce that non-iterable. If we had, that would be, you know, a valid thing to point to as precedent. But instead, we did not, so it is actually precedent in the other direction for throwing when we see non-iterables. + +MF: So this is our path forward on this last point of contention. We have four options that I’ve identified. Maybe there are others. The current spec text is it throws. As I said, we validate all parameters eagerly. So when we see a non-iterable, we will throw instead of returning an iterator. A previous version of the spec—possibly the last time those in the room right now have seen it—was when it produces an iterator that only throws when it gets to that point. So that would also align better with that naive generator that I have just shown on the slides. That’s also fine behavior in my opinion. We would drop the eager parameter validation. The option number 3 is the one that was suggested and the one I disprefer, which is yielding the non-iterable value. And option 4 is also an option, but it does not follow precedent within the language, and that’s yielding nothing for that non-iterable value. + +MF: So depending on how we move forward, it was requested that we might want to choose a different name to avoid any hint of a relationship to Array.prototype.concat. And I have some options for us here. We don’t have to look at them right now because we don’t know if we’re going to need them, but we can use this as a reference later if we need to get into that discussion. + +MF: That’s all I have. I would like Stage 2.7. + +RPR: Before we ask for 2.7, can you bring back the options slide? JHD is up first. + +JHD: Yeah, I wanted to address a few things you said. If you could step back a few slides. Okay. We’ll start here. So first of all, if they behave like empty iterators, that would be a massive footgun and horrific API design. Throwing is infinitely better than that; I don’t think that’s an option as all. As far as principles, I think you’re assuming principles that aren’t universal, and given that using array concat in exactly this fashion is actually somewhat common, albeit obscure, like, that has been an idiomatic thing for a long time, although certainly never a widely known one. So I just don’t agree that it’s unprincipled. + +JHD: As far as the existing places, none of those are—many of those, at least, are not combining iterators. They are just iterating iterators. So, for example, I think all of those cases take one iterator or iterable. And none of them take more than one. Like, you `for-of` one iterator and you destructure one of them. Spread is—they’re all applying to one iterator, so you talked a lot about precedent, but there’s no precedent for combining iterators at all, so I don’t think any of those things are, like, knockout precedents for this, one way or the other. Because this is a combinator, these are not. + +MF: I can respond to that point. I think spread and `yield*` are precedents. You can do it multiple times within an array initializer and that is analogous here, because you spread one of the parameters to Iterator.concat, you spread each of the values in an array initializer. Similarly with `yield*`, you `yield*` over one of the parameters on `Iterator.concat`—in the generator implementation. You are applying it in the same way. + +JHD: I can see where you’re coming from, but in the array spread case where you make the array literal and you’re spreading many iterators in, you can in fact, and people very frequently do, many times more than with array.concat, combine iterables in that, because the spread syntax applies to one item at a time. That flexibility exists because it’s needed. This proposal's API method does not have that flexibility because it’s taking an argument list. So it’s just not a clean mapping in the same way. But I’m also not saying that those places are unrelated. They’re clearly related. I just think that there’s—it’s not like, oh, because of those things we must do this, because they’re just distinct. They apply to one iterator iterable versus applying to multiples, I think is just—it’s distinct enough that we have flexibility here as a committee. But if you wanted to go to the next one, so adding support for a new protocol to an existing class is always a breaking change. Unavoidably, period. There’s no way to avoid that, because someone could have already been using it in a code path that could branch on the presence of that protocol and it’s the same as if you add a new method to a class, that’s potentially a breaking change. The difference is if the method or property you’re adding is not part of a known protocol, then there’s no intention or likelihood or expectation that someone is going to be branching on it, and so it’s reasonable and common and typical to call that a semver-minor change. If you add a new property called `.then` to any object, that’s a breaking change because it’s a known protocol. + +MF: I agree with you on `.then()`. I skipped that slide, but we can talk about it. `.then()` is a case where definitely you can not add it to -- + +JHD: It’s the same with `Symbol.iterator`. You can branch on it or throw or not throw. You can—it will do-produce different object identities if you pass it into `Iterator.from`. Right? I understand the desire for it not to be a breaking change, but it will always be one. You cannot just suddenly make something, like, be part of a protocol. Similarly, when we add a new protocol to ECMAScript, every built in that needs to have it can have it. If we add a new—I don’t know, a “TC39able” protocol today, we can’t add that symbol to existing built-ins next week. Like, it has to be one and done. Because it’s a breaking change. + +MF: I just don’t agree with that premise. + +JHD: I mean, in practice, it is objectively, empirically true. Whether it should be or not, that’s a debate. But to me, it’s irrelevant whether it should be, because it is. It will break tests and production apps, therefore, it’s a breaking change. + +RPR: There’s a couple of replies to this. + +MM: So the—I agree with JHD technically, that it’s a breaking change, but I disagree thematically. The—what we do as software engineers is to try to minimize rude surprises. And to have something be include in these lists of things being concatenated together as iterators. Change its behavior in the silent way, you know, if one non-throwing behavior to another. I think is just introducing a surprise, and it’s introducing a surprise that will mask bugs. So just I’m going to also absorb my other comment on the queue. The—I think that the behavior that MF is suggesting does the best that we can with regard to avoiding masking bugs. The thing about iterators is the things that the iterator that—that happens with the iterator happens some arbitrary amount of time into future, it might never be gotten to. And the fact that he’s not just checking, throwing on a non-iterable, but doing it eagerly, is the best we can do at alerting the programmer of cases that are more likely to be bugs in their program than they are likely to be ways in which they wanted to code things. If they—and once they know it’s a bug, it’s easy to correct. + +KG: Yeah, this is disagreeing narrowly with the point about we can’t make things iterable later. That’s just false. Readable stream was not async iterable for the first several years. It is starting to be now. That was a totally web compatible change. We can make things iterable later. People aren’t just switching on iterability very much, and that’s great, and they shouldn’t because that’s generally not a sign of well written code, so we should also not switch on iterability and not do in the language and not encourage people to dot and I expect that most people won’t, which means that as a sequence, we can make things iterable later, as we have successfully done with readable stream. + +JHD: Yeah, I mean, I’m—it’s great that it was web compatible, but that doesn’t mean it wasn’t breaking. That just means nobody did the thing that would have broken. + +KG: I mean, adding any property does that. There’s no difference between adding this property and any other. + +JHD: Yeah, you’re right. And as you and MM have alluded to, and MF as well, the likelihood of people doing something matters. And if someone’s unlikely to do it, we probably don’t have to worry as hard about breaking somebody and so on. I get all that. and branching on iterability - I wasn’t claiming that that’s a common thing people would do. It’s just in the general case, if branching on the protocol is something people are likely to do, then adding that protocol to an existing thing is going to break people. Maybe iterability doesn’t qualify for that, but as a general principle, it’s a concern we have to be aware of. As far as masking bugs, it is a fair point that if you pass a custom iterable or iterator, that—and you intend to surface something iterable and you implement it yourself for some reason and you did a bad job, that it’s nicer to have it throw early rather than wait until you happen to get to that iterator point maybe. But that’s also a thing people just don’t really do. Almost every—in practice, most people’s iterables or iterators are things that the language or platform have produced, which work correctly, and then that’s not really a concern. + +JHD: The one thing that I did think about earlier in this presentation, so the decision in general to try and start to no longer accept strings as iterables by saying, no, we’re only looking at iterable objects, that’s a good thing, because primitive strings being iterable caused a lot of problems. And I can see, if we make the change I want to `Iterator.concat` - you pass an array and you pass a string—it is true that it might be subtly confusing to some people who are aware of the footgun of iterable strings, that it does not spread the string, that it just sticks the whole string in there. And that the behavior that you want, MF, would throw right away and say which one do you want, turn it into an iterator or wrap it into an array or something like that. I agree for the people aware of the edge case, that’s clear. But I also would put some money on the fact that basically every single time somebody typed that array, string kind of combination, that they intend the string to be added as an item and not spread into characters. But it is a fair point that with the current semantics, we’re not guessing the user’s intention, and in general, I like that principle and I’m on board with it. + +JHD: Iterator helpers has been really valuable for me. I’ve refactored a lot of code to use it and it’s gotten things a lot cleaner when I’ve done so. And the array concat use case is a big point of friction, which is why I’m glad for this proposal with or without the semantics I want. But there are a lot of use cases I have in a lot of code that I’ll have to stick in a bunch of extra array brackets in order to use the current semantics you’ve got here. And that feels like an unfortunate ergonomic and memory hit when I can’t imagine anyone—basically I think that likelihood that someone is going to run into the footgun you’re concerned about is very, very small and I think the likelihood of somebody wanting/needing to wrap in a throwaway array literal is actually significantly higher. + +RPR: CM says that yield versus yield star seems like the definitive precedent. + +WH: I first have a question about this slide. Why is the `Nothing` line here? `Nothing` is not used anywhere. + +MF: Sorry, it’s just a common implementation of Maybe. Are you familiar with, like, an Option type with Some and None? + +WH: You define `Nothing` on the second line, but then you don’t use `Nothing` anywhere. + +MF: It was illustrative of a library that defines Maybe that would have these things. It is not used elsewhere in the slide. + +WH: I agree that this thing should throw on non-iterables. On the question of eager versus non-eager, I have a slight preference for throwing when you get to it. But it’s a very weak preference. + +RPR: Thank you, WH. It’s a weak preference of 2 over 1. NRO prefers throwing. + +NRO: Yeah, I—I think we should try here, because, like, it’s very likely—well, it’s definitely possible the user will make mistake by passing a non-traveler, and it’s based on what they want if we wrap it in array brackets. I prefer throwing eagerly, just because that thing will throw eventually, and it’s better to throw consistently rather than just depending on how exactly to use iterator. So that maybe at different times us you have different values and maybe it throws different times in the test. + +RPR: Thank you. YSV. + +YSV: Yeah, I’m also preferring the eager throw for reasons that Nicolo just mentioned, and also then we don’t have to produce an iterator. And since it’s an error, if necessary, we could discuss potentially changing that later, but I do prefer the current spec as is. + +RPR: Thank you. RBN. + +RBN: So I posted this also in the Matrix, but if you imagine iterator concat as essentially being flat map over the arguments passing in an identity function, flat map throws for non-itersables eagerly, so I believe we should be consistent with how iterator works in the regard. + +MF: To clarify, RBN, would that be option 2, then? + +RBN: I would probably go with option 1 to be immediate. + +MM: Hello. Okay. Yes, much better if I turn it on. I think we’re all arriving at the eager throw as the dominant position. But I just want to raise this because it’s something to keep in mind in general for many proposals, and for such software engineering, is the eager throw will have a much better stack trace for helping the programmer figure out what the problem is. That’s not a trivial issue and that’s something that we—I don’t know that we’ve ever paid good enough attention to and we should start. + +MF: Yeah, this was the point made by KG to me, which convinced me to make that change in the first place. + +RPR: All right. We’ve got through the queue that almost exclusively has focused on this one issue. + +KG: Sorry, I didn’t have time to get on the queue. I want to make really clear: because RBN mentioned the thing about flatMap, Flat map, because it’s on iterator prototype method and because it’s flatMap, it expects the mapper function to return iterators. This function as currently proposed only accepts iterables, just like iterator.zip and basically everything else in the language. It does not expect you to pass it iterators, it expects you to pass it iterables. It produces an iterator, but it expects to be passed iterables. I just want to make very sure everyone is very clear on the proposed behavior. + +MF: And, KG, flat map takes iterators or iterables. The mapper produces iterators or iterables. + +KG: Oh, yeah, that’s right. We went back and forth on that. Anyway this one only takes iterables. + +RPR: MF, do you want to propose the next step? + +MF: Yeah, sounds like pretty strong preference for 1, which is the current spec text. I would like to ask for Stage 2.7 for iterator sequencing, as written and presented. + +RPR: All right. Any objections to Stage 2.7 with this unmodified proposal? Specifically with option 1. There is support on the queue from MM, WH, YSV, ACE, and a clarifying question there JHD. + +JHD: So is nobody else concerned that it’s named “concat” but it doesn’t do what array concat do, which is take a collection or an item and produce a collection? If literally no one else but me is concerned about this and thinks it will just be fine because the error message is helpful, then I won’t object to advancement. I just wanted to ask the question explicitly. [silence] Okay, thank you. + +RPR: Okay, so, yes, with—so with that conclusion that there is no one else is asking for renaming, I think I would say congratulations, MF, you have consensus for Stage 2.7. + +### Speaker's Summary of Key Points + +MF presented iterator sequencing for Stage 2.7. We had discussed options relating to the last point of contention, which was how non-iterable parameters are handled. We decided to not make a change to the spec text as presented. And have consensus on 2.7. + +### Conclusion + +- Proposal achieved Stage 2.7 +- Non-iterable arguments are rejected eagerly +- Name: `Iterator.concat`. + +## RegExp Modifiers for Stage 4 + +Presenter: Ron Buckton (RBN) + +- [proposal](https://github.com/tc39/proposal-regexp-modifiers) +- [slides](https://1drv.ms/p/s!AjgWTO11Fk-Tkr0bZS4Dw79XSH9V-w?e=PIuioJ) + +RBN: Yeah. So I’m RBN from Microsoft. I work on the TypeScript team. The RegExp modifiers proposal is one that we’ve been working on for some time as a very brief overview. It allows you to change a subset of the currently active regular expression flags within a subexpression such as enabling in your case, multiline or single line mode specifically for a subexpression. There’s currently no conflict with consisting regular expression syntax in any mode. This is new syntax that has prior art in other languages such as Perl, PCRE, RegExp, et cetera. There are a number of flags that are not supported because they have more intrinsic effects within a regular expression itself such as the global and sticky flags. So right now, this is currently limited only to the I M and S flags. Some recent updates, the ECMA262 PR for this proposal was approved by KG. I have been waiting for the final approval from MF because of some feedback he provided that has been updated. There’s a test262 PR which has been merged for this proposal. And there are now two implementation. This has been shipping in Chrome 125 and Edge 125 for I think two, three months now. And it’s shipping in SpiderMonkey, it shipped in Firefox, 130 behind a flag and I believe is now shipping in Firefox 132 unflagged in nightlies and should be shipping in the main channel soon. + +RBN: Now, I originally intended to ask for stage 4 and I still do. I just have one open question that I would like to resolve. There is a—there was a comment from a Babel maintainer regarding the current set of semantics, the early error that we applied when you have the syntax that allows you to have modifiers you add and modifiers that you remove. Currently we only early error when both the first and second mod fire sets are empty. But it allows you to write patterns like IM minus, which is essentially the same as saying IM since minus has no effect. This could be confusing and they suggested we instead error when the second modifier says it’s empty, which would cover both cases. This is a kind of late edition and late change since this is already shipping it’s more of a pedantic change to the early error rules. It doesn’t—if we allow this, it doesn’t really impact what the regular expression does. This would be more to try to catch a developer intention error, so the question is whether or not we actually would be interested in trying to take a change for this at this late stage. My impression from speaking with folks on the V8 team is that there’s little interest for this change, but I wanted to brick it to committee in case we had other opinions to discuss before to talking about stage advancement. So I would appreciate if anyone has any comments on the queue or any questions or concerns related to this change, if they feel that this would be necessary and would support it being proposed as for needs consensus. This is a late addition, so we could also consider potentially coming back with this as a needs consensus PR separately from the stage advancement or postpone stage advancement if that’s necessary. + +RPR: Okay, so let’s just do a quick check for that, because given there was added late to the agenda, it’s very reasonable to object to advancements just on the time grounds. + +RBN: And I’m not—if SYG is present, I know he had some feedback on this as well. + +RPR: Okay, so no one on the queue. Please go ahead, SYG. + +SYG: The feedback is given that we have shipped for three months, since May, I think, 125 was May, barring a pretty compelling use case to make the behavior change for the ship proposal, I’d rather that we lean towards doing nothing. That said, this is not, like—this is just a practical point that I’m making, that we made a decision already, for whatever reason, and it has shipped, and I would prefer barring a really compelling reason to just not change it last minute. + +NRO: Yeah, I don’t have strong opinion in favor or against this. If we do not throw, I’m quite confident that the interest would allow, this because RegExp already complex, so common linkers like RegExp, we have to make sure there is at least confusion possible in really different expressions. So given this, like, not necessarily and go over the same thing, I’m leaning towards not allow it. + +RBN: I think in a way, I think it’s perfectly fine to disallow this. I will say that in the languages I’ve tested this, most languages don’t even error for the cases we do error for. We chose to be more pedantic than others, but someone coming from another language that has used regular expressions like this might not bat an eyelash at the complexity because it’s not really any different. I think this is perfectly fine them to care about if we decide not to go forward with this. + +RPR: The queue is empty. + +RBN: So I suppose the best way to go about this would be to just ask for consensus and for those—and I expect it would not—or it would be blocked or potentially the best way forward is just to withdraw this. Since it doesn’t seem to be a strong interest in this I think I would probably be best to withdraw this PR. + +RPR: You are asking for consensus on this needs consensus PR now? + +RBN: I am actually stating that based on the feedback, it sounds like the best action is to withdraw this PR. + +RPR: Okay. + +RBN: So that leads to the final slide of this. Which is given the—given this again, this proposal is currently in Stage 3, has full specification text, has two shipping implementations and merged tests, assuming no other concerns, I’m seeking advancement to Stage 4. + +RPR: Okay, we have support from KG on advancing as is. Likewise, WH supports Stage 4. Any objections to Stage 4 without the needs consensus change? [silence] There are no objections, so congratulations, you have Stage 4. + +### Speaker's Summary of Key Points + +- ECMA-262 PR approved by Kevin Gibbons (Editor) +- TEST-262 PR approved and merged +- Two shipping implementations: Chrome 125, FireFox 130/132b +- Proposed Normative PR to make small change to Early Error rule, implementers expressed preference to leave as is. PR was withdrawn +- Proposed for Stage 4 + +### Conclusion + +- Stage 4 without the proposed normative change + +## Import Attributes and JSON Modules for Stage 4 + +Presenter: Nicolò Ribaudo(NRO) + +- [proposal 1](https://github.com/tc39/proposal-import-attributes) +- [proposal 2](https://github.com/tc39/proposal-json-modules) +- no slides were presented + +NRO: Okay. So I’m asking for Stage 4 consensus for two proposals at the same time. One of them is import attributes, so it’s syntax that allows passing in from parameters to the host model for declarations and parts. This is used to use the `assert` keyword. It was changed to `with`. Chrome successfully unshipped `assert`, I think, around June, and there have been no problems with that. Node also successfully unshipped `assert`. And then we have the modest proposal that actually defines, like, JSON types. It requires that when an import types JSON, if an import succeeds, it must go to the JSON parsing algorithm. Both proposals have been shipped in Safari and in Chrome for a while, so I would like to ask for Stage 4 for both of them. + +RPR: All right, so we’re asking for Stage 4 for both import attributes and JSON modules. We have support from CM. And YSV. Any objections? No? So I’d say congratulations, NRO, you have Stage 4. + +NRO: Yay, thank you. This was probably the quickest Stage 4 ever. Two minutes for two proposals. + +### Conclusion + +- Stage 4 for JSON Modules and Import Attributes + +## Iterator Helpers for Stage 4 + +Presenter: Michael Ficarra (MF) + +- [proposal](https://github.com/tc39/proposal-iterator-helpers) +- [slides](https://docs.google.com/presentation/d/1z9kpu_CITxmNls7aOn61uswOK1SqHqijngtMcDTo8zk) + +MF: All right. I’m going to go quick. Iterator helpers for Stage 4. I have a pull request, 3395. You can check it out. Summary of the proposal, there’s the Iterator constructor, the Iterator prototype already existed and now it’s available via Iterator.prototype. We have those two weird setters, constructor and `Symbole.toStringTag` that are funny accessors that we had to do for web compatibility. We have `map`, `filter`, `flatMap` and `take` and `drop` as transforming functions on the prototype, `reduce` `forEach` `some` `every` `find` and `toArray` as consuming functions on Iterator.prototype. We have a lot of tests. It was a struggle. It is implemented in V8. We had to unship and reship, and it recently shipped in SpiderMonkey. And JSC appears to be done and shipping soon as far as I’m aware. That implementation only completed recently as well. And we've had a LibJS implementation for a while now and we have a polyfill implementation as well. But lots of implementation. Stage 4 for iterator helpers. + +RPR: MM has a one-liner. “Thanks for the attention to security.” And MM supports Stage 4. As does YSV. As does NRO. As does DLM. LGH has support and excitement, JHD hates the web compatibility hacks but supports Stage 4. CM supports and RKG supports with a “whee”, which I think is very positive. A support from OMT with a “yee-haw”. Okay, I’m—so I’m going to say congratulations, MF, you have Stage 4. + +### Speaker's Summary of Key Points + +- Iterator helpers has been implemented and is now Stage 4. + +### Conclusion + +- Stage 4. + +## Source Phase Imports + +Presenter: Guy Bedford (GB) + +- [proposal](https://github.com/tc39/proposal-source-phase-imports) +- [PR](https://github.com/tc39/proposal-source-phase-imports/pull/65) +- [slides](https://docs.google.com/presentation/d/1o-b8xJ7DzVUnLYJ7pMjVJwyBWsLxW-MLQjJ6pV9q1-w/edit?usp=sharing) + +GB: This is hopefully going to be a really short item, and I think we will go hopefully under on the next item as well, so we should be able to give you all some time back. The first issue is this Stage 3 PR update to the source phase imports proposal. So to just give a very brief update on where we are on source phase imports, we finally landed the HTML PR for the WebAssembly web integration into HTML last month based on the Stage 3 source phase imports proposal. And the V8 implementation has a CL up for the last piece of dynamic import support, but I believe they may be pending feedback on us on this error case, so it would be great to be able to clarify this PR. We are also starting to consider the process for moving to Stage 4. In particular, rebasing to import and moving that forward will unlock a lot of progress on the overall module effort I believe as well. + +GB: So to summarize the issue, the question is what to do for a source phase import of a module that does not support a source phase representation, for example, JSON doesn’t currently have a source phase representation. And the source phase import proposal itself specified this originally way back as a reference error. And then in the WebAssembly ESM integration, we were slightly more informed at the time and specified it as a SyntaxError if you—for browsers that don’t support either phase 4 Wasm ESM integration—but we need to agree on it is, because we can’t have one type of error for WebAssembly and another type of error when it’s an ECMAScript source phase fallback. So as I mentioned, the reference error was originally justified as being a reference, but the syntax error case is just an early error. It’s a link time error and hence failing under the criteria for early errors. + +GB: We have PR 65 on the source phase imports proposal which changes this default error from being a reference error to a syntax error to unify on the early error model, but we require committee consensus to make this change under Stage 3. I wanted to open this up to discussion. There may be some aspects of linking that could be considered a gray area, but at the same time, having a consistent early error semantic would be good and we do need to get consensus on this to move forward. So if anyone has any thoughts, it would be great to move to the queue. + +JWK: I support syntax error because as I can recall, all the errors before any code is evaluated this is a syntax error (early error), like, a const declaration with no initializer. So I think this one should also follow it. + +NRO: Yeah, and specifically when it comes to imports, try to import a binding that doesn’t exist for a module is already syntax error. And, like, sources are kind of like a special export of modules, and so it makes sense to use syntax error here. + +USA: At the end of the queue, we have a comment from Ashley who says the slides link is required for the agenda, so I’d really appreciate if you could put that in. And, yeah, that’s it. The whole queue. + +DE: +1 (via queue) + +### Speaker's Summary of Key Points + +- For a missing phase implementation, the WebAssembly ESM Integration HTML specification specifies that a SyntaxError is thrown when initializing the module bindings +- In the Source Phase Imports proposal itself, we specify that for any module with a missing source phase, a ReferenceError is thrown when initializing the module bindings +- To align these early errors we are seeking to land PR 65 to default to the SyntaxError form in the Source Phase Imports proposal + +### Conclusion + +- Consensus for changing to the syntax error early error form ([PR](https://github.com/tc39/proposal-source-phase-imports/pull/65)), and we’ll update the proposal accordingly. + +## ESM Phase Imports + +Presenter: Guy Bedford (GB) + +- [proposal](https://github.com/tc39/proposal-esm-phase-imports) +- [slides](https://docs.google.com/presentation/d/1HF4COMfypVzftilhOIlGxz9Fn1iaiFdwYDZwzqQt1gs/edit?usp=sharing) + +GB: Just to give a status update, as I mentioned in the last item, source phase imports, having an implementation would be a huge step and a huge step to see ESM phase imports move forward to give the recap on where we are on ESM phase imports. It’s the extension of the module source that the source phase introduce to JavaScript modules, so it’s a JavaScript source phase, which forms the primitive for JS search initialization. But since this is kind of an abstract concept, we seek to motivate this proposal on the worker construction use case, so it’s Stage 2 proposal currently motivated on this kind of worker ugly syntax situation we have today where it suffers from many of the problems that we had number WebAssembly modules we were solving with the source phase and the idea being you could import a module in its source phase and instantiate a worker directly from it. So potentially even with being able to remove the second argument, because you would know that it is an EMCAScript project module. The HTML integration is of course a really major part of this work, and I was really hoping when we put it on the agenda we would have an update for this meeting, but unfortunately things got pushed back from TPAC and I’m currently scheduled to discuss this in the regular whatnot meetings coming up on this Thursday. And if anyone else would like to attend that, please let me know. But it’s obviously a critical piece of this proposal in proving out that motivating use case. The harmony layering objective is to create the representative of module exploration to module exploration, and when we originally presented this for Stage 2, there were still some outstanding design questions that we wanted to explore as part of the Stage 2 process, which I can give a brief update on where we are today. So to give a recap on what we want to be able to support, we want to be able to support module expressions, which are object values that can be passed into dynamic import, and they also support transfer fire structured clone, so this is the example from the module expressions proposal where you can take one of these module objects and pass it into post-message. So to support dynamic import, we have—we still – + +(slides are re-shared, GB backs up a few slides) + +GB: Okay. Oh, dear, should I go through these again? I guess, yeah, it might help to just go through the important ones if we missed the slides. So the worker case is this syntax being replaced with a module source syntax. And potentially even removing the separate type argument, so it’s doing a source import and passing that directly into the new worker constructor. And the module expressions examples are exactly from the repos. So, yeah, we didn’t miss too much. Okay. So dynamic import remains supported for syntax that we presented previously. If you pass a source into an iframe and import it, it won’t be supported. But the big thing we’ve been working on is the behaviors of module source transfer. So the base scenario is relatively straightforward for this, which is that when you transfer a module source, you just create a new object that has the same underlying source, and host-defined meta data so the URL and also, you know, anything else to do with the identity of that module. But there are some edge cases that come into play when considering identity under import, and in particular, what goes in the canonical registry for a given ID. So this is an example of the kind of identity cases that we have been discussing in quite a bunch of detail recently. In this case, we start on agent 1 and you have a foo.js module that initially contains the source `export let a = 1`. We import that source and store it in a source object, but then we change the source and we post the source that we imported across to agent 2. Now, on agent 2, we receive this source from agent 1, which contained the export A equals 1, but now if we do an +`import.source` of that same `foo` in the other agent, there is the question of are we going to get the exact same source or are we going to get the update because we didn’t load that source before in this module registry, because both agents have different module registries. And when we look at these two namespaces, are they going to be unique identities, are they going to have the same identity, and how are we going to deal with that? And so there is this kind of question of when you transfer a source, does it exist as the canonical source in the module registry for that ID or not? And there are a few ways to deal with it, each of these four bullets represent one of the alternatives. And there’s the link to this guest when I share the slides that also goes into more detail on this. But there’s ideas of maybe the first person to import a given defines the module at that ID so if it changes, then tough luck, and if you transfer an ID, it defines into registry and that’s what you get. That transfer itself updates the registry overwriting everything else, so instead of just first wins and then you just consolidate, actually, when you do the transfer, it overwrites what is in the registry and maybe you get some kind of orphan imports or something like that. Throw imports entirely to avoid this mess or even something more advanced like agent synchronization where all the agents share their knowledge of some kind of canonical synced ID source registry. A current conclusion in discussing this quite comprehensively at the moment are that the most important property when you perform an import source is you get the source that you provide to the import, so it would be kind of terrible if you provided a source whose underlying source text was one thing, but you get back an import that is something else. And then the other conclusion is that the last item of synchronizing agents and having some really fancy design here is really inhibitive to implementing, and so all these other cases of throwing, overwriting and providing what is not expected is likely not viable, and that kind of leaves us in the space where transferred sources should not inject into the canonical registry under their IDs, and instead behave more like module expressions where they represent always a unique ID. But if you transfer the same source multiple times into the same worker, it can still retain the same underlying unique identity even if each of those objects has a different object identity, so there’s identity under import and identity of the object of the module source itself. What this means for this example is that when you transfer the one source that has the old A equals 1, and you import it, you get the old A equals 1, and when you try and import that same source and you’re going to then directly hit the network from your local agent again, because you don’t see it as being then as the source that you previously had. And so the namespaces are not equal, the instances are not equal. It does mean that we have this instance inequality for these cases of transferred modules. That this is distinct from worker invocation, and we get the expected source of equality. So, yeah, that’s the transfer problem, and what we’ve got on it so far. And then finally, there’s this kind of secondary use case around module sources for JavaScript that we can extend them with module source analysis data, so the—having it have this close relation to the source text means we can provide tooling for import analysis as well as understanding exports. And so one of the updates we made to this recently was we previously just supported discovering all of the export names and star export names. We’ve now moved to a comprehensive fine-grained exports analysis so that you can actually track export binding uniqueness since that was discovered to be something that people did want to obtain. On the export, it’s now under this type property discriminants, which gives all of the full parity with the underlying binding information on exports. Where a given direct export can have multiple aliases and represents a single slot or a single binding slot and export can be traced through that so that you can directly relate exports to their bindings. One result of this is that there is some duplication between imports and exports, the specifier is duplicated in the reexport case and the reexport all case. But since they are distinct use cases for using imports and exports, it doesn’t seem too bad of that. And, again, we have the metadata options we have. So those are the main two updates. The spec text remains mostly complete down to defines the source phase object, and the analysis now with the fine grained analysis behavior. The transfer behaviors are pretty much what we’re now looking to settle on and start to specify the HTML integration being the big piece with upcoming discussions. But if and when we have determined that we’re able to proceed with the HTTP integration, that’s when we’ll be looking to bring this back to committee for Stage 2.7. So that’s the update. I’m happy to have as much or as little discussion as folks would like. + +USA: There’s nothing on the queue yet. Maybe let’s give it a minute or so. Let me check the chat. There is a question for you on the chat, but I believe you can answer that async. + +GB: Yeah, sure, then I’m happy to conclude it there, and any further thoughts, folks are welcome to join the module harmony meetings, and thanks. + +USA: Wait a minute. Sorry, there is now a topic on the queue. + +YSV: I was asked to bring the topic that I had in the chat onto the queue. This can be discussed async because it’s not directly reto your update. But a question we had related to this proposal again is why there new methods added to this module source for the proposal and what’s the use case for browsers. And we couldn’t understand how that would work. Nicolo said that in terms of making static analysis work, you needed them but this is more useful for node, and the concern we have from Mozilla side is it makes the justification for landing this in browsers quite weak. + +GB: Yeah, so I can certainly speak to that. And just when I thought I had gotten off easy. So the analysis is also necessary for loader construction. And when we get to compartment loaders, you do want these analysis functions, even in browsers if you’re constructing virtual environments. We added it here as a kind of a nice to have, and the way it’s always been presented here has been as a nice to have, that we could always remove if anyone has any problem with it. I think it’s a well-defined space. I think the methods have, you know, clear utility in various scenarios. And it does make the virtualization discussion easier when we get there in already having this worked out so that virtualization discussions can focus on virtualization discussions and not analysis problems. But if there is concern, we can remove it from this request. + +YSV: We can take this offline and discuss it further. And we have concerns, and if we get into the discussion of virtualization, even if it’s static analysis, probably we should link that to core motivation of virtualization rather than yes, sirly this static analysis problem around workers. + +GB: Yeah, I mean, there is basically—we talked about a library ES module lexer, which is used for, you know, commonly for analyzing graphs, and the use case of analyzing ESM files and graphs is a very common one. And solving it at the same time is creating this primitive. Adds utility to this proposal. And removes the need for an ecosystem library that has shown very wide uptake. But as I say, it’s not something that we need to, you know—that has to absolutely go with this proposal. It was more that it’s nice to solve it, but, yeah, that’s -- let’s continue those discussions. I’ll create an issue on the repo and it will be great to follow up on that. + +USA: All right, so that was the entirety of the queue. Would you like to summarize or make any concluding remarks? + +### Speaker's Summary of Key Points + +- We gave an update on the proposal, including the proposed semantics for transfer of module sources +- We’ll be hoping to give a future update once we have feedback from HTML on stage progression +- Some committee members asked about the motivation of the reflection features of JavaScript source modules. These are originally motivated by build-time tools, rather than usage in browsers, so they might not be needed in all environments. The champions will follow up with the analysis motivations separately. + +## Structs and Shared Structs for Stage 2 + +Presenter: Shu-yu Guo (SYG) + +- [proposal](https://github.com/tc39/proposal-structs) +- [slides](https://docs.google.com/presentation/d/1_LOkmaeM0S-VOvhJ28eeDJUmkMeHVyltxSt1lGLF5iA/) +- [spec](https://tc39.es/proposal-structs/) + +SYG: This is a presentation for structs and shared structs for Stage 2. This presentation is laid out by logical feature. This proposal includes a bunch of different features, and it’s presented here feature by feature. With the goal to add fixed layout objects to JavaScript. The biggest motivation is to enable shared memory multithreading, which is a new capability for performance hungry power apps. So, this is not a hypothetical thing. There are real world big applications that are interested. To list a few, spreadsheet application, IDEs, TypeScript compiler. And to be perfectly clear, I am talking about a demand here for shared memory multithreading not strictly via JavaScript but also by WebAssembly. It will be a capability that JavaScript gets, even if we do not reflect that capability directly in JS. + +SYG: So it’s important for JavaScript to also reflect capabilities that are coming through from the Wasm side. As a pure motivation, the opportunity cost to not do shared memory multithreading is high, as a matter of competition with other platforms. As a proponent of the web platform, developers are often faced with a choice, and should they target the web platform or a mobile platform. Capabilities are easy to come by on the mobile platform relatively speaking because there’s—there’re fewer concerns like interop. I don’t want the web to miss this opportunity here, especially there are—as the web applications get more and more sophisticated, that it’s important that we rise to meet this need. As a side benefit, it also makes sense to add on shared structs and unshared structs will show also have a nice—some nice benefits for additional guarantees and performance, even if you don’t care about shared multi memory multi threading, + +SYG: so structs, this is the unshared structs. At a high level, unshared structs, you can think of them as restricted classes that trade some of the expressivity you get with classes for more guarantees and more run time performance. MM used the term'Defensible classes' and this along the same lines as that. The high level semantics are the struct instances are constructed sealed. They are otherwise look and behave like ordinary objects and they are sealed, which one of the integrity levels if you’re unfamiliar with the integrity levels in the spec. This basically means that their layout, their own properties are fixed at construction time and cannot be removed. New ones cannot be added. So with—in other words, these are closed instances. By default JavaScript objects are open and you can just keep adding to them. There’s something we are calling one-shot initialization, which means when you construct an instance, it will be constructed entirely in one shot. It gets all its own properties that are declared in the struct. There is no return override trick you could do via the constructor, which is something you can do with classes and ES5 style classes, and there’s no half initialized instances. So if you have a field initializer that throws, the field is still there. It’s just undefined. The inheritance hierarchy as consequence of one shot initialization, the inheritance hirearchy must all be structs. The methods on struct are non-generic. This means that you cannot do ad hoc calls on them with an arbitrary receiver and have that work. There is an implied—there’s an implicit brand check that the receiver must either be an instance of the struct or a subclass. So this is good for performance in the VMs meaning that there is one shape check on the receiver at entry instead of every owned property access. For generic methods, when we compile them, you know, this could be anything. So the way it’s compiled is pretty much generic. But when you access on owned property that goes through a normal owned property lookup and try speed that up, we use inline caches and things like that. For structs, if we design them to be non-generic, we can combine all the checks for owned properties into one entry, which is good for code size and performance and good for predictable performance. As a caveat here, the prototype property, so the methods, they can still be polymorphic. The shape check things really only applies to own properties and not necessarily prototype methods. Otherwise the idea is that these structs are usable just like classes. + +SYG: So by example, I have a Point2D, has a field X and Y, has some method distance2D and the constructor just assigns to X and Y. It’s okay to reassign—it’s okay to write to fields that already declared right here. But it is not extensible. You can’t do without Z because it’s not declared, and this instance is sealed. You can’t set the prototype of the instance because you can’t to that on sealed objects either. + +SYG: There’s one shot initialization. So it’s okay to extend structs with other structs, so if you want to add a third dimension to your point, you can do that. This for a snippet here is to show that by the time the constructor execute, you already have a fully initialized in the VM sense, you already have an instance of the struct object, the struct instance with all of its own declared properties as own fields. Sorry, as own properties even if the initializers have run already. I’m also proposing there is no this-TDZ because it’s already initialized. You can call super, but this is no longer required to be—because there’s no this TDZ, the super can occur later, and just runs the super constructor on the ‘this’ that’s already initialized. This our understanding of somewhat controversial. Ron as the co-champion feels differently about this as well. This is I think a point that might be debated during Stage 2 on whether we want to require super to be called first if it is called. But I just want to call that out as—the main thing here for me is not really super, but that the ‘this’ is already constructed with the fields. And return overrides does nothing. If you want to do a return override, it doesn’t work. The return value is just ignored by the struct constructors. And that’s one shot initialization. Here is an example to show that the methods are non-generic. If I make two Point2D objects it’s okay to call distance with the—with the actual point 2D instance as the value. Subclass is okay. Okay, this example is completely wrong. I think what I meant here, like if I had a 3D point, anyway, this part is just completely wrong. But please imagine that there is a Point3D instance somewhere that is a subclass of point 3d. It’s player if you to call a Point2D method with a Point3D receiver. But you can’t call—you can’t call it with a object that happens to have the same fields with this value. Like, this is not allowed. You can’t do that. There’s an implicit brand check when you create a brand instance that it’s branded a point 2D or 3D class—or a struct, rather. Otherwise, class features are allowed. So this is the—this is the unshared struct. If you want to use getters, setters, static methods, privates, that’s all fine. + +SYG: The finer points of unshared structs. Each struct instance has a list of brands in the structs brands list. Internal field for subclasses. And this brands list is what is checked upon entry to a method. Struct methods are, therefore, specced as exotic callables in the brand checks. Struct prototypes are also sealed. Struct `constructor.prototype` is not writable. You don’t want people to willy-nilly overwrite the .prototype on the constructor. Because the idea here is we’re trading some expressivity with guarantees. Struct constructors are currently not sealed. Maybe they should be, happy to debate this more during Stage 2 as well. And this last point is also that struct inheritance hierarchies are currently allowed to top out at object. Object is not a struct. It exists. We can’t really change it to be a struct. So—but at the same time, objects don’t have any fields, like the whole point to top out an object is if you want convenience methods and object.prototype, they are there on struct instances. That seems more use to feel me than to require that struct hierarchy to top out at null or something. But I can also be swayed one way or another on this point. I don’t think it’s too important. It seems better to have the convenience. But this is, like, the one exception to the entire struct hierarchy must be structs, is if the very top of your hierarchy is object, that’s okay. + +SYG: All right, that was unshared structs. So we’re going to build on that and to further restrict it, for shared structs, to show how we can further restrict it to make these things amenable to can be used for shared memory multithreading. The shared structs at the high level, you can think of them as restrictive structs that trade even more expressivities to be shared across and be simultaneously accessible by multiple agents. So they have all the restrictions and properties of structs, they are actually shared across agents, meaning they’ve not copied. Their own data properties are shared. They can only reference primitives and other shared structs. So you can’t put normal, ordinary objects into a field—into an own data property of a shared struct because that would be not thread safe. And this—at this time, they have a null prototype. There is a big open question to be further iterated on and decided before 2.7 on whether they ought to have something called a realm local prototype. The idea there, I’ll go not more, but currently they have a null prototype pending the open design question. And we’ve been in many discussions with MM and company who have a lot of insights there on how that to be done. The second big open question about there is the last bullet point is whether these should be usable as keys in weak collections. In particular, WeakMaps. Now, it is the case today that all objects are usable as keys in WeakMaps. These are special because if—WeakMaps recall are basically like properties. So if you have an object as a key in a WeakMap, it’s like adding a property to that object from the point of view of the garbage collector. So if you allow a shared struct to be usable as keys in WeakMaps, that could open the door to cross heap cycles, not from a language safety point of view, because your WeakMaps are not shared, but from a GC reachability point of view. And the GC reachability point of view is very, hard to implement. This question is being heavily debated on the Wasm side currently because of the GC implications, and I think it’s very important, it’s a hard constraint, that we follow whatever comes of that conversation. So for now, the spec draft for—I’m asking Stage 2 for does not say anything special about disallowing them as WeakMaps. But I want to explicitly call out here that if Wasm decides that they should not be usable as keys for WeakMaps for the GC difficulty reasons, that we disallow use of these. We throw if they are used as keys a as WeakMaps so that we can relax it in the future instead of allowing them now and changing the semantics to be something weird. + +SYG: MM has his hand raised. + +USA: Is that in relation to your clarifying question, MM? + +MM: Yeah, it’s exactly the clarifying question. Given what Shu is describing, it seems like a great timing for the clarifying question. There’s another hidden WeakMap in JavaScript which is using return override by regular classes to add a hidden properties, hidden fields to other objects. So the question is, you know, you establish these things can’t do return override, but if someone else does a return override returning one of these struct, can they add a hidden field to it, and if so, that’s—then that basically is a—is logic Cal equivalent to a WeakMap hidden? + +SYG: They cannot. These shared structs are sealed from the get-go. There’s no state where user EMCAScript code sees a shared struct and add things to it, including private names. + +MM: This is distinct from sealed? There’s one case where we prohibit the return override, which is the browser global, but that does establish a precedent, that there is an object that you can’t add properties to in return override, so these would fall in that precedent? + +SYG: Yeah, so I agree that we should—we also—if we decide by WasmGC side that these ought not to be usable as WeakMap keys, we must also ensure that we do not accidentally reveal this capability through some hidden field. Those are the two big pending questions for shared structs, but the rest I hope are more or less settled. + +SYG: So to give some examples, there can be no shared-to-unshared references with shared structs, because that is just straight up unsafe. Supposed I have my Point2D, but it’s shareable. I make this MicrosoftSharePoint and I can assign things to it. Here I can assign 42. Ignore this unsafe thing for now, I’m going to explain the unsafe block coming up. You assign P.X pause it’s declared can it’s a primitive. All JavaScript primitives are immutable, so they are from a language safety on it of view unproblematic to be shared. You can assign it to itself because itself is a shared struct, so that’s fine be. You cannot assign, like, an object literal, because that’s not shared. I am proposing to act—to add the ability to programmatically test if a value can be shared. If the `call.reflect` can be shared, and you can test whether it’s shared or not. Shared structs are actually shared, so if I make—so assume P as before, so this is Microsoft SharePoint is the P. I make a worker, I post the P, I assign main to the one—to 2.X, and then I log it. On the other side, I receive P, I assign worker to X and then I log it. This code can log any of the following interleave its, because these things are actually shared. + +SYG: This is to show, again, the point that this is actually shared in that if you have an Echo worker that just post messaged the same thing back to the sender, you get the same object identity. This is not a copy. A core Larry to be added with—along with shared structs with a careful case of shared instructor, called shared arrays, which are basically shared fixed length arrays. This is not growable or shrinkable. It’s a fixed length at construction. Instances cannot be resized, dot length is a frozen owned property and like shared structs, because it’s a special case of shared structs, it also has a null prototype and this is also pending the same realm local prototype design question, if we can work that out, that means we can add methods here that will be helpful. If not, then our hands are just tied that this has a null prototype. So shared awry, by example here, it’s basically the same worker main thing I just showed, except instead of using dot examine, it uses a shared array and assigns the zero element. The reason we’re adding shared arrays is that you really do want index collections as a building block so you can build more sophisticated shared data structuring otherwise who is going to make a shared struct and manually declare index like zero to 100 or something. This is a shorthand for that. API editions, I’m not going to go too much into detail here. You can read the spec draft proposing to overload some atomics methods. So that you could use sequentially consistent access. So currently the atomics methods take a TypedArray as the first argument and then an index as the second, and the overloads basically now overload them to take a shared struct to be able also Titanic a shared struct, that Tess first argument and the field name for second. And as I showed before, also a product to reflect that tests if a value can be shared. The final points for shared instruct versus they are not quite sealed, like, as Mark observed earlier, these are—these have even more restrictions than sealed. One thing I do wants to call out here is that they actually cannot be frozen because the idea with this is that the layout is immutable, so that when you share them across threads, the VM in its—it uses—the VMs use the shape to decide how to design an object. If the shape is immutable, you can access them without worrying about the date being out ofize. Freezing changes the descriptor and when you assign to a property, you need to know if it’s writable or not. Freezing makes something non-writable. So in order to check the writability, you want that answer to never change. And if—you want if the answer to never change, we can’t actually freeze something. So this points out a gap in expressivity, which a fruitful thing to follow up on, if we think of structs as declarative sealed things, this points out, well, why not also have declarative frozen things, and I think if you want something to be frozen immutable, instead of freezing a programmatically as we co-for ordinary objects, they should be declared frozen but as a follow proposal. For now, it—shared instruct fors if you are more mutable shared memory. I won’t go into too much the memory model arcana. It is written up in the spec text. + +SYG: The executive summary is basically that bare access is unordered And accesses by atomics are sequentially consistent. And no tearing allowed, meaning you can’t see half of one write for any shared struct field access. On the implementation, that basically means that all fields need to be pointer aligned. There’s an implicit allocation as publication in there. That says you cannot observe a partially initialized shared structs across threads. There’s a spec section that uses a critical section—it should not be implemented that way. But that’s the—that’s the underlying assumption of why that spec fiction critical section exists. I’m happy to answer more offline. This a bit involved. But the guarantee that I want to get across is that allocation is publication. Meaning that once you have a new shared struct, there is no other thread under any circumstances that should be able to see that shared struct and see it in a half initialized state with some fields missing and crashes. That must not be allowed by implementations. So to recap the open questions here, open question one is per realm prototypes. Why might we want per realm prototypes? Was a they allow us to attach methods the shared structs. If we believe that, you know, after object orientation has its benefits, it’s very ergonomic to be able attach methods to structs like you can attach methods to class, but methods and functions are deeply unsharable things. They close over the realm, you can make closures, they’re very much shareable. So the way we were hoping to bridge that if you have a per realm prototype, it’s a plain object. And as an ordinary object, it can have ordinary things including methods, and methods would allow better encapsulation of thread safe behavior because otherwise your choice is to use free functions. So, okay, so that’s why we want per realm objects. What’s this correlation part? If we have per realm prototypes, if I have a struct declaration S1 in worker W1, because the prototype objects are per realm, that means S1s prototype in W1 is a, ordinary object in S2 even if S1 and S2 are evaluated from the same source text. Imagine you imported `my-shared-struct.mjs`. They’re two different evaluations, they will have two different per realm objects but from the user intense point of view, they’re intended to be the same. So wouldn’t it be nice if we could correlate the two somehow. It’s difficult to correlate them in a way that satisfy the constraints of no ambient communication, global communication, and some other details, and we have been working through this design space with MM, MAH, and other folks in that space who are interested, DE and NRO as well offering help from the module side. So this is yet undecided. We have a direction, but this is not yet fully done. Why not? As I said, the correlation mechanism must be carefully designed to not be a global ambient communication channel. The promising direction is to use module source identity in this correlation map thing. But this must be designed to satisfy all the stakeholders or dropped to enter Stage 2.7 or 3. That’s open question one. + +SYG: Open question 2, are these usable as WeakMap keys? I think that for full composability with the rest of the language, the right semantics is just that they are usable because they’re like any other objects. But that’s a big if because it is very hard to implement this in existing garbage collection. So the next steps here is that this is basically difficult for everybody. Mozilla has expressed a lot of feedback here that is the very hard for them. And V8 is prototyping something that we hope is pragmatic that is possible to ship, but the take away I want TC39 to have is that this an important question to be decided before Stage 2.7 or 3. It’s mainly drive on the WASM side and the exact same concern for Wasm side because it’s a garbage collection thing. It’s driven by the Wasm side because there are more implementors(?) there and the immediate use cases are there, so I want that to be driven there, but we ought to follow the decision that comes out of there. If the decision is we support this kind of express I havety in the GC, then it ought to be usable in WeakMap keys. If the decision is we ought not to support it because it’s too difficult, we must throw them if we want to use them as WeakMaps. This too must be decide before 2.7 or 3. Okay, so that’s the bulleting of the main thing. That’s shared structs and structs. + +SYG: Moving on the next two points are somewhat orthogonal and intended the complement the use case of shared memory threading. The first one is something called unsafe block. And the high level, the unsafe block is motivated as follows. So everyone knows multithreaded programming is very difficult to reason about. It would be nice to have some guard rails. We’re proposing an unsafe block to be that sin tactic guard rail. The idea is you want to eyeball a program and know if it has possibly erase things in it. So by defall, barrack sayses to shared structs which are unordered in the memory model, so straight up data race. They are disallowed if they’re not syntactically lexically an unsafe block. I will show some examines becomes. But this is about lexical scoping. If you’re familiar with unsafe blocks from rust, this is not function coloring. This is straight up lexical scoping. Hosts are also kind to have an extra linguistic programmatic whole Internet. The web already has. This you might think—your application should not have any shared memory access at all. And I think that’s a reasonable policy decision to make. The web chooses this to—with a mechanism called cross origin isolation. The server must send special headers to give you shared ArrayBuffer access today. This would extend to any shared memory—any shared memory facilities including shared structs. This is extra linguistic. This is on hosts and this is not in the spec test and this is the extra expectation. Node wants something like a global function, that’s just like turn off all shared memory, that’s a one-way latch or something like that. There’s also an open question here I will get to about the safety level of APIs, because the unsafe block, what it lets us do is filter out property access request dot and bracket, but there’s a question of what do you do with library `Reflect.get` and `Reflect.set`. We’re work through that. Here is an example. You have a shared struct. So normally you can allocate them outside of unsafe context. That’s totally fine. But if you just want to write stuff it to outside of unsafe block, that throws. If you want to read it, that also throws. You must lexically nest them inside an unsafe block. Hopefully that’s clear. And this is to hammer home the points. This is lexical scoping. Even though increment function is called from an unsafe block, increment is not itself in an unsafe block, or plus plus is not in an unsafe block. Is not about function calling. This is about scoping. The final points this is expected as a Boolean parameter. That’s whether it’s unsafe or not. By default, it is not unsafe, so no existing uses of get and set are changed. And reference references track if they’re in an unsafe block. So if third open question of all of the open questions in the spec is safety of API calls. We have some choices here, so you can say—you can do the maximally permissive thing and say all API calls are unsafe, meaning something like `effect.get` always works regardless of whether `effect.get` is inside on unsafe block. You can do maximally restrictive. + +SYG: So it always throw on shared structs. This means the proxies break the unsafe scoping. The trap handler has to have an unsafe block inside. And perhaps the third way is to thread the Boolean parameter that we have on the peck side via bracket bracket get and bracket bracket set to [INAUDIBLE]. And this seems like a promising direction forward and we’re still noodling on the complex questions. Maybe we’re okay. Unclear. Is a question that must decided before entering 2.7 or 3. + +SYG: Okay, the final part of the proposal is that—so we have these shared memory struct things. You want to be able to synchronize them. This is also the most orthogonal and also the most split outable part of the proposal, and we believe that it is important to have higher level synchronization to maximize the likelihood of correctly using the shared memory things, so that end, we are proposing to provide Amy tax condition variable. And it’s not cursive and the mutex itself is shared, and O cap—in O cap language, you can think of the mutex as the lock cap and the when you unlock it, it gives an unlock cap and the this helps to—misuse like double unlockses and provides a blocking lock and timeout static methods. I put the star next to static because ideally these would be prototype methods, and barring local prototype, these are static methods. And just like atomic style wave block, it only works in [INAUDIBLE], so try to go through these quickly. I don’t think these are—I’m sure there’s more interesting discussion about this. But the idea is that you have—you can make a mutex and lock it. And it gives you back in a lock token and you can pass interlock token if you want to reuse it so that you can save yourself the allocation in hot paths. And lock if available is the time out taking version that give us back an unlock token if you can lock it and null if you cannot lock it. The unlock token has—it’s just an ordinary object. You can unlock it and dispose it, which is just unlock, and returns untwinned and you can check if the unlock token in fact unlocked. + +SYG: So by example here, we have the SharePoint make a mutex, you can use the lock thing, which gives back an unlock token which has a `[Symbol.dispose]` +with using and you can use that to protect whatever critical section you want. Hopefully that is pretty clear for folks who used to using mutexes. Lock if available is kind of like—it’s kind of like try lock or lock with a timeout depending on whether the timeout is zero. If it’s zero, it’s see if you can acquire the lock right now, and if you can’t, just tell me you can’t, and with a timeout, it’s the wait for that long and then tell me if you timed out or actually acquired it. The final points is unlock tokens can be pre-allocated reused for hot path and if timeout zero, it’s greater than zero, it’s time scout to and uses the same waiting mechanism as atomic style weight. This is probably the least important of the open questions, but it is an open question on what should the return type of the lockIfAvailable be. Currently, what I have said is, Unlock token or null, so if it could acquire the lock, it gives you back a token. If it could not, it returns null. What this means is that if you naively use the using syntax with lockIfAvailable, you don’t know without check what L is, whether you acquire the lock or not. If you wrote this, you might as a naive reader think that you entered the critical section after you come back because null can be used with the using syntax. So the open question is should we allow this? If we allow the previous example a because using the declarations were already designed to start—from the start to allow conditional acquisition of resources, so you might think that, well, we ought to athis because—you should check the value of L regardless, but you might think we shouldn’t allow the previous example because this isn’t like a resource and it’s a correctness bug if you allow this. And we could disallow it by clanging the return to something grosser, like if you returned a stripping, a string can’t be used with using it so it will just throw if something times out. But that’s something to be discussed and also to be figured out before Stage 2.7 or 3. A condition variable is just basically that, a condition variable. It works like you expect condition variables the to work. + +SYG: There is a waitIndefinitely method and wait for some amount of time method and wait for the notify method. I’ll skip this. You can read it if you want. I think it’s not that surprising. OK, almost done with the deck. I haven’t really talked too much about the theory. There is a lot of movement on the Wasm and a lot on the GC implication question that manifests on the JS side as the WeakMap question. It is a hard constraint, hard requirement of the proposal to be reflection of the Wasm GC capability. Because more is happening and the GC thing is likely to be decided on the GC side, I think Stage 3 is not Stage 2 but 2.7 and 3 of the proposal should be sequenced after the Wasm GC or after the lock step because we don’t want these to diverge at all and it would be bad for the ecosystem. So to recap opening to the queue big open design questions to be worked out during Stage 2. Biggest one is prototypes in the correlation problem. 2 is usability or map as WeakMap keys that is the GC question and 3 is the safety or the unsafety of API calls in unsafe blocks and somewhat relatively smaller one is return type of lock type available. What that to the queue. + +USA: There is quite a queue. I seen it’s whittled down to ten minutes. You have around 25 minutes at most. So if you would like to prioritize things in the queue, that could be nice. Or go down the queue as it is. + +SYG: Let’s go down the queue as it is. I did a quick look. I’m not sure how to prioritize this. + +USA: Okay. I request everybody to be brief. First we have Jack. Jack works, are you there? + +JWK: In the extends slide, what if you called super() after this access or omit the whole super()? + +SYG: Thank you. Don’t call the soup inner the spec text you don’t call the super constructor but this is constructed. If the constructor did something, you don’t call the function. Imagine like a normal function call. + +JWK: Isn’t that break some property of the constructor? + +SYG: No, because the constructor of a struct is really post construction initialization. It doesn’t do the construction. The construction is like the one shot construction. By the time you enter the constructor you already have a usable this value. + +JWK: Not the language, but expectations of the author of the class. + +SYG: Then it does. This is possibly another open question, should we be more—should we build in more guardrails here and still require super to be called in some way without incurring the problem of a this-TDZ, I think that’s a very valid concern. + +USA: We have a reply on the queue by WH. + +WH: I see a couple consistent ways of going about this. One is to insist on `super` being called. If you don’t insist on `super` being called, then you might as well also allow `super` to be called twice or such. Why not? + +USA: I think Ross up next. + +RKG: Yeah, this almost could just be read off and we don’t need to say anything further beyond the question. You had mentioned you wouldn’t be able to do the return override hack simply nothing would happen. I just wanted to clarify is there a need to silently nothing instead of having that be an error? + +SYG: I think it should be fine to make it throw too. These are syntactically distinguished constructors. I don’t think there’s a lot of cost to making it a throw. + +RKG: Just clarifying. + +SYG: Please with an issue and we can discuss the issue. + +RKG: Sure, can do that. + +USA: WH has the reply to that. + +WH: A return statement with no expression should be fine, but if you return something other than undefined, I agree it should throw. + +SYG: Sounds fine to me. + +JHD: So since they’re objects, I find it very important that all current object invariants hold, or at least as many as possible. Except for `window` that was a special security motivated case that I’m not anxious to see extended, if it’s an object, you can weakly hold it or stamp private fields on it or extend from it (altho I’m less confident about that last one). Those must I think remain true. So we’ve been talking about that a lot in matrix. It’s fine if the struct part of it has to be one shot as is in this proposal and then if you stamp private fields onto it maybe it doesn’t go to the fast path and gets slower - fine, whatever. We had this discussion in the Symbols as WeakMap keys proposal. I requested a “can be weakly held” predicate and we were told we can’t have that. One of the consequences of having it, and indicated in plenary after that decision, that we won’t be able to change the set of what can be weakly held or not because code has to make their own predicate and branch because most of those will not just be try-catching around the “weak hold” attempt. So if we make objects that are not weakly held, there are code paths now that will break if one of these finds their way in. I’m not suggesting that the stamped-on private fields or whatever be shared or violate any of the constraints here otherwise, but private fields were very explicitly designed to be essentially sugar for putting something in a WeakMap and I think that we need to retain that capability. + +USA: There’s a reply to that by Mathieu. + +MAH: Yeah, I think the fact that designed that way doesn’t mean we have to uphold this as an object in variant. I find it very unfortunate that we got in a situation where you can stamp private fields on almost every object. I don’t think this should be considered as an object in variance per se. I would like the committee as I expressed before to be able to reconsider that and weaken this ability of stamping private fields on objects. + +JHD: Right. I mean, that’s a fine belief to hold. But like it is currently an invariant and at least one of us believes it should be one and should remain one. And as far as, I mean, even if we’re not talking about weakly held or stamping private fields we can make a side channel with the memory leak putting in a regular map and preventing me from doing that as a struct or shared struct with a key in a regular Map would be bizarre. I don’t know why it’s so okay for me to make a memory leak with the side channel and not okay for me to use the actual mechanisms we have for non-memory-leaking side channels. + +MAH: It’s not a question of side channel. It’s the question of WeakMap is API that can be virtualized or denied. Stamping private fields is effectively that capability but syntactic that you cannot virtualize or deny. And this was not—the consequences of that were not fully realized when we allowed this, I think, and that’s my concern. And actually is the same concern I believe for why window—why the web host asks for an exception for the window object and as I mentioned before, I have similar motivations for other objects because I again would like to be able to do behavior like host to do which is virtualizing special objects. + +SYG: Keep it—with a long queue, I want to pre-empt some of this. This comes back to if it says object. + +JHD: It’s not about the type of value because it functions in null. If you pass it into the object constructor and get the same triple equal thing back, it’s an object for example. Or you could exhaustively do the type of checks. + +SYG: What is object construct. + +JHD: Call the object construct as a function and pass to it. And return is triple equal to the value pass in it is in object if it is not equal it is primitive and those invariants hold. + +SYG: I see. I take the constraint to be— I think you and I are agreed from the language point of view, it makes sense to allow these. But our hands are tied by software engineering reality. + +JHD: Sure. + +SYG: If we cannot realize the ideal language thing, then it sounds like we need to look for—we need to enumerate the places where to make them sufficiently different from object but object like but not actually objects so they can be used as objects but from like a reflection point of view they don’t look like objects from your point of view. Does that sound like the path forward? + +JHD: But then what you’re talking about is taking the world for decades had two kinds of things in it and adding a “secret third thing”. + +SYG: This is not a secret— + +JHD: That is a high, high cost. And against the benefits that you hope to get from the proposal. If we have this sort of untenable situation where we can’t make it a regular object and can’t make it a regular primitive, it’s not a given that we do the third thing. Just putting it out there that it would be great to solve those problems but these are some important characteristics of the language that have been reliable for a long time. + +SYG: My read of the characteristic does not—of that characteristic that you pointed out with the feedback of Mathieu and others that keeping that characteristic outweighs the opportunity cost or the thing it’s seeking to enable and leave it to the rest of the queue. + +YSV: And clarifying question JHD, cost for who? + +JHD: Every user and learner of the language. And I understand that there’s different kind of stakeholders including implementers and different costs for the group and they’re all important. I’m thinking of it in the priority of constituencies kind of way that users of the programs created are the highest and then the developers… and I’d have to pull it up to cite it; I can’t cite it from memory. Implementers are near the bottom of the list in terms of figuring out whose constraints override whose. + +YSV: In that priority of constituency, items such as philosophical concerns and specifications is lower than implementations and when we designed WeakMaps and registry, did we have beginners in mind? + +JHD: For those specific things, no, absolutely not. + +YSV: So from my perspective,tial costs we’re talking about today to implementation is extremely high? + +MM: I think we have weak references and finalization registries are definitely not for beginners. WeakMaps are within the language for beginners. + +YSV: Would you consider it to be an unclearable bar for beginners to not be able to use certain things in WeakMaps? + +MM: No, I don’t. JHD and I disagree on that. I wanted to clarify this issue about what obstructions were designed with beginners in mind and WeakMaps are definitely in the beginner + +YSV: Perfect. Thank you for clarifying that. From my perspective and talking about costs here, it’s not a very high cost to understand that shareable memory, shared objects is something special, this is really something new. We never had something like this in the language before. And in fact, multi-threaded programming is a known hard problem. This is not something that a beginner will be able to work with easily. There will be a lot of stubbed toes in this kind of work. Now, we have a very strong concern about shared to unshared edges because the kind of modification you need to do to GC that can’t do cross help garbage collection already is frankly extremely prohibitively expensive. If we decide to do something like this, this will probably prohibit us from having shared structs. I think that shared structs has value. We need to take this concern very seriously. I would say that the consistency argument for the benefit of learnability is not as strong and does not provide as much benefit given the nature of the feature we’re discussing which is already something that is very complex to work with. + +JHD: That’s a good interesting perspective. I will think about it. But I recognize that having shared memory at all, having shared primitives at all, may conflict with the goal of making the language approachable and usable for newcomers. + +SYG: Again, for reminder for the rest of the committee and also for JHD, the counter factual here is that it comes via Wasm GC as something that they design in Wasm CG that frankly don’t care about that characteristic. So I think it’s important to keep that concrete counter fracture. + +JHD: Yeah, if you’re referring to why we added WeakRefs to JavaScript because the capability was coming regardless. + +SYG: I’m saying if you care about objects, everything that looks like objects, what you consider to be an object to all have this characteristic of WeakMap keys and tank this in TC39 because of concerns and that looks like objects to you. + +JHD: That would be a violation. There’s nothing in the spec that would allow that. It’s not a JavaScript-compliant engine if it doesn’t allow all objects as a WeakMap key currently. + +MAH: Sorry. I would like to object there. The host hope that we added allows the host to stamp any object that they want with that behavior. So not for WeakMap key but private field stamping. WeakMaps can be replaced by the host so it’s just—I don’t know. I wouldn’t go as far as saying it wouldn’t be compliant. + +MM: I’m sorry, I think it would not be compliant. + +SYG: This is already called out as an open question on the usability of yes or no. And I take JHD—thank you for raising the concerns. I will take them as part of what must be addressed during the working out of that question. Please move on with the rest of the queue. + +KM: Sorry. I couldn’t unmute. My original thing about locking and I guess also locking is obviously covered. But I guess the other question open parts of it is deadlock detection part of it? + +SYG: For mutex is the deadlock if FD current. If it is the current thread it throws or otherwise blocks. Straight current calls will throw. There’s no cycle. + +KM: All right. For the whole question. Couldn’t edit in, other barriers like acquire or release barriers for storing? + +SYG: No. But I’m not close to adding them if the use case arises. + +RPR: Before we move on with the queue, could folks clarify if any of the remaining items on the queue are specifically pertaining to Stage 2 which is the important item for the agenda here? Or, you know, if there’s a chance that it could be discussed later? + +USA: Let’s keep moving in the queue instead of asking people to do that. Next we have Dan. + +DE: I’m happy for this proposal to go ahead to Stage 2. The design here is really good in terms of how it integrates with JavaScript. It’s much better than some earlier drafts and I really appreciate the attention to detail that was put in here to make sure that it’s bridging JavaScript and WebAssembly and the needs of hardware. I have two quick questions. One about the super TDZ thing. Is that a real requirement that we not have super TDZ? super TDZ was added before class fields in the first place. + +SYG: I’m not going to die on the hill it’s my personal preference. I can explain async why it is. I won’t die on the hill. + +DE: Will Mutex have an async form for acquiring a lock, like Atomics.waitAsync? + +SYG: Async blocking and conditioning it’s not conditioning, unconditioned variables are called out as a follow up proposal. This scope here as you can see is quite large. I don’t want to add it here. They are planned. We already have a fairly worked out design already. Not going to be part of the proposal. + +DE: Sounds good. Thanks. + +YSV: This is call out about potential issue with the per-realm prototype described. That is another case we can potentially depending how it’s done shared to unshared edge and concern about GC, I could see this working. I think there is a future for this particularly if we rely on module system to load the prototypes and separately from the shared struct. + +WH: I’ve been chatting with the proposers about various aspects of this proposal. My biggest concern at this point is the complexity of `unsafe` blocks and whether the benefits exceed the cost. I think the benefit of `unsafe` blocks may be a mirage and create a big mess for abstractions and indirect accesses. I just don’t see whether having `unsafe` blocks pulls its weight and whether it’s something that we think should be included in the language, which is what we’re trying to determine when asking for advancement to Stage 2. I like the rest of the proposal. I do like the idea of structs especially after you removed `struct` expressions which fixed the syntax problems I found. + +YSV: I have just a follow up to WH’s concern and we have the same concern around complexity of the `unsafe` blocks in terms of how to implement it. It looks similar to how we implement “use strict”. + +SYG: It’s like that. + +YSV: And it’s quite a bit of complexity for what it is. And we’re not hundred percent sure about this one. + +SYG: Yeah, that’s fair enough. The motivation definitely stands, though, I think the folks are earnest concern there around this—I’ll let MM speak. + +MM: Yeah. I’m not going to try to weigh whether unsafe blocks are worth the complexity or not. But the proposal without unsafe blocks, I consider to be unacceptable. So unsafe blocks were proposed in order to deal with why I considered a proposal in their absence to be unacceptable. And the reason is that the properties, one, the shared structs are public if basically the absence of unsafe is just basically the whole world is the case that would in this proposal currently is called unsafe. It means that you’re not in a position to create APIs that expose the shared structs to non-concurrent clients while encapsulating the raciness of concurrent data. If all of the fields of the shared structs were more like class private fields and they were encapsulated and only the exposed methods of the abstraction could operate on that would be a different story and that would also be a different proposal. Given the publicly accessible nature, the thing that I consider to be a requirement is that the—is to not contaminate the ecosystem with sequential code needed to take on concurrency awareness and not bleed into the ecosystem the cooperative non-defensive nature of the concurrency that’s inherent of this concurrency model. So WH is correct that the methods can still expose concurrency but they also have the choice not to. The properties do not have the choice to not expose the currency. + +YSV: So just to make sure I understand it, your concern is with the developer’s mental model and how they work with racy concurrent data structures in a—the word is on the tip of my tongue. You know what I mean, right? + +MM: The issue is really the provider of the abstraction they’re an expert that thinks they can write shared method multithreaded code and get it right and there’s the vast ecosystem out there of regular sequential JavaScript code. If the expert has two options, they can try to expose abstraction and the clients of the abstraction must be aware of concurrency and there’s plenty of cases of that. There’s also the case that I hope will be common in that the creator of the abstraction is intending to expose a thread-safe abstraction. And they can’t expose the objects directly in exposing a thread safe abstraction if data fields of the object are public. + +YSV: Okay. So I support the motivation. But it is the implementation that we currently take concern with. + +MM: I’m fine with that. The issue is to satisfy the need here and the unsafe block is the only mechanism that is so far proposed that satisfies all of the constraints including this one. + +SYG: Thanks for raising that Yulia. Let’s cooperate on the orientation on this. I see point of order on time. The queue is not empty. Eyeballing it, I don’t see anything that are obvious Stage 2 blockers. I would like to move on to asking for Stage 2 with at least these four open design questions. Sounds like some of—you know, the WeakMap thing got some additional concerns from JHD that we would need to address. The unsafe block got complexity concerns we need to address. There are certain more sub questions now. So with that, I would like to ask for Stage 2. + +USA: Yeah, real quick while we contemplate Stage 2, we have one topic from JHD on Stage 2. + +JHD: Yeah, I mean, Stage 2 means we’re going to put it in the language. We just have to figure out some minor little details. And like not just the one that I expressed concern about but these feel kind of major to me. It seems like some of them should be like largely figured out before we go to Stage 2. + +SYG: What do you think ought to be—like most of the design—my mental model of the process is most of the heavy iteration happens between 2 and 3. Like, between 2.7 I guess. The thing we want in the language is not like this. The thing we want in the language is we agree on the motivation and like the rough shape of it. Now, I see where you’re coming from. But I want to show there is significant agreement on some of this stuff and here are the disagreement that must be worked through. I think given the scope and the envision of the proposal, this is not like majority of the thing. And if, you know, the quibble is with what Stage 2 says, how do we—I think it is important for me and for committee to demonstrate some kind of progress here in the agreement. + +MM: Yeah, I mean, I think like I’ve heard a number of—like, I think MM said without unsafe blocks, he doesn’t think we should have the proposal at all and someone said we shouldn’t have unsafe blocks. Sure that can be worked out one way or another. But if we know that one of—if we know there’s a possibility that the whole thing won’t advance, then it feels like Stage 2 is premature to me. If no one else feels that way, that’s fine. Because obviously they would be blockers for 2.7. But like in other words some of the answers to the questions determine whether the proposal is feasible at all. And that to me feels like a Stage 1 question. But, you know. + +SYG: I think in this particular case, of course, I’m biased. I do disagree in this case. + +USA: If we could be really quick. WH, you’re next. + +WH: I am not comfortable with advancing `unsafe` blocks for the same rationale that MM stated, in that `unsafe` blocks will make concurrent code contaminate the rest of the ecosystem. + +YSV: I support the current proposal shape with the null prototype and no WeakMap integration for Stage 2 with follow on proposals for those two items. I understand that for MM and Agoric it is a requirement to have an unsafe block and still discussion needs to happen there and this is a smaller concern than the other two. + +SYG: So the hardest constraint I heard sounds like from WH on the disagreement with unsafe blocks as it looks today, I don’t know. That makes me pretty unhappy if that is like given the scope of the rest of this proposal, that’s what hangs on Stage 2 and that’s also kind of like—I also don’t know really how to make progress on that given the constraint here. So I don’t know. WH to clarify you are blocking for Stage 2 with the presence of unsafe blocks here? + +WH: Yeah, I’m not yet convinced that we should do `unsafe` blocks. + +SYG: Okay. And you are uncomfortable—that is called out whether—that’s not called out whether we should do it. It is called out on the design questions. Mark, given that you are on the other side of the constraint, what do you recommend that I do here? + +MM: So I think that the problem—I mean, this goes back to the collaboration that Yulia raised, the problem that I raise I think is a blocking problem. It doesn’t have to be solved with unsafe blocks but unsafe blocks are the only proposal so far that have solved that problem within all the other constructs. So, you know, it might be that you can’t get—there’s nothing to propose today given that it’s either with or without unsafe blocks that will get both me and WH to not block. And, you know, that’s unfortunate. But that doesn’t mean that the proposal stays blocked. It means we can search for other ways to solve the problem. + +SYG: Concretely, are you or WH comfortable with searching that given the constraint during Stage 2? Like, I would like to show some progress here. And it’s actually documented that the rest +– there is a surprising amount of agreement. + +MM: So somebody needs to clarify for me. I know I keep asking this over and over again. What does it mean to have approved something for Stage 2? + +JHD: It’s been a lot of years. Can I read the process document again. It says—we have actually tweaked the wording since the last time. Says the committee expects the feature to be developed and eventually included in the standard, but due to reasons that may not yet be apparent the feature may not be included. It’s not a guarantee that the thing will land. + +JHD: I think the only time something left Stage 2 never to return is `Object.observe`. That is very rare. + +MM: Okay. So I’m comfortable with that part of it. Does it say anything about the degree of stability, the – + +JHD: Entrance criteria says proposal document describes high level APIs and syntax and illustrated examples and all major APIs and editorial issues are acceptable. + +SYG: The very first sentence of two says committee preferred solution or solution space and design is draft and may still change significantly. + +JHD: Like the rest of the process document, it’s sort of a mess of slightly vague conflicting statements. We have to interpret it. + +MM: So, yeah, I mean, hearing the language again I don’t feel lawyer-y enough to make a decision. I feel—as long as we are explicit what we’re agreeing to leaving the question open, I am comfortable saying we have agreed to Stage 2 explicitly with this question of – + +SYG: Thank you. That is the intent. That is not the only thing that might make this not happen. The GC thing is probably higher on the list as well. This will be in good company, I suppose. WH with that, how do you feel? + +WH: I’m fine with the rest of the proposal as long as we’re very explicit that the existence of `unsafe` blocks is still unsettled. + +SYG: Perfectly clear. I will record it in the summary. Okay, great. + +RPR: All right. Then we have our consensus. Thank you. + +SYG: I will let people go to break and write the summary in the notes. I will just write the summary in the notes. + +WH: Is there anybody else who has any input on this proposal that might be an issue? I heard GC raised earlier? + +YSV: I raised my concerns to SYG and I believe those are understood. + +SYG: Yeah, the GC thing is understood loud and clear. Probably even a harder constraint than the Wasm GC side. + +WH: I also have concerns about this potentially slowing down programs which do not use shared structs. + +SYG: For sure. This one need to be pay as you use is a hard non-negotiable for VMs. + +### Speaker's Summary of Key Points + +Known open questions that must be resolved during Stage 2: + +- Whether to have per-Realm prototypes (to allow shared structs to have methods) and correlation (and how). Mozilla’s concern here is related to the next point. +- GC difficulty and whether shared structs are usable as WeakMap keys, including JHD’s newly raised concerns that all objects should be able to have private names stamped on them and be usable as WeakMap keys +- Some TC39 delegates raised concerns around the complexity and importance of `unsafe` blocks, given the narrow nature of the guarantees that they provide. However, these blocks are considered important to Agoric stakeholders and others, to have guardrails for data race contamination. +- `Mutex.lockIfAvailable` return type +- whether struct subclass constructors have TDZ for the `this` value before `super()` is called. Shu proposed no this-tdz, because there is no return-override and struct fields are eagerly initialized to undefined before calling superclass constructors, and others suggested that this-tdz should still occur because `this` isn’t logically initialized until the superclass constructor runs. + +### Conclusion + +- Stage 2, with the explicit understanding of the open questions listed above, all of which (except probably the Mutex.lockIfAvailable) may block the proposal from happening + +Reviewers: + +- MM +- WH +- YSV +- NRO + +## Porffor JS engine + +Presenter: Oliver Medhurst (OMT) + +- [slides] + +OMT: I’m talking about Porffor that is a new JS engine developing. I will just go straight into it. JIT is very cool and they have trade-offs that are unfortunate. Mostly the compiling on the system device. You get a fast running time if you compile longer and also longer loading time. And the security can be questionable sometimes. But the ahead of time domain which I guess more traditional languages like C plus plus or rust design will compile to the binary ahead of time which I hope most people will know. This can solve some big problems of JIT and not compiling on device and the loading time issue doesn’t exist and using the binary format like WebAssembly sand boxed the security can be power graded. The down side is you don’t have eval or anything but arguably that’s good. We’ll say since you’re basically making a new JS engine, that’s unsurprisingly hard. So Porffor is my engine that does this. It currently passes around almost 50% of Test262. Just confirm the entire, that correlation there is funding. The other advantages of this is native compilation to binaries and not shipping an entire JS run time you get tiny binaries that are 15 megabytes than 19 megabytes or something. It supports native TypeScripting but since you have control of the entire engine and written in JS and TypeScript. I wanted to participate in TC39 because I like standardization and I think it’s a unique interesting implemented view. And I will hopefully do a live demo. Do people want to shout out things to run? + +OMT: I do not have BigInt but I have classes. I don’t have BigInt yet mostly because I would never decide on a good way to do it. I started doing regular expressions and then kind of gave up because unsurprisingly they get very odd quickly. I have weak ref essentially and I have very basic promises like—yeah. Very basic just no real event loop but makes enough promises. + +JWK: I have a question. So what does it mean running on Node, because it’s in JS engine, right? + +OMT: It’s written in JavaScript. So it’s using node to run it. Does that make sense? + +DRR: I think they’re asking what exactly is the mechanism that makes the code that you are generating run? If you can get into the specifics. + +OMT: It’s compiling to WebAssembly that uses V8 to run it. + +JWK: Do you mean the compiler itself is running on Node? + +OMT: Yes. Unfortunately. I support OXP for this. so if you do TypeScript input, it can use these type as optimization in the compiler. If you have no type inferring, if you did this, it would then know like there’s no chance of sting concur. + +DRR: What happens if you say let C be a type number and then assign a string to it? + +OMT: Yeah, that’s one. it will give you the points of the string. You can then do cursed inline WebAssembly. You can get the length of the string with the because pointers. + +DRR: Okay. And going the other way, if you say, you know, you have something that is of type string and you assign it a number, does it coerce? What is the behavior there? + +OMT: Well, because the TypeScript just kind of expects you to be correct. But I guess for this, it will probably—like if I do a point string, I think that’s funny—16. If you do this. Yes. + +MF: Just understanding what your previous example with plus—it looks like you’re doing type directed compilation so when they're both numbers, you output WASM that assumes they’re f64s or something? + +OMT: Yeah, probably do by example. You can see the WebAssembly output here. Ignoring some stuff, it just does add sort of any sort of – + +MF: Can you make a `c` variable that's a string and then do plus and show that the compilation is different? + +OMT: If I restart this quickly. So you can see here it’s doing concat string. + +USA: There’s also a question on the queue by CDA. + +CDA: Do you have any web platform API, like, set interval? + +OMT: I have a few things like some basic 64 things. I have a few but mostly experimenting with things, because the actual built in API are written in TypeScript and then compile itself like the base 64 break down. With the WebAssembly but it’s written in TypeScript and then compile itself to WebAssembly. And basically every single API is like this. Every single spec. I have a few but essentially it’s kind of good for—I’m using my own engine, it’s good to test the limits of it with APIs. + +DRR: Yeah, I see that you have—I guess I kind of see it maybe in your code up on the screen right now, but I was going to ask, you know, because naively every number in JavaScript is just a flow, but it seems like you’re using I-32 operations in WebAssembly. But I guess do you basically just try to stay in Int land as long as possible and jump back to floats when you’re uncertain about a type? + +OMT: Yeah, essentially. I have like special built in I-32 and 64 types and any—internally if it’s indexed to the string or something I try to use Ints as much as possible. + +DRR: Gotcha. Whenever you—what is the code that you generate when you let’s say access a property off of the object and you don’t necessarily know what that type of that—like, let’s say you have an untyped parameter in the function and you assign that to something of type I-32 what is the—does your compiler have to generate a specific set of codes or run type helpers? I’m just kind of curious about this stuff now? + +OTM: I guess essentially it’s basically just like a big switch statement. It checks prototype and calls different functions depending on the type. Usually there’s some inferring that you can do that helps a lot like if it’s computed you don’t need to do like property key stuff. And you can do some fast things arrays. Just a dot something. + +DRR: Okay. If you wrote a function in Porffor just on this, would it be able to spill out the instructions from that? + +OTM: I guess really basic example. You can just recall to this WebAssembly and just use it as a module. Because it doesn’t know the types, it has to potentially do a string look up. which is checking type things and then—I guess for now optimization mostly because I’ve just been grinding conformance. + +DRR: Pretty neat, though. Yeah, as long as you stay within the subsets of typed things that you want to use, it seems like you can keep things pretty type. + +JWK: Does it support dynamic part of JavaScript, like it can change the object’s prototype on the fly? + +OMT: Yeah. + +JWK: Get the correct results? + +OMT: It should. It can be a bit iffy but you should be able to—some things don’t work but you can hopefully do like a dot or—like, for each custom function. It doesn’t not work because of some—yeah. This might not work because there’s some caching stuff. + +JWK: Can you change the prototype of A? + +OMT: Oh, yeah. t does some weird stuff like primitive objects and do some optimization and if you do an object. Yeah, there’s some weird stuff that I try to do some optimization. I have like hidden internal types which if you ever have, that can be strange. + +DRR: One of the things that you mentioned early on is that you could get very, very tiny binaries at the end of the day, because you don’t necessarily need, you know, certain runtime helpers and things like that. But are you able to—I mean, it seems hard to guarantee that when you have dynamic access to properties and things like that, right? Are there like—are you planning to disallow those? Are you planning to just say if it happens, it happens? And what you get is what you pay for? Or are you thinking about guardrails there? + +OMT: I guess I at least I hope most of the time that wouldn’t really be a thing. But like for this example, it’s pretty tiny. But if I did like globalThis, it would probably include everything. Yes, so that’s a hundred kb and including every single function but it’s pretty tiny. + +OMT: It’s still relatively small compared to everything else. So I’m reasonably confident it would still be good enough. + +USA: That was all on the queue. + +DRR: Are there anything that are staking out that, you know, you see as major hurdles coming up that you’re either thinking, well, that’s going to be a lot for me, like, you yourself, or that you specifically need help with or feedback on or things like that. + +OMT: I guess stuff like `eval` I will probably never do. Like, you could do it but including the compiler in the binary, but at least for now, I’m not really worried about it. I guess my hope is most modern JS doesn’t rely on that sort of stuff. But I guess there’s some things like having a really good GC and stuff. I hope to have an option to use Wasm GC or like my own GC because if you’re say using this sort of website, you want Wasm GC because that makes sense because it’s an embedded device, that won’t support Wasm GC any way. Any other question? + +DRR: Are there any specific code bases that you have tried to get working in the wild at this point, that you’re sort of using as a test bed or really working on Test262? + +OMT: I guess I’m just starting to get to that point. My hope is some day to get it like compiling itself. I’ll probably use my own code base for that. Because I use ecorner, so I will hopefully get that working relatively soon. + +DRR: Very cool. Just want to put that out there. Thanks for presenting. I don’t want to hog out all of the questions. I appreciate it. + +OMT: Thanks. + +### Speaker's Summary of Key Points + +I’m on matrix if you want to talk to me or I have a website that you can follow things. There’s an online playground thing you can use for compiler with the graph of test 262. Thanks. + +## JSSugar/JS0 + +Presenter: Shu-yu Guo (SYG) + +- [slides](https://docs.google.com/presentation/d/1ylROTu3N6MyHzNzWJXQAc7Bo1O0FHO3lNKfQMfPOA4o/edit) + +SYG: Okay. We should get started. All right. This is going to be a spicy topic. It’s broken into two parts. The first part is a description of what the folks on this title slide here think of the current problems with language evolution. Inside TC39 and how we evolve JavaScript and the second part is possibly the more spicy part of a solution that is endorsed by myself as V8 and Google, not necessarily endorsed by folks on this slide as a possible way forward. So with that, going to go through first—so part 4 is the let’s talk about a possible future and possible solution. And parts 1 to 3 is a description how we see the state of the world today and problems with it and some of the issues with it. So who is involved in language evolution of JavaScript? I think the stakeholders are, on the one side JavaScript developers and these are the folks that author JavaScript and I use it loosely and don’t mean Ecma 262 and could be JavaScript and extension like JSX and these are developers front end and web and native via web tech such as Electron and the Bloomberg terminal and backend stuff like Node, Deno and bun and also embedded like Moddable. In the middle I believe are the implementers and are implementers of tools that transform travel script like like Babel and like TypeScript except and then also the implementers that usually talk about in this room who are the engines of which there are SpiderMonkey and Firefox and JSC and V8 and Chrome and XS in Moddable and they execute your JavaScript. And the other are end users of JavaScript and don’t care how they’re written and what technology and have the use of the browsers and user of the apps running in web browsers and a lot have service side render on days and apps on server run times and embedded device and JavaScript. They look like this. And JavaScript and JavaScript adjacent and feed to tools and possibly none and the data bears out that everyone in the ecosystem use one tool and spits out JS and run time and engine consumes and run the application that the users interact with. And + +SYG: I want to start with really calling out the JS is really special. JS as a programming the practical programming language in the world is really outsized in everything. Has tens of millions of developers, some estimate on some like estimate—some statistics service website estimate 25 million JavaScript developers and billions of users of software written in JS and they are users of web browsers and Chrome has billions of users with Safari and Firefox and I i don't know how to count the IOT stuff and I imagine a bunch. There’s many, many maintained independent implementations. This is really pretty special that we have such a vibrant standardized interoperable ecosystem. So that’s the stakeholders of who is involved. + +SYG: Why do we evolve the language and why are we here and why do we work on JavaScript and how do we evolve the language? I think we evolve the language to help developers. We want to help developers to be more productive in general. And we change the language to try to help them better match their mental models and this could be functional programming, this could be OOP style programming and type systems and with the recent share of iterator proposals and people who are used to thinking about streaming and transforming data. We want to help developers to be more productive by enabling them to express new domains. And these things can come up as the world discover technology and machine learning and model stuff and finance and science computing and want to help developers build new class of apps with new capabilities that weren’t possible before and things I’m doing with shared memory in the camp. I want to help people write a new class of apps with the new class of performance that wasn’t possible before, WeakRefs similarly to enable a way to hook into the garbage collection was not possible before and help developers new interfaces by us and assembly and help simplify their lives by removing dependencies and build tools. And I don’t want to be so bold as to say this is exhaustive, but this is like most of the reasons of how we motivate proposals here and why we want to evolve the language at all. How to do this? + +SYG: Abstracting away complexity in mental model and new domains and reify common patterns and remove dependencies and did see new abilities and new capabilities and new interfaces of Wasm and explicit coordination point that we think is a benefit to the ecosystem. And the first one is bolded because I think that is the bulk of the work here. Important point here is abstracting away the complexity usually moves it. It moves it and may look like it is removed from the surface language. But in terms of the full picture of the stakeholders, we think it moves it, it doesn’t remove it. The complexity move from developers' code to the language implementations. So when the complexity is moved, the characteristic changes a little bit. So an application—complexity in a particular application, the application level complexity has negative impacts limited to the user. If the particular application is complex and has problems arising from the complexity, if you don’t use the app, it shouldn’t affect you. It’s usually easier and faster to patch. Because on the web you can push out a new version. You can deploy a new version without asking everybody to install it. Implementation level complexities in the engines in the runtimes are—they’re broader in their negative impacts. It impacts all of the implementation users. Impacts all of the browser’s users. Doesn’t matter the application they’re running, the complexity in the browser itself and harder and slower to patch. In Chrome, and I can only speak for Chrome's engineering practice that I know best here, for zero day security patches we do a respin and release a new binary because we think the cost of having to ask everybody to do that outweighs the cost of people getting exploited in the wild, but respins is also a very real cost. It’s just harder to roll out because people don’t want to update for whatever reason. So that’s the how and that’s the why and the how. So far, we talked about developers and the implementers a little bit. What about the users that actually use the applications that are written in JavaScript? So from a user’s perspective, what do the users want? They are not developers. They are probably not necessarily computer savvy at all. But they want security. They understand this at a pretty visceral level. And if we’re doing online shopping, you don’t want people to steal the credit card information and money. The scary version of security is that for a piece of software like a web browser that has the reach it does, zero days are known to be used by, let’s say, I think The term of art in the industry is like "surveillance vendors" or something, I forget what the the actual thing is. It’s much scarier than it sounds. Basically these are folks like nation state level who are in the business of exploiting these zero days to, you know, destroy their dissidents and enemies of the state in some way. This is real bad shit, that at the end of the day why we care so much as the browser vendor about security. Yes, it is bad if your credit card got stolen, but it can be a lot worse than that. + +SYG: There’s performance concerns. Everyone wants the stuff to be fast, naturally. Hopefully that’s self explanatory. Users want stability. They don’t really want their app to suddenly change; especially to suddenly break. And users in general want features. They want new features. So remember the end users are, for the engines, the implementer’s primary customers. We want to help web developers but at the end of the day the people that use the product are the end users and we want to do right by them. What user impact do language features have? From the security impact it’s almost always negatively impacted. I will go into it. As the performance we hope best case we hope it’s neutral in that it is pay as you go. If you don’t use the new feature and introduce a new feature the existing codebase of JavaScript programs are not using it. The purpose of the application is not affected. But sometimes we do introduce new features aren’t pay as you go and add extra costs to the runtime and parser. Sometimes it's worth it. It’s still a fact that performance is negatively impacted by adding new features and stability is sometimes negatively impacted. I will go into that. The main part where we have positive user impact on language features is on app features. The hope is and very hard to show direct causation here, the hope is if we add new features, that developers truly want, that helps developers have higher velocity to ship new users that delight the users. That is the ideal path and why we add new features. But that’s the main hope people ship better features faster is what we hope new language features to do. All right. So to drive home the security points and more, I think the takeaway here is that complexity equals security. This is an undercount of security bugs in browser engines. Because these are CVEs and we don’t file CVEs for every security bug. Probably not every zero day either. These are the ones I took a quick survey of sampling of the last like five years of CVEs with V8 and SM and asked other developers to implement those and most were V8 and took the ones that are feature related. Part of the point I’m not showing that syntax features are bad or that a particular kind of features are bad, it’s that all features basically result in security bugs. Any feature we add is complexity-- BigInt and promise scope and built-in subclassing. Everyone probably hates that. Probably shouldn’t have built in subclassing. + +SYG: You know, again, like, I think for a lot of these, we still made the right choice. We should have added those features to the language. We will get security bugs, on the other side of the coin. The point is not like we are bad engineers. Hope we’re not. I don’t want it toot our own horn, but we have security reviews, a lot of compute dedicated to fuzzing, we have bounty programming to throw at external folks and their compute. So they report security bugs to us early and often. Despite that, this is the state of the world. + +SYG: So the point here is that VMs are very complex because there’s pressure to be fast because we as users, we want that, and that cross-cuts security and we want to do better obviously. Again there’s competing pressures here. That is bad when complex features, responsibility for complexity don’t get adoption, species, BigInts to an extent. We add things on the hope that the arguments we were convinced by in committee, are true, and but the real test is, when we ship it, in five years, did their motivations come to pass? It feels bad when that happens. We incurred a lot of cost and didn’t get the gain. + +SYG: So that’s security. For performance, a lot of new features are slow because we put in guardrails that can’t be optimized. I think we put it there with the best of intentions, right? It’s like why would you ever want to use that before a dev? It’s great. If you do use lexical scoping, you get this additional guardrail. But it has since then, come to light that some—well, some large applications who are performance sensitive and have regression tests can’t ship `let`/`const` natively that—because the TDZ checks can be not elided in case and they transpile it to vars. The transpilers did not have TDZ implemented because that is too slow. So the state of the world ended up being we added a well intentioned guardrail that doesn't get used in matter in practice. There’s some designs that are I think in a vacuum the right one, but are now realizing even back then may be inherently slow for the network. Like our module system. Native adoption is on the rise, but I think this being a round trip network time is like something we must solve, otherwise, it will never get native adoption. We design static semantics. The file system locally. You know, the performance tradeoff is different. That’s fine. But we didn’t design it to be correct for a performance free network. The iterator protocol is one example. But there are things we design with too many allocations… we really didn’t work out. + +SYG: Moving on to stability. I think this one we're very good. Probably the best language in the world at this. But still, we have sharp corner here. We are good with backwards compatibility. We take "don’t break the web" very seriously, but there is a sharp corner: enterprise, the smaller corners of browsers users at least that institutional for whatever reason they decide to use long-term support. The problem is that the reason they freeze it, they want the browser to not change because they're a risk model for hospitals or decide that’s too much risk for them to take. + +SYG: But then, applications on the—in the wild, may move on features faster than that. I think Firefox had experience with nullish coalescing where nullish coalescing was not supported on older enterprise versions, but supported by the modern browsers, recent versions I should say, that a lot of apps switched. But then that broke for these enterprise customers. And you know, they complained to the browser vendors and this is something we see. + +SYG: And finally, the positive impact on new app features. Now language features… we see some holes in this as well. Sometimes it works out. But when it hasn’t worked out, we have seen that native adoption of features is lackluster because sometimes they find performance regressions. The play book here goes, some app with performance regression test suites, which is already a very small percentile on the app. They decide, it’s the best thing to transpiles generators and to ship them natively. For whatever reason, maybe it’s better debugging in the wild. Maybe it’s smaller wire size, they think it’s like the best practice and they should modernize their deployment for best practice. And they find out it regresses performance. They file a Chrome bug, a Firefox bug, a WebKit bug. And we try our best and sometimes we can make it faster and sometimes we can’t because the stuff we design ES6, it resists that level of optimization to bring to parity with ES5 performance. + +SYG: And sometimes use cases that we hope would materialize never did. Like BigInt. And some of it I think is—I don’t have, you know, I don’t have a way to substantiate this. There is lack of incentive to stop transpiling. The users aren’t complaining about being too slow. And why fix what is broken? When you wrote it, it wasn’t—widespread enough support for like cost. And why bother to update. There’s inertia on a lot of part of the web A reasonable thing to ask is, we do important work here. Good work with evolving the language. Aren’t these negatives to the users offset by the benefit to the developer? The developer builds more things. Higher velocity. Isn't it like an aggregate negative positive on the JS ecosystem? No. This is a bit flip. Which is why I have this small note on the bottom. There’s a trade off. Obviously, there are cases where a win for the developer. It’s not a clear win everywhere. + +SYG: And the main reason I think is that the cost to the users of these browsers of these runtimes, the cost is in aggregate to all of them. When we incur complexity in the language in the VM, it’s a cost to everybody regardless of what app they are using. Just from principles like it would be nice if complex things were only at the edges, foundational things ought to be simple. JS runtimes are used by billions, a security flaw there has wider implications. The user base of each application is much, much larger than for the vast majority of apps. And I think there’s an important note, there are corners of the web, the ecosystem I know best, but where there are apps that are really platforms and something worth attention. WordPress is something like—I don’t know crazy like 30% + of the web. Something there deserves special attention. Something in FaceBook and Google and these web apps, they do deserve special attention. I am not saying all applications are much smaller. And complex illustrated costs in the runtimes have a storage, memory and performance cost to every user regardless of adoption of those—of those features. + +SYG: So the way I am framing this, I think that there’s a polarity here and the goals are the intentions on either side. We want to help developers to be productive. On the other hand, our attempts to do so in JavaScript given the constraints that exist in the world, puts that intention with doing right by users along the security performance stability axes. We are here, to develop developers and be more productive. I would like for us to move over here, I think that’s an eagle. That’s what I searched for. It looks like a pigeon. I want to move us more to the right of the polarity of doing right by users because that is the people that I think we should be looking out for, there’s just billions of them, and there’s, you know, there’s consequences. + +SYG: How do we get there? So far this presentation is how we understand the world. I think the implementers get the brunt of the fire and we deal with the zero days and patch and do everything and we would like things to be better. How do we get there? + +SYG: How do we get there? We need to have a talk about how we introduce new features. This is a broad discussion and I am going to offer one possible path forward for one narrow kind of feature. But this is—the description, hopefully makes it clear it’s a broad problem that I think concerns all proposals. + +SYG: So let’s talk about the future. What engines want, which hopefully I impressed on people, we want security, performance and stability for the users. So the takeaway: we will be conservative to move the polarity more towards doing right by users. What I am not proposing here is a new process or demotions. I think it would be bad to try to demote stuff ad hoc, stuff that has reached a certain stage, not demote for this abstract reason. If we want to demote stuff, we should discuss it case by case. This is really about steering the future. + +SYG: So before we get into that, we should—first take a taxonomy of the new language features. Broadly, we have syntax, we have API features and capability features. And perhaps some combination of both. + +SYG: Syntax features are characteristics of the kinds of features. They may be decomposable. This is a polarity shown by the trade off arrow there. Represented as API. We saw this with BigDecimal proposal because of constraints from the engines, the champions have decided that it’s worth the addition as a pure API, even without the syntax. I guess it’s like fully decomposable syntax. Syntax that can’t be proposed by API, but people want to type not a API—like pipeline. Syntax where the textual representation enables its function, like records and tuples proposals, structs in the declarative immutable sense of it. There’s also a language distance polarity for sine features. Some syntax fit without issue into JavaScript which is dynamically typed. The other languages are statistically typed, they are perhaps—designed with the static type system in mind. The history is better fit, and sometimes we kind of butt up against that when trying to make the same patterns work well in JavaScript. + +SYG: APIs, we have built in methods. So iterator helpers, things like that. We can also add new builtins. + +SYG: For new capabilities, these are often about exposing capabilities that are on the underlying machine, but not inaccessible through JS for whatever reason. So things like float16, we added new machine type. That existed in some machines at least. Access to GC information, not exposed that we decided would be useful to expose like WeakRefs. And for the shared structs, exposing a shared multithreading, that native applications have and we don’t have and there’s opportunity costs in that Sometimes exposing the host platform. Not like characteristics of the underlying machine or OS, but about one layer on top, so web APIs, WinterCG, TC53. These are new capabilities and they have special or not opportunity cost of not adding new capabilities is large because the alternative is you can’t do it. The alternative is not that it’s more unfortunate to type or that it is less understandable, or more verbose, but that you can’t do it. This is where a lot of the risk appetite for implementers is because we are interested in growing the pie and also help developers and this is—capabilities are basically an un addressed corner of the developer market that we hope if we add a new capability we can—bring them in. + +SYG: So the rough guidance for what the implementers thought should be added to the language. New capabilities, simplicity for understanding implementation for security reasons, for the ones I discussed. Features that have wide adoption and known performance advantages by being in the engine. High confidence needed in the use case for history of, you know, agreeing to implement things, hopefully—hoping that some use cases also pan out and finding they don’t. And the costs associated with that. And particularly the existence in another language should be sufficient motivation given the cost that we are currently bearing. + +SYG: With all that said, how do we do that? We could like not—say no all the features. That’s bad. We want to help developers. Polarity is not a black and white thing. Can we have a win-win flexible situation? I hope there is a solution where engines being more conservative for the reasons we say, does not directly imply that we stop evolving the language. Here’s a possible solution for syntax features and this is mainly my preferred situation, Google's, and do not speak from other implementers from here on out. + +SYG: So our preferred solution, first take stock of how we see the world and what we see through data that we have of the developers ecosystem. Through the data we have seen that developers have embraced tooling in the past decade, I suppose. TypeScript continues to grow in popularity. Babel has 50 + million downloads. TypeScript usage outweighs JavaScript. And survey data show 55 to 9% of developers using framework. I don’t run these surveys, but everybody uses tools is the picture we see from data. + +SYG: Tools exist and widely adopted and they don’t go away because they are on an upward trajectory. Can we use that for a win-win? If we standardize JS as the ecosystem uses it, perhaps we can split the language into two layers. One, which is a directly executable layer, for the sake of being concrete I am calling JS0. And then a layer that sits on top, which is still standardized, with normative semantics. With normative syntax called JSSugar, it must be compiled down to JS 0. They are not directly executed by engine. But standardized features that are progressed by tools like bagel, whatever tools there are and spit out a JS0 that has observable semantics equivalent to what we spec it to be. + +SYG: And for developers, the thesis is that because the majority of developers consume JS by tools, they don’t really need to learn the difference. From their point of view, they continue to write is JS. Some of it gets transpiled away and standardized to be transpiled away. And this is a reflection of the world today in that not everything they write is—pass through and shipped. They have to choose what target they want to ship it. ES2025? ES5? 6? And from the workflow point of view, the proposal is basically just instead of thinking of ES6 or ES2024 or whatever, they now think of it as a JS0 version. + +SYG: So hopefully no additional cognitive burden to work flows for developers who already have tooling. So taking the previous picture, the proposal here is JS is split into JSSugar and JS0. From the developers view, it’s JS. We reduce the surface of things directly implemented by engines to address the stability performance that the first half the slide deck was about. + +SYG: Concretely, the proposal is that we allow future syntax features to be specced as desugarings. This is scant in actual detail of how this would work in 262. I think it’s more important to discuss the high-level of ideas here and then the nuts and bolts of what we can iterate on as needed. But I don’t want to lead with the technical details. They are still standardized features. They can’t implement whatever they are not. There’s a notion of compliance. I think tools for those features count as implementers as like the engines count for all features, I think tools ought to count as implementers and engines don’t for the JSSugar features. API and capability features go to JS0 because that’s where they—because that’s just a proposal, I guess. Other folks in the space, you will have some exciting ideas about how we can apply the same possible win-win flexible thing to libraries, but those ideas aren’t fleshed out. One still counts as implementers for the JS 0 features and one standard. + +SYG: And nuts and bolts aren’t worked out, but Test262, you can imagine that there’s a section that only the tools run and they are also, you know, compliance matters for the tools. + +SYG: This is not a proposal again for tools to do non-standard extensions. This is a misunderstanding that I got multiple times. I want to iterate this. This is not anything that transpiles to JS like a JSSugar thing and give up on that extension space. I am not saying that. One way to think about it, it is a proposal to call for more tooling into TC39, I think we are glad to have babel implemented here and TypeScript implementers, but the tooling space is bigger than that. And honestly, we should have more direct participation from those folks. + +SYG: So continuing with the proposal, you can also think of JS 0, the directly executable layer as a ABI layer to aid coordination of compiled JSSugar features. JSSugar features need to demonstrate that they are debugable at runtime and sufficiently supported by source map. We can’t add the sugaring and them out high and dry off. In my vision, the JSSugar and JSh 0 are fluted over time. If we ship something, via JSSugar as part of the language, because the state of the word today is that early adoption or mid-term adoption of new language features and syntax is via tools, that—think of it like a long incubation time, we’re shipping it and telling people it’s okay to depend on it. But that gives us some data on the adoption on did the use case that we think—like, did the motivated use cases we think would pan out actually pan out? And give us data on our design. And it could come to pass over time, shipping via the tools teaches us that we are leaving something on the table by not having it in the engine. Leaving performance and security. It’s actually—you know, we can argue with more confidence in data that it’s a good thing to support in the engine natively, and we should also be free to move such features into JS0. + +SYG: It’s totally reasonable, it may composed of a JS 0 and JSSugar. There’s syntax and but a small runtime core to support that syntax to be used. + +SYG: So is this a win-win? I hope so. This is highly speculative. It’s meant to provoke discussion. Here are my arguments why I think it’s a win-win. The hope is the language can improve DX without impacting all users’ performance and security. JSSugar focusses on DX. JS0 features focusses on simplicity and security and performance. And rough guidance for higher bar applies to only JS0. Horses for courses. It is unfortunately just true that JS VM teams are poorly staffed to understand the broad spectrum developer feedback on the web. Is a very large space that is very heterogeneous and for us to fully understand we are not equipped to understand that, very well, but we’re good compiler authors. And this exchange would let us be like—what we are good at. The tooling teams, the users are the developers. Hopefully they get direct feedback from the developers. + +SYG: And the improved runtime security is mainly by slowing down the increase of the JS attack surface. Because I am not proposing we remove anything of the JS today. We must maintain backwards capability. But hopefully, the slow down lets us secure and battle harden at a slower rate. + +SYG: I also hope that JS0 can have fewer guardrails that are slow and complex and put those guardrails in the tools. You can imagine an alternative world where TDZ is static. That is very hard—like there’s problematic problems with that design. Like, you want to close over stuff. So the static TDZ doesn’t make sense. You can imagine a guardrail where it’s statically checked by the tools and then compiled away and then JS0 doesn’t deal with that and developers will still use it and get a lot of the benefits. + +SYG: And I hope by standardized features expected to be compiled, transpilation will feel more permanent and better aligned with the tools. I think some tools for rationale reasons are not perfectly spec client, for too fast or slow or whatever, if we thought of features, we would design them differently to not have those problems. We spec something that has observable semantics; it's up to the tools to figure out how to do that. And how to implement that as transpilation. If it doesn’t admit a good transpilation and the choices are super slow or break corners and people use you, and you hope the corners you broke doesn’t matter. Given the wide adoption of tools, I hope this bi-level JS shouldn’t not affect the majority of the workers. + +SYG: JS features can be less stable. I don’t mean we should change things willy nilly. But obviously, we end up codifying, we agree on as a committee and agree on as mistakes. My hope is that those kind of corner cases can become less stable and easier to fix because the universe of developers is easier to convince to update a tool than the universe of browser users. We are talking at least 25 million versus 3 billions. The order of magnitude is very off. + +SYG: If we design features with transpiling codegen in mind, we should be designing it with that in mind. That is the first, you know, like that does matter. We may think to ourselves, in the long tail, that people will stop transpiling and they will ship natively. But that has its own cost as we have seen sometimes it’s not possible and it’s true that people do end up just leaving the transpiled steps in place. And there’s no reason why we shouldn’t really think about that while designing some of the features. + +SYG: FAQs to address some of the questions I heard when I talk with people about this. 1 JS. Does this kill the idea of a unified JS? I hope no. The state of the word is that JS is authored with tools in the loop. Tool-using developers will care about JS0 in the same way they care about what version of ES to downlevel today. I don’t think that is not one JS. I hope this doesn’t also break one JS. This probably the most conscious one. Why does it take too long? The starting point for me, we look at the data and the author, directly for runtime model is outdated and no longer borne out by the data in the [juz] ecosystem. JS grew up basically. And it is definitely true that there are many hobbyists developers small teams that directly benefit from a toolless model of JavaScript. But if we are kind of maximizing the demographic that we want to support, what we see in the data is that people use tools, frameworks. So the state of JS development today already requires tooling to be productive and competitive… I don’t see any evidence that the adoption rate of tools is increasing, despite efforts of influencers in the field saying that toolless JS is where it’s at. Aspects of web development today that preclude a frictionless direct authoring like secure context server. Direct authoring can be recovered also during development at great performance cost. DE, who has concerns about this proposal, I don’t mean to imply he endorse this proposal, but he pointed to me back when CoffeeScript was around, there was a way to copy-paste this in an indication tag and have a CoffeeScript tag. + +SYG: You can author CoffeeScript directly. If that is—if you don’t care about performance and want frictionless performance, you could cover that. + +SYG: Won’t devtools be worse? I would like to turn this around. The opportunity for having a standard ABI layer hopefully can improve debuggability of transpiled features. Right now, there’s no market pressure for different transpilers to coordinate at this ABI layer. It’s an opportunity to require better source map support for transpiled features What about my proposal in the pipeline today? Best I can offer there, see the rough guidance and it remains important that proposals be individually considered on their merits as before. So next step and challenges. Tooling representation TC39 needs to improve, if this were a thing and just for a process-wise how does this work? Keep it in plenary? How do we aspect the sugaring. Is this a good or bad idea? These are specific technical things that have not worked out and are less important to work out until there is broader alignment. There’s a giant queue and people are angry about stuff. So let’s hear it. + +CDA: There was a slide that said for developers, JavaScript would be a combination of JSSugar and JS0. It seems like it’s not transparent to the developer. They need to use an intermediate tool. Or will a transpiler be baked into the engine? + +SYG: it’s not baked into the engine. It was for devs, the majority they have a tool, with a build step. For those developers, that is transparent to their work flow. + +WH: I like the direction this is going in. I really resonate with the second half of the talk. On the other hand, I find the first half problematic in terms of setting up framing as a conflict between users, developers, and implementers. The premise was that there is a constant amount of complexity, and abstractions just move it around from one of those three to another. I don’t think that that’s necessarily the case. Complexity is not constant. Most of the complexity we are dealing with came from subtle decisions — little syntax decisions, little decisions how functions scoping should work. We are suffering for past bad decisions. The best way to reduce complexity is to think much more carefully about the complexity that we are introducing now. A timely example of that would be `unsafe` blocks. + +YSV: I think we agree WH that we need—if I understand correctly, what you are saying, and please correct me if I am wrong, you are saying that we should be more careful in terms of what we are introducing to the language from the syntax perspective. Is that right? + +WH: Not just syntax — behaviors too. I’d say 80 to 90% of the complexity we’re dealing with is not inherent to the problems we’re solving. It’s from bad decisions like adding `Object.prototype.__proto__` to the language. Things like proxies. Complex syntax… + +YSV: Yeah. I 100% agree with you. It’s not just the proposals. It’s how we go about, for example, resolving constraints in some cases. Or trying to solve actual problems within proposals. I agree this is one of the issues that we need to discuss solving. And I think we would be open to proposals about guidelines or other mechanisms to use to help us not make mistakes. The problem; once we land something in implementations we can’t unship it. It really raises the bar in terms of what we are doing here + +WH: Absolutely. Absolutely. And that’s why we’re suffering so much, from past decisions such as adding `with` to the language. There is nothing really inherently expensive about the TDZ. It’s a problem for implementers due to excessive forward hoisting within scopes, which was just a bad decision. + +ABO: Right. So I think that there are some of the issues in the first half of the presence, that would have full effect inside the proposal. So if you—the way—I think something that I am not sure if—well, that—okay. More many websites, I would say most of the websites that use tools are okay. Maybe there are some—my feeling is that for many websites, some are shipped, and it stays there. And maybe some things get added later on. It’s not like the whole stack of code gets rebuilt with every change. And that means that you—if you pass the buck out of implementers features like transpilers, can you fix mistakes like in those—in that compiler that will not change? And you will not—if you—security will not be—much worse in browsers and you would not have undefined behavior in, like, in transpilers. Could you still have thoughts that—that end up being, like, and so on. These might get them fixed in many sites. And about the enterprise thing, the use case of some sites starting using optional chaining, and like that—I think that is unavoidable because you might be only focussing on JS because you work on it. But that will—like, that doesn’t happen only with syntax. It will happen with new APIs, it will happen with new web platform features. That is unavoidable. And this proposal will not change that. It will go beneath the surface + +SYG: I will be respond to that real quick. What you say is true. I put this slide back up because the ideal number of security bugs is not zero. The ideal number of complexity is not nothing. It is a polarity. And I am talking about moving the needle to the right here. Which does not mean it is a 100% thing. We’re talking about—I am talking about things we hope will move the needle. So any impact here is a good impact, even if we don’t completely solve the problem because as you say, I think the problem is unsolvable for 100%. But it’s incorrect to conclude then we don’t try to solve it. + +YSV: I just have a quick reply to that. I think also, JS should—the proposal shown here is looking at one facet of the problems. We can look at these from different perspectives and I think it’s worthwhile investigating it from alternative perspectives as well. + +ACE: So TDZ, as a concrete example, there's something where, as you say yourself, Shu, tooling, it can pen more time analyzing, it can still, at least from what I can tell, tooling, they won’t check TDZ. It might catch trivial cases and even the runtime. When you get functional declarations, it’s like everyone agrees that’s hard. + +ACE: If we went back in time, and we had this world before we implemented those things, I am wondering what we would have done. It feels like on both sides, both tooling ahead of time and at runtime, it’s a case where we kind of want the checks to help developers catch these mistakes. But it’s too costly to do those checks. So in that world, I can imagine that split hair isn’t necessary like which layer. But it’s like at what time should the semantics hold? You could say in the spec, we think developers tooling should check that this—this variable is only accessible after it’s been initialized. Because sometimes in tooling, maybe DRR… What about the tooling ban? Sometimes things that it will allow. It’s the gray area, if the spec allows it, maybe we should allow it. And this TDZ is a place where we want to say this should a check. It has to be done at runtime because it’s—if you want the tooling to emit code that adds that check, only when you are doing a debug build, it’s going to have to, like, wrap the reference in like—either a site like—use some magic symbol to represent the uninitialized state to add all the runtime checks. I am wondering your thought of that split, of a spec designed for clarifying which things the tooling should be. Trying to help the developer cap, even if it’s an actual runtime assertion. But I want to avoid that runtime assertion once we go to the production. + +SYG: I think it’s—that sounds fine to me. If we have stat semantics, run by the tool, that works for something like TDZ. Like sugar, how would you frame that? + +ACE: I was thinking more, with TDZ, what the tooling is going to output, what it might want to output is going to include checks for the whole thing we want to avoid, we don’t want the checks altogether. So if let and const, and my tooling adds in the runtime checks, using JavaScript, like it takes every access that it can prove statically isn’t going to be safe. It will add if this something is some magic symbol, that means I am initialized.. If that’s what the output should be that will happen in production so it will be slow so it will have the problem, we are now doing the checks. When really, we want to not do the checks altogether. + +SYG: I think I am confused on what your concrete proposal is. I would hope something like TDZ would not emit any runtime checks + +ACE: Another access error. I want two different outputs from the tooling. Tools that adds more assertion because I am a developer and I want to be called by—remove the assertions. But the spec only definitely has one output. + +SYG: I think Justin Grant in private communications said, one of his concerns was this configuration problem, it is a fact of the tooling world today that you configure things. And it would be additional friction for developers if JSSugar had—I think there should be configuration-free. This should be 0 configuration. But how would you think about—if the—are they considered clients or not? + +ACE: Right now, the too long, they are arguing the line of which things they are adding to the spec. We’re going to—the obvious, writing typed script. It’s hit and run for the team to say, we can ban that. There are some legitimate cases, they are arbitrary of what is the right thing to do. And we have—with babel and things, what is safe? Do we ignore document.all? There are cases, like ECMA and TC39. We want to avoid things. Should we start to hint where our opinions lies so the tools aren’t left to make the decisions themselves? + +SYG: That’s a good question. I don’t have an answer. This is the kind of discussion I hope would come and then if there is consensus, the broader problem statement, the broader question, it sounds like a good thing that we need committee discussion on, tooling author input on. But yeah. Thanks for bringing it up. + +CM: So I am very impressed with the analysis. One challenge, I think, we face, in this—you face in this proposal: there is what I guess you could call a form of availability bias, baked into our process at a deep level. When we talk about benefit to developers versus benefit to end-users, benefit to developers is very immediate, it’s concrete, you can say, “we do this, it will benefit them in the following way”s. Benefit to end-users is kind of indirect and abstract. And our whole process is rooted in very specific language-feature proposals. I see this even in some of the discussion by previous commenters—very fixated on the details. It’s very driven by people focussing on the language in the form it gets exposed to the developers. + +CM: And I think you're advocating trying to pull away from that focus a little bit, even though the underlying process is very much the product of a lot of evolutionary selection effects over a long time that biases it against you there. And I think—I am impressed with where you took your diagnosis of the problem and where the proposed direction to take came out at the end of the analysis—but I think we need to have a meta-level discussion about where we can take our process so that the kind of decomposition of the language in the way you propose will successfully achieve the kind of benefits you are looking for. + +SYG: Thanks CM, I will respond. Real quick. I agree with you. Very much, but I have—I am skeptical of productive discussion on that because I think we don’t share—quite bluntly, many delegates don’t share the same goals. We work as different participants of the ecosystem and we have different goals because we are employees or have personal biases. It’s absolutely worthwhile to have the discussion. Sometimes they are irreconcilable differences. If you look at other languages, to get at the JS’s special slide, with the lot of other languages, it is the right move for them to focus on direct benefit to developers because they are in the business of trying to grow. Of trying to grab more developers, to become a—a language with more mind share because, you know, they have beliefs that they are a better language for whatever material benefit, memory, safety, that kind of thing. JS is so widespread that I don’t think we should be operating in a growth mode anymore. Again, this is a personal judgment. Value judgment on the state of the—on the state of JS. There’s so many users that we must be more risk averse. + +JHD: Yes. The security risks and performance issues and complexity and stuff of implementing syntax features in engines, I can’t speak to that at all. So I take you at your word, that it’s difficult and you have listed all the CVEs and that makes sense to me. But if it’s not difficult for tooling authors who are exceedingly less well-funded than engine implementers, as a transformation, then if it’s less difficult for them, I am not sure you couldn’t have a on the fly thing. If you want to do it faster, run this transpiler plugin or whatever, it will drop it down to JS0. Right? If it’s hard for them to do it directly or whatever, it’s just syntax sugar, like you said, why are they going to do a better job than the far more funded and large population of engine developers? + +SYG: Well, so + +JHD: I say that raising that it—let’s say a browser implemented on the fly implementers. It won’t be fast, but correct, and in theory dodge the issues that this proposal attempts to dodge. + +SYG: It wouldn’t solve—like, the market pressures for something supported in the native runtime to be fast is very large. So to bake in the transpiler I suppose solves the security issue in the sense that we’re not like modifying the core thing to the—that exposes the security bugs. But it’s—it directly goes against the performance goals of we want the web overall to be faster. If nothing else, a ahead of time tool, a step, is very clearly by construction pay as you go. + +JHD: You don’t anticipate—let’s say a few years down the line, after we ship some things in JSSugar, it seems to me like there is market pressure to push those things into JS0 and just slightly defer the same problem. + +SYG: But it’s a very important deferral in that we will have known at that point, if it is in fact, used and in what why, which is a thing we ship today on trust. On being convinced without, like, extension incubation. + +JHD: Would have—am I understanding correctly if we had a way to have a longer incubation period before native browser implementations, and that also created adoption when applicable, would that also address your concerns? I know there’s big ifs. But… + +SYG: It’s a psychological thing. It’s important to not frame it as actually important enough frame it as incubation because you are then priming people to never adopt. + +SYG: JHD to bluntly answer your question. I think the world I want requires us to change our thought. Like big T Thought of developers and users. But that’s not possible. So with most things in the world, best we can do is to change the incentive to move towards the change in thought that we want. And we can reasonably—people can reasonably disagree on whether they will a bear out or not. The structure we have had for the past 5 to ten years and the consequences from that, that are not ideal. If the question is like, well, if we do this other thing, would it solve your problem? Yes. If everyone in the world behaved as you proposed, but we believed differently because of things we see. Yeah. You get the point. + +OTM: Yeah. I guess my question was, I agree, but most people making websites using tools nowadays, but people using something like—most people don’t use tooling unless they are using TypeScript or something. Should Node be a transpiler? Or… + +SYG: So Node recently merged support for type stripping, which means they are running something on the input before it feeds it to V8 to run. Deno and Bun run TSC. The way I see the server runtime go, they are incorporating auto transpilation as part of the runtime model. + +CDA: People use tools, but not always, and not all the same tools. Some don’t use transpilation tools and I want to push back against the notion that this would simply be another thing you add to the pipeline and the problem is solved. It’s more complex than that. + +SYG: So CDA, I would welcome data, I guess. We looked at the best data we had and that’s what showed of the developer demographic. + +CDA: Where I have heavy tooling is the least preferred developer experience/environment of the stuff I do. I actively try to minimize the amount of processing that has to be done. And I very much enjoy the experience where I don’t have to do anything extra. But that’s one person’s experience and not the entire ecosystem. + +YSV: JSSugar is looking at one facet of the problem, which is the syntax side. But more broadly, how are we introducing new things into the language? How we benefit things in the I think beiges, I call it a waiting room for features. Developers can start using the feature. And we can get data about how the feature is being used, we have the rigor and dedication to backwards capability. And basically, the same way we do things. I still have more time to bring this proposal to committee and once it’s ready, this is something we can also discuss. But just to make it really clear. JSSugar is not intended to solve the entire scope of the problem statement we presented. There’s room for others to bring their suggestions. This isn’t the only way to approach this problem + +NRO: Yeah. You mentioned that JSSugar has to be designed, keeping in mind lack of experience and making sure that work with the maps. There’s very far from being something that can be transpiled. Like we don’t know yet—like in which to improve because there’s so many. We need a little bit of time. I don’t know what your ideal timeline here would be, but I don’t think we can answer any time soon, if you want this JSSugar, JS 0 to be from developers + +SYG: I would say Yulia’s topic may be directly related to this. I will let her go + +YSV: So we are standardizing source maps and that’s good. They are shipping in multiple implementations. But I would also say that source maps should not be our final destination in terms of debugging. We need something with better scalability with what source maps offer. Source maps are in JSON. We need a different format for—by which to emit it on the bugging so it’s lighter on the wire. I support standardizing source maps. There is a defunt group. That was intended to be for both Wasm and JavaScript and this is something that Aki would be considered as a coordination point between W3C and TC39. We need a solid debugging format that can handle multiple languages for two different targets. Wasm and JavaScript. + +CDA: Just want to note, we have ten minutes left for this tonic and an enormous queue. Be mindful, in you want to look at the queue and any you want to, because we won’t have time to get to all of them, but take a look and go to CM + +CM: Another thing that struck me as you were laying out the proposal for sort of layering the language structure in the way that you have, is that Wasm, as it has manifested in practice, is an emergent and certainly less self-aware form of the kind of thing that you are proposing, responding directly to a lot of the underlying considerations, particularly with with respect to performance, that you outlined. That suggests to me that there is something going on here that if we don’t respond purposefully and directly, the ecosystem at large is going to push it on us anyway. And so I think it behooves us to try to take control of the process ourselves to the extent we can. + +SYG: I will respond to the Wasm thing because this comes back as feedback. Why not Wasm? Well, it would be great if JS just like is compiled directly down to Wasm. We see a big part of the signal with the excitement for the Porffor thing we saw. But JS is just special. Like we still have to service three billion users today. And we are continuing—already bearing the cost and spending hundreds of engineers hours on three, at least three independent highly sophisticated VMs. In 20 to 30 years, who is to say? Today, it’s hard to argue that. + +CM: I am not proposing any such thing or advocating for it. I’m saying we should get in front of it. + +SYG: Got it. + +CDA: Are there any queue topics you'd like to move up? + +SYG: I… I don’t know. I think maybe move number 4 Nicolo’s topic up. This is important to hear direct feedback from the tools and how they see the world. + +CDA: Okay. So in a similar way this is not transparent to developers this is not transparent to the committee. Right? You mentioned things in a proposal would perhaps be in JSSugar and some things in JS0, so now, proposal authors are also not developing the proposal in the same way. They have to make the considerations and deliberate on which things will be on which side. Is that correct? + +SYG: Yes. If the split existed in the spec, part of the spec draft's major semantic past, if it's an explicit open question, called out, yeah. The thing that spec authors have to think about. NRO? + +NRO: Okay. In the past year, I would say, 6 months, I have tried recruiting two people to join our committee. I very recently managed to propose invited experts. But I’ve been doing this for longer and the risk of communicating has been they are interested in looking at what we do. But whatever proposed structurally, experting joins our meeting, they see this as a waste of time. As a place where to get anything done, you need to spend years. And there can be just more productive ways of spending time improving the tools. Especially given that most of them do this in their free time. So while I will keep trying to get more involved, I don’t know how he can get this done. There’s been multiple attempts. The meetings end up rarely happening because there was never enough people to actually join them. + +NRO: And I hope we will be able to improve this. It’s the—the current situation is far from being good. + +SYG: I don’t think it should fall on you, it should all on all of us to do that better. I don’t know how either. But I think tools are—it is true that they are a big part of the JS ecosystem. And we should involve them more. Regardless of, you know, this proposal. + +KM: I have heard from various people, historically over my time with the TC39, they would prefer to do things in other standards bodies because it’s much harder to get things through TC39 and I think part of that is potentially good and also a potential risk point for JSSugar. In the sense that, like, if people feel that it’s too difficult to go through, TC39, with the JSSugar, that could mean that there won’t be enough mind share in the JSSugar world to kind of, like, give it the critical mass it needs to be effective. So I mean, I guess that is some of, like, my personal reservation around it. And I don’t know. Like I would hope that if we—I don’t know, tried to go forward with something like JSSugar, we could come together collectively and figure out a way that is easier. That is the larger risks, in my opinion, of any JSSugar proposal. + +JWK: Actually, I support this. I even think we should do this years earlier. And this year we are—a lot of proposals are suppressed by the size of complexity. But they are useful for the developers. And if we have this, we can—going to experiment this in quite a formal way. And another benefit is, for example, if we ship all old decorators, it will be there forever. Nowadays, the tools are pushing developers to migrate to the new proposal. So although it’s not very easy, but it’s possible to correct things, if we later find that the proposal is—has some problems. + +CDA: All right. We have 2 minutes left. Maybe get to one more. + +SYG: Okay. Yeah, maybe one more question and then I will try to sum it up. + +NRO: Yeah. So I have the impression that a big reason for why this presentation—this presentation is happening is that TC39 is just like sending features to be implemented to browsers at a speed that is higher than what browsers can sustain. Like implement them with the performance and this means browsers don’t have enough, like, money, engineers to keep up with our speed. However, if we move this problem tools, this is just going to get worse. Like, tools—some tools do have—TypeScript. Most are implemented by people doing it in their free time. For a while, they had some amount of donations compared to the tools was significant. And that led us to pay—that’s the amount of money going to the engines. This is—this is practically zero if we are trying to shift the burden from browser engine developers, it even has less resources. This affects the stability point of maybe we can make changes in JSSugar. Some tools, but I think of ES lint, some tools wait to implement proposals until they are ready to ship to reduce the number of potential changes that will happen and the meaning—and maintain the cost. It’s like the opposite of the hope that maybe we can have more flexible proposals if we ship in tools, the browsers. + +SYG: Do I have like 30 seconds to respond to that? + +CDA: Sure + +SYG: Thanks for the insight NRO. I can’t substantiate the landscape of the tooling maintainership. A lot of the tools are open source projects with volunteer maintainers. One way to frame what Nicolo said for the committee is that if well-funded, relatively, but also in absolute terms, well-funded browser teams don’t have the risk appetite and tools don’t have risk appetite because they are less funded. I hope the committee sees that as a failure. I hope that like the fundability and the importance of funding tools for—we can somehow move the needle and shift the incentives there, that they can reach the level of funding the browser engines given their rise in importance in the ecosystem. I don’t know how we can make that happen. I would like that to happen and want to try. But yeah. To sum up, there is—we presented a problem statement. This is as you can see, this is not a proposal, in the normal sense, there’s no concrete details for the JSSugar/JS0 idea. There is currently a private repo, I will make public at the end of day or tomorrow for folks to—for folks to post issues on. + +SYG: But I think my request, and I will let the other people on the slides speak for themselves, take the problem statement to heart. Think about the JSSugar and JS0, if you have a great idea, please bring that to a future time there. As for some urgency here, you know, things don’t move too fast. But I think we do need a shift in our MO and let’s keep talking about it. I think the most concrete take away for individual proposal champions is that if you have issues, if you have concerns, if you have questions on how you think the proposal would look with this new rough guidance we posted on the slide deck, reach out because like I said, I don’t think it makes sense to make a blanket judgment. We need to do this case by case. Is that okay or too abstract? There is a private repo, discuss it. There’s no broader discussion repo for this problem statement. I am not sure how we do that. + +### Speaker's Summary of Key Points + +- Implementers presented a problem statement from their perspective that new features often cause security, performance, and stability issues for end users and that those downsides are not offset by the upside to developers. Because of that, that they will be more conservative in considering features going forward. +- SYG presented a possible future that splits the standard into two layers so as to more evenly spread out complexity: an underlying layer called JS0, which is directly executable that compliant engines implement, and a top layer called JSSugar, which compliant tools implement and desugars/compiles to JS0. +- Delegate reaction was mixed. Some delegates liked the idea, and agreed with the implementers' cost analysis. Tooling authors expressed concern, specifically around the cost to maintainers. + +### Conclusion + +- Champions requested delegates take problem statement to heart. +- No consensus on JSSugar/JS0 (there was no plan with concrete specifics anyhow) +- Lively discussion which will continue in the future + +## Three normative Ecma-204 changes + +Presenter: Ben Allen (BAN) + +- [Ecma-402 PR 919](https://github.com/tc39/ecma402/pull/919) +- [slides](https://docs.google.com/presentation/d/1Z-DQ3bNjrbhrPhJ9PMEjTtZr0tlXiN14REvtcPocdGM/edit?usp=sharing) +- [Ecma-402 PR 918](https://github.com/tc39/ecma402/pull/918) +- [slides](https://docs.google.com/presentation/d/1Z-DQ3bNjrbhrPhJ9PMEjTtZr0tlXiN14REvtcPocdGM/edit?usp=sharing) +- [Ecma-402 PR 925](https://github.com/tc39/ecma402/pull/925) +- [slides](https://docs.google.com/presentation/d/1Z-DQ3bNjrbhrPhJ9PMEjTtZr0tlXiN14REvtcPocdGM/edit?usp=sharing) + +BAN: While the screen share is up, there’s three normative PRs. I am hoping it will be relatively controversial. And the order is like more interesting ones sandwiched between two bugfixes. + +BAN: So yeah. The first one is a small spec bug. Essentially for quite some time we have been neglecting to use the correct numberingSystem when that option is specified. The second one concerns sort order. We are trying—it’s in response to one guidance from about three years ago. We are tying to make the order in which the elements appear deterministic and this is related to that. And the third one, again, is another fairly minor bugfix Here is the first one, as promised. + +BAN: So it is a spec bug. There are a couple of different places where RelativeTimeFormat and daytime format create NumberFormats, but don’t pass in the numberingSystem used. So in this example, implementation follows the spec, and if someone requested the number system, Hannah deck. This spec was specced to simply return with the, like, Latin character set number instead of like going… So yes. In the spec it was ignored. No browser actually implements it as specified. So this PR just sets the spec behavior to be intended/implemented behavior. + +BAN: It is essentially a two-line fix. Something like that. If I am not still sharing, let me know. + +RPR: You are sharing your web browser and going through the PRs. + +BAN: Fantastic. There’s several places where we are adding options that we simply neglected to add. This is point where we ask for questions and after, ask for consensus + +RPR: SFC is on the queue with a message. All three of these PRs were discussed in TG2. And approved by that group. + +BAN: Yeah. They were following guidance that we received from a plenary we got three years ago. + +BAN: Yeah. Nothing more. You are going to ask for consensus one by one? + +YSV: So unfortunately, this was added after we did our review. I don’t want to say that we are going to block this. I will try and get someone to look at this before the end of the meeting. But I believe that this is going to affect everything you are presenting right now. I am sorry that that is the case. But we will try and get—I will make sure that is completely fine with our folks working on internationalization before the meeting ends. It’s a conditional advancement on my part and I'll get back to you before the end of the week. + +BAN: Thank you very much. + +CDA: So we will say if we have support from the rest of the room, no other blocking concerns, then we would approve conditionally pending the team’s review + +BAN: I was to say, legitimately thank you for sort of sticking to the process there. It’s pretty important. + +BAN: Okay. So without further ado… This is the one (PR 918) that I believe is more interesting. Essentially, previously, the result options for plurals, we did not specify an order. There is a sort of like web reality order that is—I believe all the web engines use this specific order that is ultimately kind of nonsensical in a way. Also, we have guidance from plenary to—we must have deterministic orders for this sort of thing. + +BAN: So this is the wording we said based on that guidance from plenary. ECMA402 must provide the deterministic order for the elements of all arrays and properties of all objects. This order should be lexicographic except in the cases for which there is a clearly preferable semantic ordering. And there’s a link here to the notes from the meeting where this is previously discussed. + +BAN: Okay. To give the context on why the order proposing is useful: so plural give information on how to correctly plurallize different numbers in different languages. In English, if I say, talking about cats, and I say I have 0 cat, I use the plural there. I have one cat, I use singular. And for subsequent ones I use the plural. English has two plural categories. One for one thing. I apologize in advance. There are moments where I pronounce things in French. It will be a real adventure. Apologies to anyone whose ears are hurt French is a little bit more complicated than English in terms of whether you use a plural for specific counts of things. This applies to ordinals. I am guessing in the context of cardinal numbers. So French uses singular when talking about zero objects or one object. Plural in most other cases. But there are certain large round numbers where you get DE in front of it. You have 0 geckos, you have zero geckos. 100 geckos, you have cent geckos. And one million geckos, you have Un million De geckos. So there’s one, many and other. + +BAN: There are six plural categories. I believe this covers all languages. But that’s something that you can say confidently. The order that UST35 uses is 0, 1, few, many and other. And you can actually see if you are thinking about ordinals, you can see how in English, how one, two and few work. If I am talking about the first thing, first. St. Three, RD, that’s few. There is no use of many in English. But every other is—has the Th ending. That is classified as other. This order feels natural to people. It's an ascending order. + +BAN: Our guidance is to sort in lexical graphical order, unless there’s a reason not to. We have the following few, many, one, other, two, and zero. Which is scrambled, it puts others in the middle. It kind of makes it harder to understand what we’re talking about here. And also, it’s contingent on features of the English language. But so currently, the sort order is left up to the implementations. We don’t specify it. It varies with regard to the ordering that is used. Web reality appears to be, I am not sure how everyone standardized on this. Few many, one, two, and zero and other. There’s a comment from Frank where you tested this on a number of engines if you want to follow that. The PR swaps that language that doesn’t specify an order with language. Specifically, the order 0, one, two, few, many and others. It's an ascending order. Other, covering most numbers is put at the end. We have an older PR or older issue, for thoughts on why it’s important to sort these in an easily order. Users often use this API exploratively and putting it in the UST35 order makes it easier for users to learn which categories are used by which languages. + +BAN: I am going to flip over to the next one which won’t take very much time. Fingers crossed for conditional acceptance. This is another one that. We have overlooked some of the logic for handling the formatting of currency values. It makes perfect sense when using a standard notation, when dealing with money. But in scientific and engineering notation, it is just odd. So ECMA402 is drawing on ISO 4217 data to determine the number of minor units to display when formatting currencies. I say currently because there’s another PR I will discuss on day 2. We’re considering changing this. + +BAN: So if you are formatting the quantity of Japanese Yen, formatting 123.45 because this data is all data in practice, you don’t use every day—you simply don’t use the minor violence when talking about Yen. USD, you use two digits. And this ISO 2419 is what we’re currently normatively drawing on, even though in practice, they use CLDR practice Here is the problem and bug we overlooked. For example, this number is right there (`12345678`)… if formatting that quantity of then, it will always display without any minor digits. Which doesn't necessarily mean like a fraction after Yen in this context. If we are formatting USD, it uses 2 units. By chance, this is what you want there. But it’s not the right thing to do. The thing that we are using to determine that there should be two units after the decimal point is specifically and exclusively for the standard formatting, rather than scientific formatting. This also applies to the engineering notation and contact notation. + +BAN: So the solution is not don’t to that. Only consult the data on a number of minor units to display when using a notation. It’s a straightforward three-line change. Essentially, just—okay. We don’t actually call the currency digits A0 that consults that data unless we are specifically using standard notation rather than any other one. + +BAN: Okay. Thank you. I believe at this point, I would like to—unless there’s questions, I am guessing there’s not… + +CDA: There is nothing in the queue. + +BAN: Okay. I guess I would like to ask for that conditional acceptance for this one. I would like to ask for consensus and receive conditional acceptances + +CDA: Consensus for the three normative changes? + +CDA: Other than the conditional support signalled by my Mozilla friends, any other voices of support for normative changes? + +CDA: Any objections to the normative changes? + +CDA: All right. So it’s all riding on Mozilla now. + +BAN: Thank you very much. I would to say again, even though it’s kind of put me in an awkward position, I am really glad that you all are sticking to the process on this one + +### Speaker's Summary of Key Points + +- List +- of +- things + +### Conclusion + +- List +- of +- things + +## Exploring an Idea of a Proposal Management and Technical Arbitration Tool + +Presenter: Mikhail Barash (MBH) + +- [slides](https://docs.google.com/presentation/d/1gHsElv4Mfs90wxBnft-1cu3KgkGLILeP0BRxe0WR01w/edit?usp=sharing) + +MBH: So this is one of the directions that TG5 is exploring. So in the TG5’s programme of work there is a point that we are expected to look into tool support for the work of the committee that aids in design, specification, and technical discussion about the language evolution. So basically, the idea of the tool that I am going to demonstrate now is to provide a space for structuring a language committee’s work and making it more transparent to a broader community. And the tool is supposed to cover both technical and administrative aspects of the committee’s work and I will soon demonstrate a mockup implementation, so it’ll be clearer what I mean. In the first part of this slide, I mentioned a committee. In practice, this is based on the TC39’s experience. So this is not a replacement of any of the currently used tools. And rather, this assumes full integration with the tools that are currently used by TC39. So this is just an exploration of the idea. Not trying to change the way the TC39 committee works. + +MBH: So let me show a mockup of the tool. Here, we can see differences—I would call these specifications of the committee work. It has these roles, these persons, and this person has these roles and we have full-blown IDE support here, like code autocompletion, find usages, and so on. If I rename this role, then the refactored name will be propagated throughout the file. + +MBH: When I try to add a new person, I get a pre-filled template of what can be entered here. So then I can specify meetings, agendas, templates of agendas for meetings. This is an example. And, say, if I change the duration of this item, then the total duration is automatically updated in the summary section. And I could, in principle, do scheduling algorithms to find an optimal allocation of slots in the meeting. + +MBH: This is supposed to be a specification of the TC39 process or of a language process. So for now, it’s just a list of stages. But this would be a much more elaborate specification. + +MBH: Then the interesting part happens when we try to specify proposals. So this is a sample proposal here. So I could specify the stage. This proposal is in one of the stages—of course, I only get the stages that I have already declared before. I could mention champions, reviewers and again I have full-blown IDE support here. I can specify related sections in the specs. I can specify related proposals. + +MBH: So this is an example of a syntactic change. But I consider three kinds of changes: API changes, semantic changes, and syntactic changes. An API change, I would get here another view where I could add methods and could add arguments. I could have alternative designs here by different persons, they can also leave comments here. The tool should support all kinds of administrative things which are dealing with technical design—which I call technical arbitration. Say here, I have the proposal Set Methods. If it were a syntactic change, I could introduce here the new rules that the grammar would get and the new rules. + +MBH: This is a mockup to give you an idea of what in principle is possible. What if I add a new stage and advance a proposal? If I remove some proposal, what would be the relationship between the proposals that are left in the language after I had removed this one? I could combine proposals, but this requires scientific research into what it means to combine two proposals. You could preview all the changes in the language specification. + +MBH: And then interestingly, I can also use this tool to author a specification. So for now, this is just some of the sections in the document. And this is a projectional editor for ecmarkup. Let me show how this would be—how I would type this. This is the phrase “if something is not an object, then throw an exception”. I typed an abbreviation for this phrase, and got the rest of the phrase added to the spec text, and I can only modify the placeholders in that phrase that it makes sense to modify. And again, I have full IDE support. If I click here, I get to the definition. Yeah. Let me show one more example. I type “return unused”, and everything is formatted automatically following ECMA-262 formatting guidelines. + +MBH: And this is a projectional editor which means that I can only modify the parts that are designated to be modified. So I cannot type anything here, for example, because this would be invalid. Yeah. So let me just show this one. So how do I get line 1. Type “let this”. And then I selected here, new name for that identifier and basically, this tool will support a special syntax for every line that is used in the spec. This one, I would like to show a little bit in more detail. So if I have this phrase: “if something, then set something”, I can define a quick fix that would, for example, remove the condition. And now this is projected without the condition. + +MBH: Here, I have this abstract operation “question mark”. And I can also sort of toggle the usage of this operation. There is no need to parse anything. Which means, I can re-project this into the spec text in a different language. Here, I have an example of a sort of different projection which I call educational. Instead of having a question mark here, I now have a comment here that explains the purpose of this abstract operation. + +MBH: And again, everything is sort of clickable and—yeah. Presentation. So the target users for this tool would be chairs, technical secretary, facilitators, specification editors. And delegates, if they want to use this tool as an authoring tool. But also a broader language community, if they want to use this as a sandbox for exploring their own feature designs. + +MBH: So this is the functionality that I have demonstrated. And we can think about different kinds of artifacts to import and export into and from such a tool. + +MBH: So as I mentioned, this would be fully integrated with GitHub and all the rest of the things that TC39 uses. Detailed technical arguments happen in GitHub issues as this tool is likely to fixate milestones of technical design. And here on the right, you can see some of the examples of such functionality for technical arbitration. + +MBH: Syntactic proposals, think about grammar validation, grammar debugging. Live preview of the language changes for API proposals and for some of the syntactic proposals. We could think about previewing those proposals directly in the codebase of a user. + +MBH: So as I’ve already shown in my demo, this tool can be used as an authoring tool. Quite recently, WebAssembly has set up mechanized SpecTec. The tool that I suggest focuses on language design and evolution processes itself. But it’s in the same line of thought as WebAssembly SpecTec. We can think about features like validation of text of the specification at design-time, navigation within the spec at design-time, refactorings, executability of the spec at design-time. Using the approaches from the live programming environments where you can directly preview in the IDE how a particular piece of code would be executed. And that piece of code is written in ecmarkup. Integration with other tools. This is a potential collaboration with other bodies. Ideally integration with ESMeta. The suggested tool can be thought of as an IDE for the spec. So, the focus is on the editing experience and support for notation that is already used in the spec, including textual, tabular, and graphical notation. All of this can be implemented for the tool. + +MBH: All information still will be in the GitHub repositories and other tools used by TC and the model of all of these administrative and technical discussions specifications, that is defined in the tool I am proposing, can be serialized, so there’s no vendor lock-in. The tool will support evolution of itself within the—within the tool. So there will be no need to recompile anything. + +MBH: So this tool would be making the work of the committee more structured. This is specifically tailored for the programming language evolution, with the idea to increase transparency of technical design and the rationale of decision-making, as well as increase accessibility of following what is going on with the language evolution. This would help newcomers to get acquainted with the state of the art language evolution. And we also think in the future for evolution across layers. A tool for fixing technical communication. The initial step is to implement the easier parts of what I am suggesting and use the tool in a “view only” mode, to explore the already existing specifications and see whether that could make sense. + +MBH: So what is the ask here in my presentation? To reiterate, I am not trying to change the way that the committee works. Exploration of the idea. And what I am looking for is feedback on this idea. So am I missing something that is fundamentally important for such a tool? That’s the end of my presentation. And I am ready for the queue. + +NRO: Yeah. This is very cool. Like, one problem I have had in the past very often is to like forget a proposal, all the time it was discussed and find the discussions, and I hoped that this would be a way to solve this. + +MF: Yeah. I can’t stress enough how cool this is. This is really fantastic. I really like how you have tried to account for all of the different processes and responsibilities that the committee has and try to not tether yourself and think about things in a fresh, new way. I think it’s great to have something like this. The downside of having a broad scope and being radically different from what we do today, it makes it hard to get from here to there. For any kind of new approach to anything. Have you thought about how any SSO can gradually adopt this and how that would work? If not, that’s an area to flesh out. And share with us. + +MBH: Right. So this last point on this slide, to use the tool as just an explorer of the current specification, that might be that first step. And maybe individual delegates could use the tool for authoring the specification just to see how it would work. + +MF: Yeah. And—that’s not just looking at the first step, and the second step. I want the whole route to getting the full adoption to this would look like, in theory. + +MBH: That requires thinking. + +MF: Yeah. + +JRL: So how do we use this? + +MBH: So for now, this is implemented in the JetBrains Metaprogramming System MPS. It’s based on the IntelliJ platform. But different, because it’s not a textual editor, but a projectional editor. But this could be shipped for—as a standalone application. + +JRL: If we make the standalone application, does that include all the necessary—I won’t have to install Java libraries to get it to run? + +JRL: I do not have Java installed due to corp laptop reasons, so I don’t have it and cannot install it. Is there a way it make it a single executable + +MBH: Well, there are some attempts to actually run this tool on the web. The JetBrains Metaprogramming System. I think there were some implementations that allow running a server and a client in the browser who could use that without installing any of the dependencies. + +JRL: okay. + +MBH: But I am not sure if it works as smoothly as a standalone application. + +CDA: All right. That’s it for the queue. But as I mentioned at a TG5 meeting when I saw this, this is super cool. I can only pretend to understand what is going on here because it seems like magic. I am excited to see what this looks like going forward, what it does in the future. Spec thank you. There is nothing else That is it for today. A round of applause for everybody. + +### Speaker's Summary of Key Points + +- Presented an idea of a tool for supporting the work of the committee that aids in technical discussion and authoring of proposals. +- The proposed tool can be thought of as an IDE-like workbench for language specification and standardization, including both administrative and technical aspects. +- The tool would be integrated with the other tools currently used by the committee (such as GitHub). Once the key functionality of the proposed tool is implemented, it would be used in the "view-only" mode to follow currently discussed proposals. It's important to have a concrete approach/plan for a potential gradual adoption of such a tool. + +### Conclusion + +- Got positive feedback from the committee on the idea of the tool. The work on the tool implementation continues. diff --git a/meetings/2024-10/october-09.md b/meetings/2024-10/october-09.md new file mode 100644 index 00000000..61feba00 --- /dev/null +++ b/meetings/2024-10/october-09.md @@ -0,0 +1,1483 @@ +# 104th TC39 Meeting | 9th October 2024 + +----- + +**Attendees:** + +| Name | Abbreviation | Organization | +|-------------------|--------------|-----------------| +| Ashley Claymore | ACE | Bloomberg | +| Andreu Botella | ABO | Igalia | +| Linus Groh | LGH | Bloomberg | +| Chris de Almeida | CDA | IBM | +| Dmitry Makhnev | DJM | JetBrains | +| Marja Hölttä | MHA | Google | +| Waldemar Horwat | WH | Invited Expert | +| Jordan Harband | JHD | HeroDevs | +| Devin Rousso | DRO | Invited Expert | +| Daniel Minor | DLM | Mozilla | +| Yusuke Suzuki | YSZ | Apple | +| Mikhail Barash | MBH | Univ. of Bergen | +| Keith Miller | KM | Apple | +| Michael Saboff | MLS | Apple | +| Justin Ridgewell | JRL | Google | +| Andrew Paprocki | API | Bloomberg | +| Yulia Startsev | YSV | Mozilla | +| Bradford Smith | BSH | Google | +| Ron Buckton | RBN | Microsoft | +| Nicolò Ribaudo | NRO | Igalia | +| Jonathan Kuperman | JK | Bloomberg | +| Jesse Alama | JMN | Igalia | +| Istvan Sebestyen | IS | Ecma | +| Chip Morningstar | CM | Consensys | +| Richard Gibson | RGN | Agoric | +| Ujjwal Sharma | USA | Igalia | +| Ross Kirsling | RKG | Sony | +| Oliver Medhurst | OMT | IE (Porffor) | +| Philip Chimento | PFC | Igalia | +| Eli Grey | EG | Invited Expert | +| Jack Works | JWK | Sujitech | +| Daniel Ehrenberg | DE | Bloomberg | + +## `Intl.DurationFormat` do not include grouping separators in numeric-like styles + +Presenter: Ben Allen (BAN) + +- [proposal](https://github.com/tc39/proposal-intl-duration-format) +- (no slides) +- [PR](https://github.com/tc39/proposal-intl-duration-format/pull/198) + +BAN: All right. Hello, everyone. So I’ve got a couple of things to present. One is a fairly small normative change/educate bugfix for DurationFormat. The other is somewhat more substantial and requires discussion, though possibly not a huge amount of discussion because previously, for currency formatting, or currently for currency formatting, we use ISO4217 data is normative. There are good reasons that I will discuss that we shouldn’t make ISO 4217 normative for the thing we are using to are, which is the number of minor units when formatting a currency. But that one is a little bit more substantial. + +BAN: So this is the DurationFormat one. As I said, it’s fairly small. So one of the sort of quirks with DurationFormat is that we kind of have two fundamentally different ways of rendering durations and sort of most of the most recent or educated fixes have related to the digital style. So the digital style is used to format durations in sort of as displayed on a digital clock. But we realized, I think, it was Frank who realized this, so DurationFormat allows sort of like very large numbers for number of minutes, and number of seconds, when displaying in the digital clock format. It doesn’t convert seconds up to hours and then—or could be up to minutes and minutes up to hours. We realized that if you have got a very long—very large number for seconds, the seconds unit, the minutes unit or the hours unit, it would insert grouping separators. So you know, in the locale EN, it’s putting the comma characters between Septembers of three. Which is ugly. It breaks the digital [ metaphor dramatically. So the PR is very small. It’s in several different places when formatting, digit hours, seconds minutes. So when setting the options for the NumberFormat we use, when saying the option, we currently use grouping to false. So that it would render—it renders just as the, you know, without these grouping separators here. + +BAN: I will flip over to the PR to see what’s changed. If today not changing in GitHub let me number off in three grouping separators, what is rendering hours, minutes and seconds in the digital clock format. + +BAN: So I guess without any further preamble, it might be useful to ask for consensus on the PR. + +USA: There’s nobody in the queue yet. + +BAN: I figured this one would be somewhat uncontroversial. We are hoping that this is the last normative change that we will have to make with DurationFormat before Stage 4. We have two implementations now, I believe, correctly, and we are extending some tests through to reflect recent normative changes. + +USA: Okay. Would anybody like to register explicitly with support? I don’t hear anything on that either. + +YSV: I will just jump on the queue because yesterday, I had said—hadn’t had a chance to see the normative changes, just because of a no recite. I gave conditional consent there. Everyone has gotten back to me and it’s totally fine. I am the only person who missed it + +BAN: No. And the thing I said yesterday about, like, it's a very good thing to be, like, salt—to do the process properly, I mean it. It brings professionalism in the meeting is fantastic, even if it results in me being in an awkward position. Okay. + +USA: Also, on the queue, we have Ross with support. So let’s move on to the—oh, before that, Ben would you like to summarize or have any conclude remarks? + +BAN: Yes. + +### Conclusion + +We have consensus for the normative change removing grouping separators when using the digital format in DurationFormat. + +## Normative: Allow use of non-ISO 4217 data in CurrencyDigits AO + +Presenter: Ben Allen (BAN) + +- [PR](https://github.com/tc39/ecma402/pull/922) +- [slides](https://docs.google.com/presentation/d/1ryT6rRTxfb8j4BJl9OX92LGkuNiZBwJiRj7pN0vUXTI/edit?usp=sharing) + +BAN: Okay. So as I said, this one’s fairly—this one’s more substantial. It’s actually kind of the result of a long journey. I believe, like, almost a year ago, it started as a proposal to make CLDR’s data normative in certain cases when formatting currencies. + +BAN: And it resulted in overtime being down until we realized that actually we should make either—neither ISO 4216 nor CLDR currency normative. + +BAN: Now I will actually go through the slide show, again let me know if it’s not sharing. So this comes up when we are formatting currency, like I said. Different currencies have different number—different numbers of minor digits, after the decimal point. If you are formatting USD, you will want to use two units after the decimal point to represent pennies. If you are JPY, there are no minor units, no fractions of Yen that are typically displayed when formatting numbers. So you shouldn’t include decimal in those. Currently, the spec text for this aspect of rendering currency makes ISO 4217’s data normative for determining the number of digits to use. + +BAN: Before I go any further, I do want to note that making this change reflects web reality even though ISO 4217 is normative. Different I want takesses, some use CLDR and have good reasons to use it. + +BAN: Okay. Basically, I will go through sort of like references to several conversations both at the CLDR level at this, and on the PR, on why it’s a good idea to make ISO 4217 non-normative. + +BAN: So CLDR’s data on the number of digits differ for many currencies from the ISO data. And about essentially, as we will see again further in the presentations, essentially, the ISO data is kind of pedantic. It says you should format currencies with far more minor digits that are actually used in common use or in any use whatsoever. + +BAN: So this was DE weighing on this one. He notes that ISO is going for more legalistic definitions and CLDR is doing what is more linguistically relevant. Like what you would encounter if you were actually like typical encounters when talking about that currency. + +BAN: And note that, V8 is using CLDR’s data currently. SpiderMonkey has been using the ISO 4217 data which is technically normative. But that’s what we’re considering changing. + +BAN: So I love Norbert Lindenberg’s description on the ISO 4117. This is indeed legalistic and sometimes detached from reality. He reached out to ask about the Indonesian Rupiah. Having 2 DecimalDigits. 50 is half a US cent]. And basically, the minor unit is sometimes used in accounting and sometimes used in financial transactions. And so ISO 4217 has decided to have those 2 digits after the decimal point. Or after the decimal separator because there’s one context in which that might be used sometimes. + +BAN: Something that we have verified is that this data is in fact supposed to be different. Thanks to SFC for reaching out to CLDR to verify essentially their data, they look at the ISO 4217 data when determining the number of decimal or minor units digits they specify. But they update that based on facts on the ground. And yes. There’s a general consensus that ISO 4217 values are pedantic and somewhat disconnected from reality. + +BAN: Which is best? On a PR related to this matter, the one suggesting making CLDR normative, it’s noted that essentially varies based on the specific application which data source is a better choice. And through some applications neither is the better choice. Right now, the spec is specifying one data source, that data source is useful forever for some applications and not really the correct one for others and web reality is that different applications use different data sources. + +BAN: Okay. So here is the wording of the key part of the PR. Currently if the ISO 4217 currency and funds code list contains coerce occurrence coerce as alphabetic code, return to the minor unit value corresponding to the currency from the list. Otherwise return to 2. The new wording uses a topic what data used to determine how many decimal digits it uses. The new wording we workshopped a fair element was to return a non-negative integer indicating the number of fractionalDigits used in formatting the quantities of the currency corresponding to the currency. If there’s no available information on the number of digits used, return 2. + +BAN: So the wording presumption, you may or may not have data on how much minor units are used, and if you do, we encourage you to use it, without specifying any particular data source. + +BAN: That’s it. Yeah. I would love to answer any questions. This is one that’s been a fairly long time coming. And the realization that we shouldn’t privilege either data source is what TD2 is converged on. This does have TG2 approval for whatever that is worth. + +USA: All right. On the queue we have Shane. + +SFC: I wanted to note that we asked CLDR about this, this behavior is actually the behavior that is intended here. And just yesterday, they merged a change to the spec, which yeah. As you noted here, quote this—the value of this field is based on the minor university value of 4217, because it may deviate from ISO 4217 where there is compelling evidence for different customary practice. So that’s the new text that—its specification, which means, this is accurate on your end, to make such a change. + +BAN: We have got confirmation that they are not supposed to be tracking the same data source. Not supposed to be the same values. + +DE: I am very happy with this change. Thanks for following through on this. I wanted to ask, if there are use cases for different—for both data sources for different applications, should we make an option that lets people opt into the more formal one? + +BAN: So this—the reason why I sort of hesitate with that is that the—the more formal one, there are use cases for neither the formal or the facts on the ground one. So it’s a matter of, like, opting into one or the other of them or some things. + +DE: Sorry. I didn’t understand the answer. What was the reason for not exposing an option to get into the formal one if you want, and only just letting the implementation decide? + +SFC: The more formal version does things like use 6 decimal places for the Iranian currency, which is one of the examples for why this doesn’t work. Because, like, not since the 1970s has anyone used 6 decimal places for that currency. But that number hasn’t changed because it—but CLDR has kept its data up to date with how the currencies are used on the ground. So it’s not like this is formal versus informal. It’s like this is, you know, one spec versus another spec where the CLDR is what focuses on what people use. + +BAN: In some cases, the ISO 4217 data can be considered not more formal but suited for financial uses, using minor digits, if there’s the faintest hint that some financial institutions accounting practices might use that many digits. + +BAN: But the sense is that in some cases it’s more formal. It’s more something you use in a financial institution. And in some cases, it’s just wrong. Like it’s not that it’s the formal version. It’s more digits that anyone uses. + +DE: Okay. So there might be use cases sometimes. But it isn’t good enough to be a good thing for—to have an automatic option for. Okay. + +USA: On the queue we have Waldemar next? + +WH: You say there are some uses for financial purposes that use more digits than CLDR. That’s true. But there are common uses that use more digits than even ISO 4217. ISO 4217 lists two decimal digits for USD, right? + +BAN: Right. Right. Correct. It doesn’t specify using mils. + +WH: But every day I see USD prices with three decimal digits at gas stations. + +USA: Yeah. Which I see is a reason to leave it entirely implementation-dependent. + +DE: I don’t think gas stations are reason to leave it implementation-dependent. Where this is a default precision. You can always format currency amounts in any precision you want. But all implementations should have two digits for US dollar formatting. + +BAN: Yeah. + +USA: All right. Then we have Shane next. With a reply. + +SFC: Yeah. So there are different—I mean, the gas station example, is one. But then there’s also, like, you know, CLDR specifies another set of rounding for your cash transactions, for example, which is different in certain locales that only use nickels, for example, for cash transactions. But they would allow granular for more electronic ones. There are several different contexts here in which these digits could be used. And, like, the purpose of this—of this value in ECMA402 and CLDR is to be the default formatting for currency, for that specific currency. But you know, we are definitely not in the business of converting currencies and not in the business of, you know, like, trying to pretend like we know every use case for currency, because it’s quite a—we don’t want to be in—that’s not what we are here for. + +BAN: That’s very useful and I want to add beyond just the CLDR data on what currency should be rounded to. Nickel and dime rounding in some locales. Beyond that, CLDR for some currencies specify a different number of digits for cash uses versus financial uses. + +BAN: So by making the data source implementation defined, that has gives implementations the freedom to decide whether to use the cash digits data from CLDR or the typically more digits financial data. + +USA: Small point of order. We are past time. 4 minutes past time. We do have just one other item on the queue. So Daniel Minor, if you would be quick. + +DLM: I will just be just a follow-up on DE and SFC points. If it’s providing a better default, they have options in the applications using different data sources or adding more precision if there’s such—in Indonesia. As a group, we should make things more—more things left to implementation, but to agree on something. + +USA: That was it for the queue. Yeah? + +BAN: Yeah. And a response there is, there is no clear best for all implementations. + +DLM: To quickly reply, Chrome has picked one source and SpiderMonkey and Safari have picked another source and that doesn’t sound like it’s great. We’ve had web compatibility problems. But it’s nice to pick one rather than leaving it to the implementers + +SFC: I have a reply to that. Throughout the spec we refer to CLDR we recommend that limitations use the CLDR, but it’s never a requirement. We could potentially add other boxes. Like no, it’s recommended that implementations use CLDR for this. That’s the best we could do. But that to be an editorial change at that point. You know, the main thing that we are trying to do is watch web reality, and browsers are using CLDR data which is not the ISO 4217 data. This makes the spec reality. We want to tweak it, we can still do that. + +BAN: We are going to be removing several tests. + +USA: Unfortunately, we are well over time. So we—well, okay. The queue—the rest of the queue disappeared. Would you like to quickly ask for consensus? + +BAN: I would love to quickly love to ask for consensus. + +DE: I would like to follow up with that later. It depends on locale data. Anyway, I don’t want to block consensus on this but follow-up work could be good. + +USA: Would anybody like to withhold consensus for something that we discussed today or could that be discussed later? + +USA: I don’t see anything on the queue. Or elsewhere. We have Shane with support. A clarification that TG2 approved this too. So all right. Real quick. Ben, would you like to summarize? + +BAN: So it appears that this is achieved consensus, but we should do follow up work possibly with adding editorial changes and adding recommendations from one data source to another. + +USA: Thank you. Let’s move to the next agenda item. + +BAN: Thank you so much + +### Speaker's Summary of Key Points + +- List +- of +- things + +### Conclusion + +- List +- of +- things + +## Temporal progress update + +Presenter: Philip Chimento (PFC) + +- [proposal](https://github.com/tc39/proposal-temporal) +- [slides](https://ptomato.name/talks/tc39-2024-10/) + +PFC: All right. Good morning, evening, afternoon, everyone. + +PFC: This is a very short update on Temporal, the proposal that is a modern replacement for the Date object in JavaScript. I am Philip Chimento. You’ve probably seen me present on this topic many times before. I am a delegate from Igalia, and I am doing this work in partnership with Bloomberg. + +PFC: All right. Our progress update is that Temporal continues to be close to done. We have finished all the editorial changes that anyone requested. Of course, if implementations still point out typos or minor mistakes, we will fix those. But otherwise, it's ready to go editorially. We will continue to analyze the code coverage and contribute more Test262 tests to facilitate implementations. But please, if you are working on an implementation, go ahead with this. When it’s ready, ship it unflagged. + +PFC: The condition that was previously in place saying we could not ship implementations unflagged, that condition has been lifted. It’s okay to ship your implementation unflagged if it’s ready. If something is preventing you, let us know as soon as possible. If we need to make changes, we want to do that as soon as possible. + +PFC: We have a meeting biweekly at 8 a.m. Pacific time on Thursdays that you are welcome to join—now we are not making changes anymore, that meeting is primarily focussed on answering implementers' questions. And some people join regularly to chat about progress. You’re welcome to do so too. + +PFC: Last week, I did a measurement on all of the in progress implementations measuring what percentage of the Test262 tests for Temporal that pass. And these are the results. Up at the top are SpiderMonkey and GraalJS, with LibJS and V8 somewhat behind. + +PFC: If there’s anything we can do to help these numbers increase, please let us know. I note these are not equivalent to "percent done." The percent of Test262 passing is a very rough measure. So please don’t think that they are this percentage of done. + +PFC: All right. I would like to open the floor to questions. And also, ask a question of my own. If you have existing plans for a timeline for shipping Temporal in your JavaScript implementation, would you like to share that? + +USA: All right. We have Kevin on the queue first. + +KG: Yes. I want to commend all of the editorial work the Temporal people, and especially you, personally, have been doing. I’ve been following the repo and there’s a lot of very good changes. It's a very, very large proposal to do that amount of work on with consistency and clarity. So great job. + +PFC: Thank you very much. + +DLM: First of all, congratulations. And second of all, we are planning to turn Temporal on soon. We were held up a little bit by issues around dates from the far past or future. It looks like there’s been good progress in those issues on the Temporal side, but also some work required as well. When that is cleared up, we should be ready to turn it on in our nightly builds. + +PFC: That’s fantastic news. Great to hear. + +USA: And that was it for the queue. + +### Speaker's Summary of Key Points + +The proposal is as close to frozen as anything can be in Stage 3. Implementations should complete work on the proposal and ship it, and let the champions know ASAP if anything is blocking or complicating that. You are welcome to join the champions meetings. Firefox is close to shipping Temporal in Nightly builds, after some work is done in ICU4X. + +## Make toSorted methods stable + +Presenter: Kevin Gibbons (KG) + +- [proposal](https://github.com/tc39/ecma262/pull/3424) + +KG: So this is a 5 minutes web reality "oops, we made a mistake" normative PR. You may recall that some years back, there was a normative change to make Array.prototype.sort and TypedArray.prototype.sort stable. After the introduction of Array.prototype.toSorted, I assumed and I think everyone else assumed, the stability requirement was intended to apply to those methods as well. However, because of the way that the stability requirement was specified, which is just a sentence in the introductory paragraph of Array.prototype.sort and Typedarray.prototype.sort, we accidentally did not apply that requirement, the requirement for stable sorting to the toSorted methods. In addition, there is a bit of other spec text which says that the sort implementation is implementation defined basically if toString return different values. That again, because of some awkward wording here, was only applied to Array.prototype.sort and not Array.prototype.toSorted. + +KG: It is my belief these are oversites. This PR takes the sentence out of the introduction of Array.prototype.sort and instead, specifies informal language in the abstract operation, all 4 of the sorting methods make use of. So that the stability requirement is spelled out in a commonplace and formally. It takes the requirement about being implementation defined in certain circumstances applied toSorted as well as sort. + +KG: I don’t believe this actually changes for any implementation. As far as I am aware, everyone thought this was what was happening already. So I would like to get consensus for this change. + +KG: Hearing no objections, does anyone want to explicitly support? We started asking for that. + +USA: Sorry. I was muted. On the queue, there was explicit support from DLM as well as MF. Shu, would you like to add some words to your comment? I believe not. So Shu registered support for the record. + +### Conclusion + +Consensus on the PR: toSorted methods will use a stable sort, matching Array.prototype.sort. + +## `Math.sumPrecise` for stage 3 & last chance to suggest other names + +Presenter: Kevin Gibbons (KG) + +- [proposal](https://github.com/tc39/proposal-math-sum) + +KG: Yes. Okay. A substantial item this time. `Math.sumPrecise` is a proposal to add a method for summing a list of numbers, in addition summing that list more precisely than you would do by the sort of naive just "add from left to right" summation. Floating point numbers, you can be more precise without giving performance. Hopefully within an order of magnitude of the performance you get from adding the numbers. Probably a lot—hopefully, not that much slower than naive summation and you get, you know, the fully created sum to within the limits of what floating point numbers are able to represent. The proposal has been at Stage 2.7 for a while. So I have since the last time it was presented in written tests. The tests are relatively trivial except that I have a bunch of test cases that I found by writing a few different implementations and pressing them against each other. These are all things that were actual bugs and actual implementations by me, not like other people. And otherwise, it’s just the sort of standard boiler plate tests. + +KG: The requirement for Stage 3 is only the addition of tests. So I believe it qualifies for Stage 3. I want to call out that the change to the name sumPrecise as opposed to sumExact happened late in the Stage 2.7 advancement process. And no one is in love with this name. But we have said that we don’t want it use the name `Math.sum` because that suggests to readers it’s the naive sum which people might have different expectations about performance characteristics. So you want a name that suggests that something else is going on here. Given that, no one has had a better suggestion than Math.sumPrecise. So I wanted to raise the issue of the name again, in case anyone has in the intervening months decided in fact there’s another name, or if not, this is a "speak now or forever hold your peace" moment. So I guess we should go to the queue first. + +KG: DLM has support for Stage 3 with no opinion on name. Support for stage 3 from WH. Philip supports presumably the current name. + +USA: There’s also support from Ross and YSV for Stage 3. + +KG: Okay. Well, I will take that as consensus for Stage 3 with the name. One note for the implementers. I have done research on various algorithms for full precision summation. Waldemar pointed me to this paper from 1998 or—I forgot when this paper was. But of course you can use whatever algorithm you like, as long as it gives the right answer. And there’s been a number of innovations since then for faster ways of getting the same answer. In particular, I link in this "deterministic sum" thread to an algorithm called xsum that claims to be quite fast. And I believe it’s BSD licensed [actually it's MIT]. So just something to look into, rather than trying to do this yourself. That’s all I got. Thanks for your time. + +### Speaker's Summary of Key Points + +- `Math.sumPrecise` has tests written. [tests](https://github.com/tc39/test262/pull/4049) +- No one has suggested a better name, so this is what we're going with. +- Implementers are invited to look at this issue (https://github.com/tc39/proposal-math-sum/issues/1#issuecomment-2041259013) for an MIT-licensed library which implements a fast full-precision sum. + +### Conclusion + +Proposal has Stage 3. + +## `Map.emplace` stage 2 update + +Presenter: Daniel Minor (DLM) + +- [proposal](https://github.com/tc39/proposal-upsert) +- [slides](https://docs.google.com/presentation/d/1l95mluNq5GqJCj-4o1BqihlufnKuRF2rualJxfOvGu4/) + +DLM: Yes. `Map.Emplace` Stage 2 update. As motivation for this proposal. I think it’s this common case where you have a map and you want to do something, but you are not sure if it’s already present so you can skip doing the dance of checking has. If it’s there, pushing—in this example, it’s an array. Whatever. Using get or set. This is something we can do better on. + +DLM: History of proposal. So the current design is from 4 years ago. Currently, Stage 2. The champion was interested in passing in two callback functions. One in the case where an insert is necessary. So this is the missing value. And a second that is used if the value is present. This allows us to recalculate that value. + +DLM: In the design, these are optional. The method would throw if it should have been there and was not. So if you did need to insert a missing value and there’s no insert method, it throws in that case. History of the proposal. So this predates my involvement committee. So I did a bit of digging into the notes. But from my understanding, this was last presented in July 2020. There seems to have been consistent feedback from the committee to split the proposal into two methods. From what I can tell, the champion had a strong preference for retaining the one method. And it seems like not much has happened since then. + +DLM: I became involved in this last plenary meeting. We did a scrub and this came up as something was in need of a champion. And at the same time, we had a group of students at the university of bergen looking for a proposal to work with as part of an implementation class. I’ve been working with them in terms of looking at an implementation inside SpiderMonkey and V8 and trying to develop the proposal a little bit. + +DLM: I have only recently tried to get in touch with the previous champion. So I would like to give them more time to get back in touch with me. That was an oversight on my part and I should have done it earlier in the process. + +DLM: Moving onto more widely talked about the design space. Although it seems like a simple problem, it raised a lot of options here, which became evident to me as I discussed this internally with each people on the SpiderMonkey team. I like to look at some precedent from other languages. Particularly, Python and Java. + +DLM: So in Python, we have setdefault. This returns the value if it’s present in the dictionary. Otherwise it inserts a default value. Which argument functions and then returns that. So this is kind of useful in the case where the default value is an insertion time. + +DLM: There’s also defaultdict which subclass built in dictionary type. In this case, a constructor which takes a callable, used to create a default value when the value is not present in the dictionary. This is more useful if the installment value is known at construction time. + +DLM: Based on comments in the issue, this proposal as well as conversations internally, it seems that people for the most part use one or the other, but not both. In my case, I use setdefault. But other people in SpiderMonkey only use defaultdict. So in that case, even though they are complementary, most people choose one or the other. + +DLM: I was interested in looking at some precedent from Java. We have computeIfAbsent. Use a callback to compute the default value. We have computeIf[?]. This is a precedent that the previous champion was working from. + +DLM: So new hope? This is more my opinion, but I think the Pythonesque is useful. It simplifies a common pattern. It helps improve readability and other options that are optimizable down to a single map look up inside engines. We see a performance win. And point out that we don’t have to call it emplace. If you look at it the right way, it is like C++. But that’s not obvious. + +DLM: I am not convinced this is the best name either, I forget it having done this presentation. Up until now, I opt to Google for that name as well + +DLM: What about some of the other options? DefaultMap? This one can turn into a rabbit hole. Probably want a DefaultWeakMap. And other types of maps to consider adding. And I think it opens the door to a Java-like explosion of subtypes of containers. There is some risk of reuse of the default value. Like in Python, I think that’s mostly avoidable. If we are careful with callback functions. But still a concern. + +DLM: Some other things that came up. So what happens when you call has and a key is not present. When you call get, a default is inserted. And then also the fact with a default map, the construction site defining default far away code, and map is a source of potential confusion. + +DLM: What about changing Map.get? Another suggestion is coming up. So changing the get call to take a second optional value, default value for missing key. I think that’s a big change semantically. It could reduce readability of code. SpiderMonkey adds a new method rather than using the existing one. In the case where we don’t want to insert a default value, we can handle missing values using nullish coalescing. + +DLM: And that leaves the update use case. I think it’s interesting, but I don’t think it’s as interesting as this set default use case. I think it’s something that we postpone to a proposal with sufficient interest. + +DLM: So although this was presented as an update, I would like to get feedback from the committee. So sort of a three-part question. One is if there’s any objections to me taking this over as champion. I reached outto the previous champion, so I would like to give them more time to respond, but I would like to hear if there’s any concerns from the committee. I am curious if we think that the set default use case is sufficiently motivated at Stage 2. Or if we should consider moving this back to Stage 1 or if I should bring a Stage 0 proposal in the future about plenary. + +DLM: Okay. I guess with that, we can go to the queue. + +USA: On the queue, we first have a clarifying question by Waldemar. + +WH: Can you define what *setdefault* actually does? + +DLM: Sure. Let me—in my mind, what I have is like the example from Python. So it’s going to check the map. If the value corresponding to the key is present, it’s going to return that. Otherwise, the second argument, which is default value, would be inserted into the map. And then that value will be returned. + +WH: Okay, so *setdefault* takes two arguments: a key and a default value? It does not take any callbacks? + +DLM: Yes. And that’s actually—you have feedback on that point, I would be quite happy to hear it. + +WH: I just wanted to understand what you were proposing. + +DLM: Sure. Okay. Thank you. + +USA: Next on the queue, we have Ashley. + +ACE: Sorry. Just repeat the feedback I gave in person just on the record. We discussed this in Bloomberg. And are very happy with this change for the API and happy to see this proposal keep progressing. We see this pattern appear just all over, lots of projects at Bloomberg and everybody is repeating the same helper and repeating it in another language is fantastic. Very supported. + +DLM: Thank you. + +USA: next on the queue is Justin + +JRL: For update, I don’t care about the update use case and the original proposal. I do like we are looking at the default use case just so we can have a default value in the map. That aligns with almost all of my usage of map and WeakMap. But the inability to to pass a callback to set default means if you have any map that has an expensive computation, you can't use a setdefault value, which is every template function that I write. You can no longer use this emplace. You have to go back to the normal use case. Check has. Return if it exists or put it in there and return the default value. With a callback, you can perform both inexpensive computation, or do expensive computation (all of my template tags). Thank you. + +DLM: Okay. Thank you. Thank you for your feedback. + +USA: Next on the queue we have JHD. + +JHD: Yeah. I like the new direction, but I also think the default should be a thunk/callback in case it’s expensive to compute. Other than that, I am happy about that. + +DLM: Okay. Thank you. + +USA: next on the queue there is Oliver. + +OMT: Yeah. I want to say, I agree, this is nice to have something in the language. + +DLM: Okay. Thank you. + +USA: Next, we’ve got Ross. + +RKG: Yeah. Just supporting the new direction. If you like, we can surely come up with a better name than emplace, but I am not too worried about that. I don’t need to spend time doing that. + +KG: Yeah. First, thanks for bringing this back. Second, I want to toss out the name getOrInit, or maybe getOrInsert. Set default is a terrible name. Second I want to disagree with all of the people wanting a callback here. I think that having a function that takes a callback - Java calls this computeIfAbsent - is potentially quite useful. However, I think that the case that I most commonly run into is when the default value is the number 0 or empty array. And in common usage, for those cases, people are going to write the callback in line. And paying the cost of creating that callback instead of just passing 0 is wildly inefficient. I don’t think the efficiency argument takes the case for one or the other. There’s room for having both under different names. But if we are to only have one, I would prefer to not have the callback taking one. + +DLM: Yeah. That’s a good point. Thank you. Yeah. There is room for two. + +SYG: Quick feedback on the name. I think emplace is particularly bad. I think two people understand emplace. I don’t think it means C + + sense here either. Nothing else other than don’t call it emplace. I strongly agree with Kevin. If I choose a single method, I prefer not to take a callback, but I believe there is room for both. + +DLM: Okay. Thank you. + +USA: Next we have Ron Buckton. + +RBG: Spec on my side, I also support this. The name by shed, I would not go into all the different options for names, but if we have issues on the issue tracker, we can discuss that. That is acceptable. I want to say that I also believe that having a callback is important. The majority of use cases I use this, there is an expensive computation. And the cheap computation, maybe there’s some cost associated with setting up a callback to return something that has a predefined value. Those with—usually can be off-line to a function. You can reuse the function for the call bag. There are ways of doing—doing the cheap thing with a callback. I am not—I am more interested in if we only can have a single version of this—a single function for this or single method for this, it would be the callback method mechanism. + +DLM: Great. Thank you. + +USA: Then next we have Ashley. No. First, Devon, + 1. Excited for this. There’s a better name. For example, ensure. Then we have Ashley on the queue. + +ACE: Just another + 1 to Kevin when looking at this at Bloomberg, the default the most common thing is the value is the cheap thing and having to wrap that in a callback would not be so good. Consider this use case. But if we have the error, I prefer that .… a second method. And there is just the simple one where you don’t have to pass the callback. + +DLM: Okay. Thank you. + +USA: That was the entire queue. + +DLM: Thank you, everyone. I guess I can dictate a quick summary. There’s good support for two varieties of “setdefault”, one using a value and the other using a callback. And I will definitely plan to open an issue about finding a better name for this. + +DLM: Given the support here, I plan to take over as champion, but I will give the previous champion a week or so to get back, in case they have any concern about this. And that’s it for me. Thank you, everyone for your time. + +### Summary/Conclusion + +- A new design was presented, focusing on the use case of inserting a default value into a Map during a get. +- Based upon feedback, two methods will be added, one using a default value directly, and the other taking a callback that can be used to determine the default value. +- DLM has taken over as champion of the proposal. +- An issue will be opened to bikeshed better names for the new methods. + +## Extractors for Stage 2 + +Presenter: Ron Buckton (RBN) + +- [proposal](https://github.com/tc39/proposal-extractors) +- [slides](https://1drv.ms/p/s!AjgWTO11Fk-Tkr0Z2sm9bVmO_agqpA?e=7TdVup) + +RBN: So this is for extractors, the hope is to see if we can get to stage 2 today. So the motivations for extractors are that there’s currently no mechanism for evaluating user defined logic during the structuring. The pattern matching proposal produces did ability to do this with custom matchers as part of nested pattern matching. And as part of that proposal or in tandem with that proposal, the extractors proposal provides a convenient syntax that allows us to use the same mechanism in pattern matching and destructuring to maintain the parallel syntax. + +RBN: So this proposal’s goals were to investigate a novel syntax, executor user-defined code, which allows a number of things, including in line data validation, transformation and normalization during destructuring. Primarily this leverages Scala’s extractor objects and prior art, in prior presentations I’ve shown there are a number of different languages that also have similar capabilities. And, again, this is very much based on the custom matchers anymore of the pattern matching proposal and has parody in the structuring. It also provide a basis for potential future proposes around enough and ADTs which I hope to brick an updated presentation for enums in the future. Some example of what extractors look like, so if first with example is an extract that takes an input and parses it into its outputs. The idea is here you might have constructed something from these inputs from A, B and C that originally that allows you to pull this back out. Another example is option types where you can have a value that has a some and a none type. In the case, it would verify that input fine is a sum type and would then extract its value into the variable. And it could also be used for doing nested destructuring. The thing that gets pulled into or extracted into an ISO date can be further destructed to pull out the year, month and day, in this case, this is regular expression. I have another slide on this. And again, nested within destructuring, so you request have an object that has a location property and pulls the point out. And not only is this useful within variables, but within parameter declarations. + +RBN: So a brief history on the proposal. In September 2022, this was proposed for adoption at Stage 1 that included array extractors, which are essentially what you see here. Where this looks like a call, but this is actually more of array destruct here. As well as object extractors, which used a curly bracket syntax. And at the time, we were still only seeking advancement for extractors in binding and assignment patterns. The version of this that comes with pattern matching is of course tied into the pattern matching proposal. At that time, the to proposal was adopted at Stage 1. Throughout 2023, we had a number of discussions with the pattern matching champions from various discussions here, and the issue tracker. We agreed to incorporate array extractors into the design and drop object extractors that had some complexities with the syntax space that we didn’t want to potentially eat up too much syntax with identifier curly brace within something like a destructuring assignment pattern that could have otherwise been interpreted as some type of new syntax that required identifier curly. + +RBN: So in February 2024, we provide an update on the proposal. And now it’s the official drop for extractors. I mentioned some additional consideration needed regarding array destructuring and due to the iterator pattern performance. And in April 2024, we proposed this for Stage 2, but this was blocked on primarily on an ASI concern. So briefly I’ll discuss some of the semantics here and some of the changes we—a few changes we’ve made to address the last concern that was brought up. + +RBN: So extractors in the case of binding patterns and assignment patterns consist of two parts an extractor member expression on the left side, which consists of an identifier or a set of dotted names, these are the same types of types—and also very similar to what we used for decorator expression. To the right of extractor member expression is what is a entrepreneur this sized list of variable patterns or variable—further binding patterns that are pulled from each of the elements of the resulting iterable or array that is returned from the match result. When these are used in binding assignment patterns, the extractor purports array destructuring on a match result. And the extractor here is denoted by parentheses rather than square brackets despite using the array destructuring mechanisms, primary reasons for to avoid confusion with element in the case of the pattern and you could look at an assignment expression, which would be inconsistent and conflict with existing syntax. Also avoids confusion with computed property names. It has some similarity to call expression, which has useful visual similar to when it comes to the construction side, which you can see in the example in the bottom of the slide. And it allows us to mirror kind of construction application. In the case of a failed match, when doing a failed extract, this will throw a spec defined error when a match fails. In the pattern matching proposal, a failed match results in attempting to choose a different option if you have multiple patterns to match against. + +RBN: How matching works with extractors is defined by the pattern matching proposals custom matcher capability. Basically, that has a simple custom matcher method is—so an extractor itself is the object, and if it has a `Symbol.customMatcher` method, it will be invoked with three arguments. The first argument is the subject that is being matched against, so in the example below, the subject would be the—whatever the crust—the subject would be the string that is shown on the right. It will also include a hint. This hint is primarily to distinguish what the expected return type is for the—for assignment and binding patterns. The only value that will ever be passed in is list, which means that the expected return type is a list or iterator to be further destructured. In the pattern matching case, it can be a boolean, which is used for very simple tests. In which case the result is passed via a boolean and to help you know you can avoid any type of expensive construction that needs to be further destructured. Again, for the extractor case in binding assignment patterns this will always be list. And the last argument is a receiver. The receiver is a essentially the object that the pattern matching was called on. It’s—or the extractor expression was called on, so if you had `foo.bar` open receiver, it would be the food receiver since that would get lost due to the invoke indication of symbol.customMatcher, this will be the result of the expression rather than this—or the left-hand side that it might have looked like. + +RBN: So some updates occurred the last time this was presented for advancement. We have a PR that addresses a glam at the Cal issue in the cover grammar for cover can call extractor arguments. The extractors reuse part of the binding and assignment pattern case. Eyes the binding pattern syntax which allows for implication. However, the arguments cover—using arguments as a cover grammar, arguments does not allow illusion for invocation, so we addressed this, which says basically that illusion will be allowed extractors and not allowed for extractors in normal invocations. The other change we made, this was the original concern from Waldemar from the previous meeting, there was an ASI concern. I’ll kind of get into that here. So the original concern was that in the case of a binding pattern, extractor in the binding pattern, your expectation would be let X open receiver A equals B would perform the extractor. If there’s a new line after the X, then this is a valid simple assignment even though there’s a entrepreneur that sized expression on the left. And this is a conflict. So the—so no address this and headache sure that we don’t conflict the with existing valid JavaScript, we need to introduce a no line—new line terminator assertion to the grammar. But this results in an inconsistency to make sure we have discussed in plenary. So that inconsistency is that in the case of a call expression, which is the exam on the far right, a call is always a call whether there’s a new line or not. In the case of extractors, for consistency with both binding and assignment patterns, it is not an extractor if there’s a new line many between. In general, since there’s a major semantic difference between call and extract, we—both the champions and for the champion for this proposal and for pattern matching generally concur that the difference in semantics is enough that the minor difference between what is considered a call and what is considered extractor is considered acceptable. I would like to get some feedback from the committee on this. Res with looked at an open receiver of a new call is rare and generally forbidden by linters as the default behavior for various rules. There’s also a discrepancy in—just in arguments in general in that call a new expression are similar, but they—the new allows augmenting to be commutely alighted, so we are not completely consistent within languages or expectations around what an argument list looks likes. Again, I think that a reason why this discrepancy should be viable. + +RBN: And then the last—this is also—we presented on this at the last meeting, which was that the current proposal uses array destructuring, except array destructuring is actually iterator protocol which is not very efficient. As no implementation currently optimizes array destructuring, and we’re still hoping to address this post-Stage 2 or during Stage 2 as we are looking into currently putting together some benchmarks to do some analysis that we can try to determine if there is an option for implementations to optimize cases like very simple array destructuring, which is commonly used in many places like in use states, so, fen, this requires bench marks, further analysis. We are in the process of putting that together now. And, again, I’m hoping this is Stage 2 concern, I’m hoping to continue with iterator as the protocol now with the goal of finding ways forward to—seeing if it’s possible for implementations to improve that support. If we determine that it’s not a available option moving forward, then we are likely to go with using an array as object to structure mechanism, which just requires the result be an array and the destructuring actually occurs via indexed elements, so actually look at length and zero, 1, 2, 3 within the array rather than performing iterations since that would be more efficient. + +RBN: Finally, I have the slides for some examples. This is example of something like a point class where you might want to have the X and Y value be read only. And you can extract those values. Also do system brand matching within the custom matcher. So this does brand testing, extracts the values in a way you can actually receive them and could be used both in the destructuring and pattern matching case. In this case, this is something akin to an algebraic data type where you have a Rust style enum where you have a number of options you might want to consider for what you’re actually construct examining then when performing a destructuring you can verify if I’m taking into something that has a line which is two arguments that has the same properties, so align has p1 and p2, when you try to extract that into a rectangle which is expecting a different thing, that you get a different result or that you don’t blindly succeed. Instead this would throw because this is the wrong type. And examples here with pattern matching showing branches on each of these individual cases using custom matching. And, again, some examples through—of qualified names. So message.write, option.sum, just a momentary.point and nested patterns. And this is kind of briefly—this example is kind of briefly shown on the beginning slides without the context, but the pattern matching proposal is intending to add a custom matcher to RegExp instances. Which would allow you to use them in patterns, so you could have, ISO date parsing via regular expressions, and do building extraction of these into the component parts, and potentially even nest them to break out these things within a normal destructuring—nested destructuring or use pattern matching to break these down. + +RBN: So with that, I will go to anything that might be on the queue. + +WH: Going back to the slide about the ASI issue, can you say what happens in the not-an-extractor cases? + +RBN: So in the let declaration or in the first example where it says not an extractor, this would be parsed as a let X variable declaration and a simple assignment. In the second example, this would be considered not an extractor, would would fall back to the default of behave of X entrepreneur is a closed expression which would throw at run time because you cannot call at a run expression and the the right example would be a call expression, because that is how it’s parsed today. + +WH: Okay. + +DE: This is a great proposal. It’s a very clear design. There are still some things left for Stage 2 discussion, but I think those are the normal kind of details to discuss during Stage 2. This proposal fixes the biggest concern that I had for pattern matching besides the too much complexity, which was being able to use these constructs in different kind of left-hand side situations. Custom matchers been found to be important in lots of different programming languages and it would be great to have them in JavaScript. + +YSV: I’ll reiterate a concern that was brought up in the last meeting when this was presented, which is we are unsure what the performance impacts of this will be. Since then, we’ve also discussed more specially, and what we found is we fall in line of being weakly negative towards this proposal. In terms of whether or not the power that it brings to the language is maybe not too much. That’s a fear that we have. That this will bring too much power to what you can do on the left-hand side of an assignment. We are not currently blocking Stage 2, but we wanted to register our concern. + +RBN: I appreciate that, thank you. + +MM: Yeah, I am strongly supportive of extractors. I think you’ve done a great job here. I appreciate that you made it work with the pattern matching proposal. I think we need something like the pattern matching proposal, but on—but way simpler than the existing one, so I’m not supporting the current matching proposal, but the things that you have done to live with the pattern matching proposal, I believe, are the right things to do to live with an acceptably simpler pattern, so I’m supportive. + +RBN: I would like to say we have been working on some updates to the pattern matching proposal that I’m not sure if that’s on the agenda for this meeting, but I know the plan is to at least discuss it again for potentially Stage 2 at the next meeting. We have simplified a number of things. Hopefully we can come to some agreements on what that needs to be at an upcoming meeting. + +MM: Okay, thanks for the heads-up on that. I am interested. So in particular, I think the existence of extractors means that much other complexity can be removed from pattern matching because they can be subsumed by using patterns and extractors. + +RBN: Thank you. + +SYG: I think this is a prime candidate for a JS sugar feature. This—I won’t block Stage 2 here because the presentation yesterday, you know, I explicitly—I have not proposed any of the actual detailed workings on the JSSugar proposal, but I think—as the champion, you should—I should be explicit that I currently don’t think this meets the bar to be implemented natively in engines. That said, I definitely see the utility and the demand on the developer side for folks who want to do this kind of thing. And you have a run time component that may make sense like a new well-known symbol. Part of the, like coordination point. But this—and pattern matching certainly, but both strike they as prime candidates for a desugared feature. + +RBN: So I’ll mildly disagree on this. Mostly because the destructuring—sorry, the JS Sugar for this inside of destructuring requires reassuring all of destructuring. Everything at every level, because you cannot run—you can’t really run user code inside of destructuring. Its very complex to do so. It’s for the assignment expression case, you can kind of work around it by wrapping something with a thing that has a get set value in doing things inside of that, but that introduces weird scoping semantics, and what TypeScript would have to do for this is, again, fully desugar all of destructuring at all levels to be able to inject this anywhere within destructuring, so I’m not—even though we are going to have to do this, it’s not something that I would—not something that I would expect would be very efficient within something like JS sugar. I know that performance is likely to be negligible and introduce a bunch of temporary locals that could leak. So I have some concerns about something like this being a purely desugared feature. Also, there is discussion about what the runtime behavior would be around things like the array destructuring performance, and if we were able to get, say, engines to investigate iterator—array iterator destructuring performance improvements, we might not be able to actually benefit from that if this is fully desugared, especially if we have to do the nesting to make he’s things work. I have some concerns because I think it’s much more efficient to do these things within the runtime. + +JWK: I strongly support this. And if JS sugar is a thing in the future, we still need to discuss all these semantics, otherwise tools cannot implement them. + +SYG: Right, I’m not saying we shouldn’t discuss this. I’m saying to Ron that I don’t think this meets the bar for native implementation. That’s, you know, not to kill discuss and not kill the feature. We ought to still discuss this. + +DE: I’m grateful that Mozilla and V8 are not blocking Stage 2. In particular, I think because we just now started the discussion about native implementation and that the issues with that in general, it seems reasonable to not block Stage 2 for proposals. But we definitely need to figure out the relationship to JSSugar by Stage 2.7, because this affects a lot of these semantic points as Ron explained. + +RBN: And as I said, as far as the desugaring for this, even if this were in JSSugar, even if this is, I am part of the TypeScript team, I’ve been thinking long and hard for this and the pattern matching proposal, we’ll have to do these types of implementations and it’s a matter of what level and to what extent could we in the desugar emulate the semantics exactly that we would propose, and it’s very difficult to emulate those semantics exactly. Not impossible, but TypeScript tends to take shortcuts to avoid really messy outputs because it can affect the ability to debug at times. So, yeah, we are considering this. + +KM: I guess this is kind of carrying off the same point that Shu and Yulia were raising. I guess similar kinds of concerns about the implementation side of this. I definitely see if use case here obviously if languages that have, you know, pattern matches, like, in camel and has kel, I definitely appreciated it, I just worry that the complexity is very difficult. And, yeah, agree—I could imagine some idiom that works with this. I mean, I think the nested one would be difficult. I think I—yeah, I just want to reiterate the shared concern about implementation, like, the value add to implementation complexity on—within engines themselves. + +RBN: Yeah, so the—again, the reason why this is proposed is because let xy equals point.extract is not sufficient for the cases that we are trying to cover. One of the main motivating use case is to allow the ability to perform some validation and extracting in context in the middle of destructuring. In places where you can’t do let xy, where you might want to have a more efficient mechanism of extracting, so the example here with option some of message move pulling at the X and Y properties, this would be, like four or five statements, and very complicated. But you could also then do this in the parameter of a function to perform, again, data validation, data transformation, custom extraction, things that you can’t do with destructuring today, and the makes things somewhat more complicated, and again it was one of the main motivating cases for this. + +KM: You could do something like option.sum.extract and then as a parameter, pass message.move. And then that means that inside of there, I want to extract the message move as like a subextraction point. + +RBN: Which introduces errors—zero functions and doesn’t introduce the values at the local scope, so you have to declare the variables outside of it and extract them. It’s not a very—it’s not a very efficient Mechanism. + +KM: Yeah, yeah, I mean, I agree there’s definitely down sides. I think the—if this proposal still—it feels like at first glance it’s trying to hide the same downsides inside of the engine. And that does not mean those downsides—those difficulties go away. They just have to be handled by somebody else, which, you know, is, I guess, the point of the JS0 discussion. + +JHD: Yeah, so I just wanted to add, like, I’ve—I have in the past had similar thoughts about extractors as in, like, I don’t really know if, like, I didn’t know if I valued the use cases enough to, you know, put in the language. The thing that convinced me personally, which is not to be clear, is not the motivation for Ron or the proposal, is that as side effect of this proposal, I would be able to just this in the function parameter, so all of the runtime type checks I’m always doing for the top ten lines of my functions, I would be able to very nicely cleanly put in evaluation extractor functions and reference directly inside the function signature and that will be hugely useful for me and allow me to very cleanly and in a reusable way share my validations in a way that will also be able to have probably a more helpful stack trace, although, that’s not a guarantee. You know, again, that’s not why Ron wants this proposal and, you know, what the proposal is aiming to solve, but that’s a side benefit I would see if this proposal advanced as is. + +USA: Thank you. Next we have—okay, next on the queue we have Linus. + +LGH: So that seems fine to me for solving the issue with the let declaration, but one of the other examples you had was labeled assignment extractor, and I believe that is currently already valid syntax. It’s one of those weird edge cases where it throws a ReferenceError at the point of assignment. And if I recall correctly, engines tried to change that to a syntax error but came to the conclusion that that is not web compatible. So I wonder what you think about that. + +RBN: I’m not familiar with the concern about web compatibility. But this was discussed when this went up originally for Stage 1, is that the reason that a sign-up pattern extractors are part of this proposal that there is no case where an assignment to call actually works today. It will always throw. It does throw at runtime. You could theoretically put a try catch around it as the way to test whether extractors are a thing. But beyond, that you can’t actually leverage that behavior in a way that doesn’t result in an exception. + +LGH: Okay, think it would be a breaking change, but it turns out to be web compatible, that seems fine. + +RBN?: Yes. + +KG: Generally speaking, taking something that always throws an error and making it not throw an error, while that is technically a breaking change, it almost always works out in practice. Because very few people are intentionally throwing an error and then, like, depending on that error continues to happen. And in fact, in existing code, it almost certainly would, because, like, whatever it was that they would have written would turn out not to have been an extractor because extractors didn’t exist when they wrote that code, so it should be fine. + +YSZ: Yes, I have some, reassuring thought with Keith and Shu, so basically, like, this feature implemented in the run digit, in the other run digit, it’s pretty much a runtime bar edition like on Similar Ra, and we can implement this kind of feature if it has a type processing, like a TypeScript, and if we implemented this in engines and we needed to do the runtime call in other things, and I feel like it is kind of like the same sentiment to TDZ, and I’m in the saying that I drag the opposite, but I’d like to see that this thing is, like, evaluated with the soul after JSSsugar and we did the discussions. + +RBN: So I’ll say a couple things. One, there is some benefit to TypeScript in that TypeScript could leverage the syntax to give you early compiler errors if you tried to do the wrong thing. We do that today for many things. But if your suggestion is that TypeScript would use this type information to rewrite the destructuring, we would not do that. TypeScript has for many years had a policy of no typed emit. We have a very few small cases we do early on in our initial design, but we do not do type driven emit and we refuse to add new feature that are purely type driven. If TypeScript could do there because it could use the type information and remove all this, that’s not something we would do. + +ACE: Just echoing what Ron said. At Bloomberg, we’re also a bit nervous. We’re very happy that TypeScript’s current line is ignoring the legacy parts of type driven emit, which are really, really old versions of TypeScript. Very happy with that, because we freely update TypeScript quite regularly, and we can do that because it has no runtime semantics. So we can update, you know, a million lines of TypeScript to a new version of TypeScript and we know that’s not going to impact any financial calculations. If someone adding a different return type to something could now change the runtime behavior, that means we’re going to have to be much more cautious with how we upgrade TypeScript, which may end up in being on older versions of TypeScript. It’s unclear how that would work in our world. And I can’t speak on behalf of Google, but I know they have a similar thing where they have a large monorepo and try to have the same thing across. I can see any kind of large code base having that. Plus one to line of reasoning. + +SYG: I’m not sure I want to—I want to better understand the type driven emit thing. What I heard is one, it’s a policy and the actual concrete reason I hear from ACE is that you don’t want—it sounded like the failure case is that someone added a type, and it still compiled to something. Like, it didn’t error out during the translation from TypeScript to JavaScript except now the output JavaScript has different behaviors. Is that the failure case do, I understand correctly? + +ACE: Could you say it again. + +SYG: Is the failure case that someone add—your example is someone added a new return type and now there’s different runtime behavior. My question is why didn’t adding the new return type cause the TypeScript entire project to fail to compile because you changed the type? + +ACE: So you can change the return type and the previous return type was equally correct. You’ve now moved to a different type but it’s now correct. You could have something that was very specific, like this returned 1 or 2, now it returns number, so it compiled before, and it compiles again. You can also not change anything, like, updating TypeScript, they change their inference, and it now just infers a slightly different type. And both types are correct, neither produce an error. But it’s a different type now. You’re right, that sometimes you change a type, and it causes an error. And that person now has to—so in our world, if we did if we update TypeScript and they now infer a different type, that may now produce lots of errors. We then go through and try and fix those errors, and we’re just trying to—these errors are purely static things that we can detect statically and fix them. + +SYG: Right. Can someone help me understand, then, how would—if we are to—in the hypothetical, if we were to use type-driven emit for extractors here, what is a concrete example where it would break in the way that you are afraid of? + +DRR: There’s a couple things to note. One is that I mean, I think DE put in on the queue already, that this is—we don’t really have a sound type system, and so, you know, your mileage may vary in terms of, like what actually happens at runtime versus not. Between what the—between what the deficiencies and runtime deficiencies actually does. But on of the, that you have to consider the fact that there are other compilers that are independent of TypeScript’s type checking, right? So the type system that—the principle of type eraser is partially to open the door up to other implementations of the compiler, right, and those compilers all do purely syntactically driven operations maybe with a little little semantics around simple scopes, and binding information and simple information, things like that. So for the last few years, it’s not—it’s actually been something stronger than even note-type driven emit. It’s no cross file emit. Basically if you have to resolve across the file to make a determination of something, that’s no go, because many other compilers have limitations around that, too, for example, Babel basically parses every JS file its own compilation unit and doesn’t really resolve across to make any determination about what to do. So it goes a little bit stronger. It’s not just about—I mean, part of it that we have an unsound type system. Part of it is you can also get your JavaScript emit even if type checking fails. But it’s also the fact that other tools can process TypeScript without having any type information. And so basically, it’s not a block for there’s a type error. Your JS compiler, bundler, whatever, can still work on it. + +DE: KM was asking, doesn’t it change the semantics of your program if the type changes? And I think DRR answered clearly no, it doesn’t change the semantics. + +DRR?: Yeah, that’s correct. + +DE: You know, TypeScript has the thing called overloads, but those are really more like union types of functions. So there isn’t anything that happens at runtime based on the types. And it will be a huge change to design to do that. + +RBN: And, again, we have a few cases that are type direct emit that we don’t want to continue to repeat. Some continue enums are very valuable, but those change the runtime semantics when you change them, and we have a couple small other cases that we don’t want to repeat those because it introduces a complexity that is a problem downstream. So it’s, again, not something we want to support in the future. + +USA: Next on the queue we have Shu. + +SYG: So use case point, as I understood, was that—so I understand that TypeScript’s type system is not sound. But I thought what use case was saying if you could prove it, then you can do it efficiently, otherwise it’s a fallback, like VM is working comment automatically. You are saying that is also not possible. It’s also just as a policy choice or whatever that there are no, like, sound subsets of the program that you currently analyze and can do a different thing with, like, a different observably equivalent output? + +RBN: So we have discussed doing this, and we’ve considered this in the past. We’ve had discussions with outside teams about potential for compiler plugs in that could do, like, constant folding and other optimizations, but we don’t do any of those currently. That’s not a feature of the compiler. We generally just to not do any kind of runtime optimization. We do our best to emulate spec semantics. There are places where one of our—I wouldn’t say it’s a policy, but more of a rule of thumb is we try to generate, where possible, output that is easy to read so if you are debugging these things without source maps, if you are stepping through code, that you can more easily read that code. We have a few cases that don’t, namely generator down levels. That’s just impossible to do cleanly. Destructuring down levelling is a mess because it generates significant number of temporary variables. But we generally fry do this. And in some case, we leverage the fact that we can give you type checking and say, hey, this is going to be a compiler error, so we don’t—so we won’t necessarily do some of the runtime checks in our down levelling because we hopefully warned you ahead of time that this is a problem. But for the most part, we don’t really do other types of optimizations, and we often can’t even get spec semion the ticks because of things like down levelling async await, we’ll end up with more turns that what you can actually get in the smoke Semin antics because we are outside of the engine. So there are limitation what we can actually do with our down leveling. + +SYG: So I would encourage this as an avenue of exploration, given the change in risk appetite from the engines that we talked about yesterday. + +DRR: I think, you know, just to get to—I don’t know, it’s totally possible for us to speculatively to fast pass, slow pass, like what you’re describing. The thing is that it typically sits more towards the end of the build pipeline than us. It’s typically bundler or closure compiler. The closest example I can think of that does anything like this is a tool like esbuild that says I have the entire program, like, the actual source code available to me, all at once, and I am going to choose to do inlining and things like that. And, you know, it’s something that we’ve pushed back on, we have avoided doing ourselves. We don’t even bundle ours at all. Like, the TypeScript compiler I mean. And so it’s something that we’ve avoid for a very long time. And it’s kind of—I think other replies are going to get to this, but it’s generally not been the thing that we’ve harkened back to wanting to do more of, like, type directed emit, cross file emit, things like that. It’s a much simpler compilation model or checking mod it at least. + +RBN: One other thing I’d like to around and I put the slide up to show, if ex-for example, we wanted to just use, let’s say, the interest was in just using what we could statically know without type information to potentially generate something or to potentially translate the output, the problem here is that the purpose of custom matching is to have user defined code, that user defined code could do anything in that function body. If maybe the function that you are—the object that you’re referencing has a custom matcher, then it’s defined in that body—in the same file and we could statically analyze the contents of that method to determine what it does and somehow inline that into output, assuming that we can verify that that static custom—`Symbol.customMatcher` hasn’t changed at runtime because someone else has patched that object, you’re talking about extremely complicated down level emit that is going to ship with your code. It’s going to be part of your bundle. You’re going to have to pay for that on every single case, versus a maybe less performance emit and it increase code size and you get faster downloads and—you’re starting faster, your application starts faster because you’re maybe aware that I’m going to pay the cost for running the custom matchers if I use them in my code, but if I am code pass that don’t use them and I have all of this complexity that has to be added to the JavaScript to prove the customization out, that’s something you have to pay for every single download and every single user even if it’s not touched. And that’s not very efficient. I’m more likely to say the slower code path is best for sugaring because we’re not going to introduce this without a significant amount of Overhead. + +DE: Yeah, I totally agree with RBN’s point, and I think it’s great that the engines have brought this JSSugar topic to the agenda so now we can have more discussions about what is possible. It seems like both on what is practical to do in transpilers and the nature of type checking where different people on the committee had different ideas. And I think that proposal as well as pattern matching will be a good place to work that out. On the question of sound type systems where SYG was encouraging the thing that RBN was talking about with more optimized emits with fast and slow paths as well as type systems actually proving things, I think we’ve seen in the market where there have been lots of attempts at type systems and optimized emits for JavaScript that the unsound approaches, the approaches that, you know, take certain assumptions, that aren’t always true when taking types, those have won in the marketplace. Not just for TypeScript, but for other type checking systems like flow has a sound mode, I believe, but that’s not always used. Someone can correct me if I’m wrong. And with other transpilers like closure compiler for optimization, it has this optimized mode, but it requires a lot of attention and care by the people using it, and that’s kind of a barrier to entry. So I think the separation of not doing type directed emits makes sense given the all this context of it’s hard to retrofit a type system to an existing programming language, and over and over again with TypeScript, we found that you can get better types—you can get more useful help by doing some of these more advanced tricks that are just exploiting the logic of the type system and don’t necessarily have runtime representations. + +YSV: I just want to reply to the comment that was just immediately before, that when—by shipping this in implementations you don’t have to pay for download of the code. So this also depends on how you’re using extractors. And one of the to herbal the use case for extractors is making sure that what’s being passed to a variable or a function is the thing that you expect. So effectively, kind of like runtime type checks. And if it’s being used in that way, in is where the performance concern comes in, the cost of doing this ahead of time is cheaper than doing it during runtime. Because even if the download, like, I imagine if you do this statically, then the download will be about the same size as before. You’ll just get compile time errors. Even if there is something additional in the download in terms of checks, it will still be cached and reloading your application won’t cost more. Basically, it’s cached, so the cost of download is less than landing this directly in engines depending on what you’re doing with it. So I think it depends, and might be a balance—there might not be a performance improvement, because the benefit of a smaller bundle is offset by the type of code people start writing. + +RBN: So I’ll say a couple things. One is that if you’re looking at something like TypeScript, this doesn’t really—this wouldn’t be the best way to do, like, static type analysis, because we can do static type analysis without runtime, any runtime code. So if you wanted to say that the argument is string, then—and had a matcher that tested for that, in TypeScript it’s cheaper just to declare the argument as a string. At JavaScript at runtime, if you want to actually do some type of runtime validation of that nature or checking the contents of the stripping as part of validation, that’s something that if you didn’t want to use TypeScript, you could here to take a look at this as runtime to do this type of validation, and that’s not something that could necessarily—especially for things that can’t really be statically, for things that have multiple steps. So the other thing, too, is that talking about download caches is that really depends on your usage metric for an application in a lot of websites are things that you go to and use very infrequently, you go to a search engine and you go to a site that you load up and it depends on what percentage of time is spent on the home page versus deep within the application, depends on whether or not your bundle size matters. And download caches is kind of a complicated and tricky subject because it really depends on your usage metrics. I just—I do know that bundle size is important to many customers. And having really complicated, large emit for down leveling has been an issue, so it’s generally why we—one of the reasons we try to avoid it. + +USA: All right. Before we move on with the rest of the queue, I would remind you that we have around eight minutes left, and beyond that we’d eat lunch. First on the queue, we have a clarifying question by Justin. + +JRL: So the last couple of minutes of conversation have been specifically about TypeScript optimization passes, whether or not it can do anything with the type analysis it already has. I think these are good topics to have, but not on the current discussion topic we have. This is supposed to be about extractors. If any of the questions are explicitly about extractors performance and not TypeScript optimizations, then let’s keep those. But if they’re not, can we move on to another topic. + +SYG: So I think—it seems like a lot of the lines of argument here is that, and I’m willing to take this perfectly at face value from the TypeScript folks, that this is for architectural reasons for state of the world reasons, that it is difficult or undesirable to do this TypeScript. And I guess you think that would imply, then, that the engines are a better place to do this. But if the conclusion here is that something like extractors—sorry, not something like, if extractors cannot be done efficiently with all our giants for bundle size, runtime per form answer, even with static analysis, another conclusion to draw is it’s a bad fit for language given our constraints. Use case points still resonates that the languages that enjoy something like pattern matching and more sophisticated destructuring are all static languages with pretty sophisticated type systems where the—where the pattern matching end up mostly being a zero cost thing at compile time. And, yeah, it’s really nice. If you had that starting point, but we don’t have that starting point and we would like to get to a place that everyone can live with, but I want to remind that if the argument is that there’s just no realistic way to make it fast even with sophisticated static analysis tools, the conclusion is probably not that, oh, okay, then we should just do it in engines. I think I want to reiterate the complexity is moved, not removed point. + +RBN: So this in general—the expectation is that when you’re executing a function—or using an extractor you’re using yourself code, and it’s as slow as executing a function would ever be. If the interest is in can we make this faster through static analysis, that’s not something TypeScript could do. It is here for something that engines can do. Engines have more knowledge at runtime about the runtime semantics if well, for example, had a—had a very simple extractor on, say, the number constructor, that all it did was say, yeah, that number and then returns the number. You could here toette lake—an engine could here to ethically had the desire to optimized say, I’m looking at number, I know this the global number, I know this is what it does because no one tinkered with it so I’m going do that thing. For TypeScript, it requires a lot of complex code that has to be put in the output to be put in the bundle. For us that, static analysis could be done, it could be done to make things faster and increases bundle size, which is negative. Runtimes could here do this. And I’ve had hopes and even discuss—or tried—I believe I started some initial discussions with you about the possibility that in the future, if something like ADT enums were to exist that allude to you create data types that had very simple data structures, you had a known set of possible inputs and outputs that could be supported by pattern matching we might be able to gain efficiencies in the runtime because the runtime could know all these things and make benefits. Maybe that wouldn’t be something that on day one. Like in Mings, it’s going to be a feature that performance would improve as need arises and start adding optimizations to it. In the worst case, it wouldn’t be like if we didn’t have this feature and you had to do everything one line at a time. The expectation is this is not going to be slower, other than reassuring is slower than the native equipment. + +USA: We have only three more minutes left for this and the queue continues to grow. I would suggest now to leave time for consensus as well as the conclusion. So what about we limit rest of the discussion towards stuff that pertains to stage advancement, and the controversy queue could be captured and we could get back to that. + +RBN?: Correct me if I’m wrong, JHD, it shouldn’t be slower than current code. + +JHD: The thing I just wanted to say is that the,will—like, in terms of market pressure is and performance and browsers and stuff, it’s not an arbitrary number. The target to beat is “what are people going to do in the absence of the feature”. As long as it’s not slower than that, it should be “fast enough”. + +USA: All right, then. RBN, how would you like to proceed? + +RBN: Ashley, is your comment a quick one or longer one? If it’s a longer one, perhaps we can open an issue on the issue tracker. + +ACE: Very quick. It’s just the conversation versus transpiled versus runtime performance, would engines be interested in, like, implementation being done in a branch just to collect empirical evidence, so rather than guessing which one would be faster? + +YSV: One of my concerns is simply around the power of the proposal. And we might not be I believe to anticipate how people start to write their code. People might start writing applications in a way that makes them slower in order to have a certain mental model, whereas restricting this power might allow them to write applications in a different way. Like I said, we’re weakly negative and not blocking Stage 2. + +RBN: All right. So I’d like to go, if I can, to a discussion on potential advancement. I do think that going into the Stage 2 might be potentially a good signal that we need to start digging into some of these more advanced and complex concerns. So if that—at this point, I think, Daniel, you have a topic on the queue. So in -- + +MM: I just have a strong plus one for stage 2. All of the issues that we’re discussing here, I think we’re all agreed they can be investigated during Stage 2. Nobody’s expressed that these issues block Stage 2. So I’m strongly supportive of this. And I think that, you know, even if there’s a significant performance cost, if there’s a significant improvement in the degree to which people can write readable, reliable code because of the power of the syntax, to make things more readable and understandable, I certainly would proceed to use it even if there is a significant performance cost, and I think it adds more value than it subtracts for most users to. + +RBN: I know we are out of time, so at this point I’ll ask for consensus on advancement, and I will happily put a summary in notes when everyone else goes to lunch. + +DE: Yeah, I support Stage 2 for this proposal. And it will be important to investigate these issues that browsers raised during Stage 2. + +USA: Also on the queue, we have support for Stage 2 by JHD, Chip, as well as Chris. Yeah. Seems like we have consensus for Stage 2. Congratulations, and before we move on to the summary, let’s ask for reviewers for Stage 2. + +RBN: Yes, I very much need that. + +USA: Any volunteers? Well, I suppose you mean Stage 3 reviewers? + +RBN: Yeah, reviewers for the Stage 3 reviewers that I’ll need during Stage 2. + +USA: Yeah. Any in-person volunteers? JHD has raised a hand. JRL also has a hand up. + +USA: Thank you both. And anybody else, feel free to reach out to Ron. If you’d like to volunteer async, and, Ron, would you like to quickly summarize or conclude. + +RBN: Yeah. I’m happy to do offline. I think there’s a lot I have to go through to summarize because of the discussion about static analysis, so I’ll probably do that offline. + +YSV: Concern about performance impacts and power brings to language + +SYG: Potential candidate for JSSugar + +RBN: TypeScript already considering complexity of desugaring as it requires desugaring all surrounding destructuring, strongly hoping engines native implementations could be significantly more efficient. + +ACE: Type-directed emit would be a major issue for Bloomberg. + +YSV: asked if TypeScript can use static analysis/type information to do this at compile time. RBN/DRR: As a policy, TypeScript will not perform type-directed emit. Any static analysis would need to be baked into the desugaring, which is prohibitively expensive for bundle size. At worst, this should be no slower than normal function invocation, not accounting for any general desugaring overhead you might expect downleveling destructuring to ES5. RBN/DE/MM: Further questions on performance can be addressed in Stage 2 Proposed for Stage 2 + +### Speaker's Summary of Key Points + +- Addressed missing Elision in cover grammar +- Introduced NLT between ExtractorMemberExpression and ExtractorBindingPattern/ExtractorAssignmentPattern +- Continuing with Iterator destructuring while we investigate runtime performance w/option to switch to array-object destructuring (i.e., destructuring based on index) + +### Conclusion + +Stage 2. Reviewers for Stage 3: Jordan Harband, Justin Ridgewell + +## Decimal: Stage 1 Update + +Presenter: Jesse Alama (JMN), Nicolò Ribaudo (NRO) + +- [proposal](https://github.com/tc39/proposal-decimal/) +- [slides](https://notes.igalia.com/p/tc39-decimal-october-2024/) + +JMN: Well, welcome back from lunch, everyone. This is JMN presenting an update about the decimal proposal. I’m presenting this with NRO, he has helped quite a bit with the slides. I’m working on this for quite some time now with Bloomberg. Let’s take a quick look. Let’s just dive in. I know that many of you have probably heard this many times, but I won’t make aunderstands. Maybe there are some newcomers here. The point of the decimal proposal is to represent exact decimal numbers for purpose of eliminating or at least significantly reducing the rounding errors that are frequently seen when handling our friend’s binary floats with typical human numeric data, which is generally expressed in base 10. I certainly don’t compute my bills in binary. I use base 10, and that’s not the only place I use base base 10. In this world, we have something like 1.3 that really is 1.3. It’s not an approximation thereof. And in this world, if we do this classic example of 0.1, 0.2, we really get 0.3, so these are going to be the same quantity in the decimal world. They’re not—0.1, 0.2, 0.3 are not approximately represented, but they’re exactly represented. That’s what’s going on with this proposal. + +JMN: I presented this a number of times, and it has a long history in TC39. And so I’m riding on the shoulders of giants here. So I just want to shout out and thank many of the people who have been involved. Since the last time I did an update, there has been a lot of progress here. I presented this in TG3 a number of times, so I was fortunate to discuss a lot of this with MM, KKL and a number of other people listed here and the emissions are unintentional. We do have a guest speaker today. Based on some feedback that we had at the last presentation and even the one before that as well, there seems to be a decimal famously just filling up the queue. It gets heated because everyone loves this topic. And last time we had some discussion coming from JHD. I thought that we would let JHD take it away now for a little bit, because I think some of the insights that he has here are really quite essential for further discussion. + +JHD: Yeah, thanks, Jesse. I wasn’t able to prepare any materials so I will just talk. I had some conversations since the last plenary with JMN and DE in particular about Decimal. And I was made aware that my position on Decimal as a feature was probably not being made clear. This could have been a vision talk if I had made slides - the sort of vision I have for decimals in JavaScript is one day we can actually just type numbers that make sense in our programs and do math with them and it makes sense, and do all the mathy things and all of the 0.1 plus 0.2 things that everyone laughs at aren’t applicable anymore. That was my hope during the BigInt proposal, but that was sadly not its purpose, so it was not designed to be smoothly interoperable with Number. My personal belief is that that’s why it doesn’t have the adoption we were hoping to see from it, but nonetheless, that’s what I want out of Decimal or any similar numbering system, is to get to that point. In other words, I think that the problems that the Decimal proposal is trying to solve are very, very critically important and very, very broadly needed. Almost everyone does stuff with numbers, and the capital N Number is just not sufficient for almost anybody to avoid bugs and to do the things they expect, people constantly have to work around all these issues. So I am in full support of solving this problem. + +JHD: The reason for my objections has been that I don’t think that we can have this vision I want unless we end up with something that is a numeric primitive that can be smoothly and ergonomically used, and that is interoperable with the existing numeric primitives. I’m worried that we will find ourselves in a place where we have Decimal as an API and can’t ever get it as a primitive, and it’s not ergonomic enough to achieve the world I want as only an API. + +JHD: Some of the recent changes that I think JMN is about to discuss definitely remove some of the known obstacles to eventually having decimal primitives, which is good. That is wonderful, and I’m very pleased with the changes that have been made, that we’ll hear about shortly. I think I’ve expressed myself sufficiently about this specific proposal in previous plenaries, but I wanted to clarify that I think that if solved properly, this is a proposal that will be broadly useful and ease a lot of developer pain and allow for more correct programs. And to me, “properly” includes primitives. + +JMN: Thank you very much JHD, I appreciate that you took the time to schedule that. So the purpose of my presentation today, again, I’m not asking for stage advancement, but I’m an update, so part of the purpose of the presentation was to allow some of these more controversial points to get fully aired, and I thank JHD for taking the time to do that. But the other is to also present the didn’t of diff, where we are compared to last time in previous updates. So the data model for decimals for us, and the API that we have in mind, this is largely unchanged, although, I’ll give you an interesting diff in just a moment. + +JMN: We’re proposing to use an already well established standard that we’re already using, IEEE 754 binary floats, but there’s a part of it called decimal 128 which is about decimal arithmetic that I propose to incorporate into JS, at least a part of it. The idea behind decimal 128 is to have representation of decimal numbers taking up 128 bits, hence the name. So it’s a fixed bit width representation for numbers. And that’s nice because it provides a safe backdrop, you might say. So that one can reason about how much space numbers take up and how big they are. So this is in contrast with other alternatives like, quote, unquote, big decimal or some kind of arbitrary precision world where a decimal number might take up a lot of space and doing arithmetic on these things could potentially take up quite a lot of space and time. + +JMN: The idea behind decimal 128, and a nice advantage that it has, is we can just do arithmetic without having to, excuse me, reduce precision. One doesn’t have to worry about the possibility that an operation is going to somehow overflow or we don’t understand what the result might be, because we know that we’re working always with 128 bits, there’s a certain number of significant digits that are always available, and there’s an exponent range that’s available, so we can just do our operations and we know that we’re still within this decimal 128 universe. And the proposal, IEEE 754 specifies quite a lot of operations that could be done on these values. But what we are initially proposing is just basic arithmetic, so addition, subtraction, multiplication, division and a couple other thing, but that’s basically it, so the biggest delta that I’m trying to present, you know, today is the result of a number of discussions with many stakeholders in the whole decimal world, and the—its interactions with other parts of the JS universe. And it’s about those trailing zeros. The—just to sum it up, very quickly, the—one of the main issues is that in the IEEE 754 standard, the understanding of decimal numbers there means that one can have trailing zeros. So in this world, the IEEE world, things like 1.2 and 1.20 are actually distinct values. Also in that world, they compare as mathematically equal, but they are distinct, they are observably distinct. And it took a while, initially when we got started with this, we weren’t really thinking about those trailing zeros. We picked this up from an earlier discussion where this issue of trailing zeros had largely been settled, we thought. But after increasing rounds of feedback, largely coming from the Intl side, we thought that adding trailing zeros and supporting them actually made a lot of sense, and so we spent some time developing the proposal to support those trailing zeros. But one of the main takeaways for me anyway, at the last plenary, things that were—something that was a bit foggy to me but was made clear, is that supporting those trailing zeros in JS, beginning between 1.20 and 1.2 as decimal values actually rules out the possibility of a decimal would ever be a new primitive type, so, again, we’re not proposing that decimals be a primitive type. If you want me to say that many times, I certainly can, but that is not what we’re proposing. I’m imagining a hypothetical future in which we were to add decimals. This is something akin to what JHD was sketching. And in that world, we would have to make decisions about these numbers with trailing zeros, and they’re very unpleasant. So the thinking was, okay, we could support trailing zeros, but that leaves us in a sad state where decimals are just this API object, and we’re not going to been looking forward to a future that JHD sketched. So that wasn’t very nice. There are some nice advantages to having the trailing zeros. But there are some disadvantages too, and this is one of them. + +JMN: One of the issues with the trailing zeros is that or even just generally, you might say, talk about what’s called precision or quantum of a number. That’s the official IEEE word for what I’m talking about here. Is that one has to specify how this information gets propagated. So in my examples, I just say 1.20, but let’s make that more complex. And IEE does specify how precision needs to get propagated. So let’s take an example. Just to illustrate that. Let’s say we have something that’s $40 and there’s a sale tax that gets applied, which is 6.75%. You can use your calculator and see that that will cost you $42.70. Okay, that’s what you see on your receipt. But let’s imagine that we take a peek at the calculation internally in the decimal 128 official—official decimal 128 world, well, the press precision of a sum, so imagine we do the calculation as 40 plus and then we do a product 40 times the sales tax here, with addition, the rules say that we need to take the minimum of the two precisions, so we take the minimum of precision of 40.00 and the precision of that product. The precision of 40.00 is going to be -2. The idea being I have a number here with two trailing zeros. And then so it’s kind of like 10 to the power of -2 is the way you can think about that. Then you look at the precision of the product, and that ends up being the precision plus the sum of the two precisions, so again I have my -2, but then I have to look at 0.0675, that has a precision of -4. Oh, there’s a mistake in this calculation, I’m looking at the minimum of -2, and -2 plus -4 actually should be -6, right? Sorry, not -4, forgive the failure there. So actually, the result has quite a lot of trailing zeros in it. + +JMN: But, hey, we’re just calculating a bill. We know what the end user should see. It’s going to be 2, right? That’s the thinking. So we do this calculation, we preserve all these—this information, we follow the rules, we get a weird result with a huge number of trailing zeros. But in the end, we probably are going to throw away some of that information. And as you can see here, these rules, you can sit down and try to reason through them and try to justify them. A lot of these things are coming from, say, the accounting world, which has some rules about handling these trailing zeros. This is an insight I got from Mark, thank you. And the challenge here is that there’s no really good theory for these things. I mean, there’s something there. I mean, they are rules to be followed. But if this example puzzles you, then even more complicated examples will be even more puzzling. And you see this is a very simple example. Right? Things get very odd very quickly with this preservation of precision. + +JMN: So in discussion, we had this realization about these trailing zeros. Once we’re confronted with the fact this even for simple examples we might have numbers that have lots of trailing zeros that we actually can’t account for, that we don’t want anyway, we realize that the main intention behind supporting these trailing zeros is to make sure that at the end of the calculation, generally, we want to somehow specify how many digits after the decimal point are needed. But in that case, then we may not even need those trailing zeros. We can just impute those extra zeros based on some kind of outthat’s coming from the outside or input that’s coming from the outside. For instance, here, let’s do a calculation that’s complicated as you would like. ABC here. And then we say, well, whatever that result is, maybe actually it’s an integer and has zero trailing zeros or maybe it has exactly two or maybe has more than two, whatever, just give me that thing with two extra digits there after the decimal point. So we could follow the rules, but in the end, we probably have to specify things anyway. The idea that a decimal number in the official IEEE sense is going to somehow thread its way through calculations and in the end, we always get exactly where we want is very, very unlikely, almost impossible, really. Okay, but then we had to go back to the drawing board, and the question for us was could we really drop the tracking of these trailing zeros while satisfy all the use cases? That’s a basic self-check, right? And we handle things like human produce and human consumed numeric quantities like money. That’s one of the very first things that we’re trying to handle here, and I think that’s quite right. + +JMN: So imagine that we’re doing some kind of calculator computation, and we get a result, say, from a cash machine that says, well, you have 42.7—you have to pay that. Well, we know that that’s fine. That’s the quantity to be paid. And, okay, we have to impute some kind of extra digit there to get that extra precision, that’s 7 cents 0 cents, not .7 dollars, right? So that still works for us. We’re interested in data exchange in the decimal proposal. That’s also a main driver for us. And we still think that removing these trailing zeros, removing support for a precision is still fine. Although, I will note one thing here, that a loss of trailing zeros could potentially be problematic in some cases. But I’m not sure if that’s really that fatal because that’s already a problem for the numbers that we know and love, the binary floats. And third or tertiary use case for us, just to be a bit sloppy, more precise floats. Something that offers a bit more precision, more digits than we get out of the box with our binary floats, and that’s still true. + +JMN: So just to give you an instance of what life would be like without those trailing zeros, if we say that the answer is 42, well, 4 2.0 is going to be equals and what we’re proposing today in this universe to the answer, and if we do some erythema take, we say 42.75 and subtract 0.75. It’s not going to be 42.0 which it would be if you follow the rules of IEE, or even if it is, it’s still equal to the answer. That’s what life would be like without those trailing zeros. + +JWK: You mentioned without trailing zeros, 42.0 equals 42, does it not equal to when we have trailing zeros? + +JMN: Right, so you’re imagining the kind of—like a previous version of this proposal, where we did support the trailing zeros. So actually, that would still be true, because we would follow the IEEE standard, but there would be other ways to distinguish, say, things like the precision. Imagine that’s a property of the object, then we could distinguish that. There is an interesting word that comes out of the IEEE 754 world, which is this notion of what’s called cohort. And mathematically you can think of a cohort as a set of decimal 128 values that are mathematically equal, but distinct in that world. So there’s 1.2 and 1.20, so with one trailing zero, two and three and so on. And these are all in the same cohort. The proposal, the main diff that I’m presenting here today is that we do not distinguish between cohort members. Maybe they exist internally in the engine. Depending on, say, like the C library that you plug in for decimal 128, but those should not be distinguishable. + +JMN: We think that simplifies the mental model. Here we can think of decimal numbers essentially as points on the number line or quantities. Which is probably just speaking to what JHD suggested, I think that’s compatible or the same as what JHD was suggesting. It’s a kind of number, right? We believe that this is also less implementation complexity. I can tell you thinking about the polyfill, it’s going to be not the nicest place to do this comparison, but in the polyfill, it was simplified by removing support for the trailing zeroes and we think this is future proof for a hypothetical future, in which we have decimal primitives. + +JMN: The second main given diff that I want to present has to do with motivating cases for the trailing zeros. At this point, you might say, JMN, what about the original reason for supporting those trailing zeros? And I think we found an answer to that. Which is essentially farmed out into a new proposal, so based on discussions with SFC presented also in the 402 meeting, and also the incubated, you might say in the TG3 call, my colleague Nick and I and BAN also are working on a kind of numeric values with precision proposal. This is a work in progress name. I don’t want to get distracted too much by bikeshedding about this. But the idea is to have some kind of new primordial object that would represent numeric quantity, points on a number line, with some kind of precision attached to it on the side. So not part of the number. So this can be useful for display purposes. Think of 1.2 meters and 1.20 meters. In some contexts, those are distinct things. They convey different information. + +JMN: An example that has motivated us in thinking about these things comes from SFC, is to think about the difference between saying one star versus 1.0 stars, notice that 1.0 stars is plural. Despite being mathematically the number one, right? And then when interacting with external numeric systems that could preserve this information or want this information, then this numeric value with precision thing could provide that external information to support round tripping, which is something that I hinted at a couple slides ago. It’s something that might get lost when we always just kill the trailing zeros. And some contexts, that might not matter, but in some contexts, it does. + +JMN: Here is just a total hypothetical. This is very fresh for us. But we’re thinking about some kind of fractional digits method, name is CBD. I realize it’s quite bulky. But let’s just take a look at that. Imagine we had some kind of number that has a width fraction digits and we have a with fractional digits on decimals too. We could look at valueOf(), we could get some kind of underlying number object and get the fractional digits. Notice that in this decimal with precision we have some kind of type error because decimal primitives don’t exist yet. The numeric object could be the decimal thing. And so on. And we can format these things as well, so it has some kind of integration with Intl. That’s not, I would say, the main thing I want to dig into in this presentation. My main option. Is to essentially announce the existence of this thing as a Stage 0 proposal. And by the way, this discussion will be continued by BAN later on. You might have seen something about measures in the agenda. So the decimal proposal is this numeric values with precision proposal is overlapping with that one, and it’s a bit TBD. Maybe these are the same proposal and maybe they’ll get fused together, essentially. Work in progress. Watch this space. But in any case, the whole point of what I’m trying to say is that this idea for trailing zeros has kind of been off loaded and addressed. + +JMN: So I’m just trying to wrap up here. There have been some concerns about decimal that I’ve heard over the almost years now that I’ve been working on this. We believe that we have addressed these. So when I first got started on this stuff, taking over some—from some earlier colleagues and who have done some work on this, initially the idea was from day one, that decimal would be a new primitive type, but we got feedback that this is a bit heavy. We’ve taken that to heart and for quite some time now we’ve downgraded the proposal you might say, so that is now a standard library object. We have removed our support for quantum or precision, if you want to think about it differently, this trailing zeros issue. But that clears way for operator overloading in the future, which is again not part of the current proposal and shall simplify things a lot. I think it’s offering a mental model that many developers probably expect from decimal numbers. Again, this is speaking to JHD’s point. And then with this numeric value with precision thing, we think that we’re also satisfying some of the use cases for trailing zeros coming from the Intl side, and possibly other use case as well. + +JMN: So just to wrap up, this can basically go in the notes as the conclusion. So we’ve almost gone full circle, although not quite. We’ve simplified the data model for decimal by removing the support for quantum precision. Intl integration is largely off loaded. Stay tuned there, you’ll hear more about this in this plenary. The spec text is available. Some issues were identified just very recently by WH. I’ll work on that. Polyfill is available, so from where—from where we’re sitting today, we think that we’re essentially ready to ask for Stage 2, but we’re not asking for that today. + +JMN: We just wanted to leave some room for discussion. JHD already kicked off some of the discussion about any concerns there. But that’s where we are. So we think we’re kind of epsilon away from Stage 2 at this point. And that’s all I had. I’m ready to take a look at the queue. + +SFC: Great, yeah. Thanks for the presentation, JMN. I had another item that disappeared, so I’ll cover that one here too. But I guess my main—I guess my first comment here is I feel like we’re getting ahead of ourselves by trying to figure out, like, the constraints of a primitive type and operator overloading and that type of thing, like, in a world that is at some point in the future, and we should focus on we have now, right? We should focus on building a really good, you know, object that does decimal 128, right, and sort of decouple these things, and the number with precision proposal, like, sort of is a step in that direction, but, like, I think we should really be thinking a lot about, you know, sort of focusing on that now, and leaving open the door for a completely standalone decimal proposal in the future from—by not forcing those invariants, because the things that, you know, MM and JHD and others have said that, like are requirements for primitive are absolutely not requirements for the object, and then it’s it sort of makes the object less capable when objects are, by definition, something that are more capable than primitives can be. + +NRO: I think if you wanted to drop the primitives in the future, we should be—I mean, it’s a world in which we don’t have two different decimals. Where one is just, like, not just—like, the object wrapped around another. Because that will be a trigger for numbers in the language. And while today we don’t see a way to have primitives, I would still like to consider that possibility to be realistic one day, and having two different decimal types is just, like, not something that I wish for JavaScript to give us. + +SFC: Yeah, I don’t mean to have all my topics right next to each other, but, yeah, I mean, I would like to see these proposals, like—it—I don’t want to be in a situation where we base—like, where like, you know, the champions of this proposal basically said, like, oh, we have this other possible approach for numerics with precision, you can go solve it this way, go, shoo, get out my way, I’m wanting to move—force my way forward with this decimal proposal in a way we know doesn’t satisfy your requirements. I don’t want to be in that situation. I want to be in a situation where we have—where we’re introducing into the language a, you know, well-rounded solution to how you represent decimal numbers, right? And, you know, that—where we really think about the big picture. Because if we—if we force through this weird decimal object thing that’s designed for a possible future primitive, like, you know, like, what does that mean for, you know—for the—the ability to represent precision and other—you know, its impacts on insill. That seems silly on its face. So, like, if we want to move into this direction of having two objects, I wanted to see those coupled. I don’t want to see at some some future stage proposal, I want to see that together in one slide, like, here is how you can do these things, so that’s my next topic. + +SFC: And I have yet another topic. Sorry, I didn’t mean these all to be back to back. There were people in between me and they all left the queue, so I have my topics back to back. I apologize for that. + +NRO: I can reply first before you move. + +SFC: Sure, we have a reply. Okay, good. + +NRO: So would you be okay—like, would but okay with having two separate proposals moved together? I believe that the concerns you’re trying to solve for decimal, while overlapping, they’re, like, we need to focus on different aspects in order to—for the two use case. So we’d be okay with two separate proposals that move step by step, like, side by side in the process instead of actually merging them. + +SFC: I mean, that’s a procedural concern. I mean, if there are two separate proposals, it means the committee could agree put one to Stage 2 and the other one, this is a silly proposal, I don’t any think it belongs in the language. We’re not going to get it to Stage 2. That is why I would be concerned about that. + +CDA: I had a reply on the queue, which is that is it fair to say that it’s truly being designed for a future primitive or is it just being designed to not make it impossible to introduce a future primitive? Because I think that’s an important distinction. + +SFC: I say that’s a question for JMN, not for me. + +JMN: Yes, that’s right. And I take SFC’s point that—as well that this might be a kind of awkward object. But I guess the issue is, then, I mean, yes, I guess narrowly speaking we trying to not make it impossible. But if there’s some other—other elements that we’re missing here, I think we’re happy to incorporate them. + +NRO: Yeah, so the proposal is not designed to behave exactly as the object of a primitive. Like, it behaves exactly like the number object, for example, except it has the various prototype methods. In another potential world that was the idea was pushing back against earlier where we have some type of decimal object now, and we say that doesn’t actually preclude us from having a primitive, the future primitive, just a different way. Like, something unrelated. I believe that’s the, like, worldview that you might consider to be just designed to not conflict with a primitive. While the one now, where we—the one I was, like, it’s designed to fit nicely with the decimal primitive. + +SFC: So, yeah, my next topic, so, you know, I appreciate that, you know, JMN and NRO have been, you know, trying to assuage my concerns with this over numerics with precision. But I just want to emphasize here, and I haven’t, you know, in my discussions with NRO and JMN and other, I haven’t really heard these topics be addressed, that, like, this the not just an Intl concern about precision. This concept of being able to represent, you know, the numbers with variable levels of precision has many other use cases, and I’ll just list a few here. So one is, you know, when we look at, you know, precedent in other decimal libraries and other languages, it’s, you know, basically the standard that, like, every other decimal or big decimal library you find in another programming language support this concept, and it’s been that way for a very long time. And given that—given that precedent, it means that, you know, I think JMN sort of—I very much disagree with the slide about halfway through that said, like, the 1, 2, 3 with big green check boxes. That’s just patently false, because we cannot round values to these other platforms if we don’t support precision in the data model. + +SFC: We just get—yeah. yeah, if you can go to that slide . It’s a slide with three little check box emojis on it. Yeah, this one here (https://notes.igalia.com/p/tc39-decimal-october-2024#/10). Okay, so on the second bullet point, like, might be problematic in some cases is just a—is just a way of saying, it does not actually work all the time. And if it does not actually work all the time, it does not round trip. It round trips or doesn’t. It’s black and white. It’s not a gray area. It does not round trip, that’s just objectively false. So that should not be a check mark. Right? So regarding the first one, human produced numeric quantities, like, you know, the Intl example that you showed shows that this is also patently false. This should not be a check box because it’s clearly false. And, yeah, we could—we don’t have to get into argument about that right now. But, like at least these first two bullet points, like, those are absolutely not check boxes. They should not be check boxes. + +SFC: Okay, but going back to some of the other things. So IEEE decimal 128 has precision in it and it defines how precision works, right and we may not agree with how the IEEE standard does precision, how it propagates it, how it uses accounting roles, we may or may not agree with that be we request debate about whether that’s the right decision or not. But that’s the decision they made and that’s the press precedent we should follow. If we want to invent our own, like—by saying oh, we represent decimal 128 except we don’t do precision here and except we don’t do this or don’t do that, we’re not really following any standard, right? If we want to follow a standard for this, then, like, you know, maybe we should go and talk to IEEE and have them release a revision that says here a binary representation that doesn’t have precision because maybe that would be more efficient representation-wise. If you didn’t have to have trailing zeros in your data model you might be able to represent bigger values or something like that. But just as an example, like, by not following decimal 128, but saying we’re going to explicitly deviate from this and go into this other subset, that’s extremely restrictive and it’s not like decimal 128 is super, super widely available anyway on chips and stuff. So, you know, like, you know, what could make the justification that for floats, we need the deal with NaNs are all equal with each other, but with by we still restrict ourselves with floats. This a completely different type of questions because floats are widely available in hardware and there’s a lot of reasons we want to be able to have an implementation apply to them. That argument does not apply to decimal 128. And is last was that precision absolutely does have applications in scientific accounting. There’s been replans the thread with the repository and people saying, yes, I actually need precision. It’s not just—I mean, I keeping about this for the Intl perspective because that’s what I’m officially here for, but use cases outside of Intl are also very easy to come by, so this proposal does not serve those use cases either. + +SFC: Now, all that said, I’m, you know, as I said previously, I’m very much in favor of, you know, having, you know, a holistic design, you know, that sort of, you know, if we want to decouple these two concepts, I’m not opposed to having that sort of holistic design. I want to see that holistic design. I’ve seen catches of stages in the proposal and that’s about it. So, yeah, I think that by making this—these changes, we’re sort of, you know, taking this proposal, this decimal 128 proposal sort of away from what it was originally proposed to do. It no longer is able to represent human produced numeric quantities and no longer able to participate in data exchange. It can still do more precise floats, exercise is great, but that’s the use case that was lowest on the list when I saw this. The numerics with precision proposal, if we see it as one big picture covers the use case if we see them together, and that’s why it’s important to keep the processes together. + +CDA: All right, just noting we’ve got about 15 minutes left. There’s a reply from NRO. + +NRO: Yeah. So actually I kept it in the queue moving and I want to reply different to SFC here. One about deviation from IEEE 754. Not in this case, where, like, the only deviation here is they’re exposing less info. For example, the equality within the spec already, it’s the number of trailing zeros, and the only thing we’re doing here is avoiding that precision to the user. Like, all the other—like, the way operation work, the way numbers round, like, we don’t have infinite precision here. So addition to numbers will round, and that will also round exactly how it’s defined at IEEE spec. Regarding the slide with the three check marks, yes, you’re right, this should have been a somewhat yellowish check mark. There are many cases where you actually do that exchange and talk about money is and do not care about, like, with exact precision number, because very often you are, like, working with some fixed precision, you’re working with dollars, you have fixed precision of two decimal digits. But it’s—it is true if cases where you do care about precision as defined by, like, about a number of years, in this case, this proposal does not cover it. + +NRO: Yeah, and that’s kind of like what we tried to get at with the first slide of the numeric values with precision idea. And that – + +CDA: Sorry, YSV. YSV is requesting that we advance the queue rather than adding replies. + +PFC: This topic is a reply, so if you’d like me to put it as a separate topic, I can. Otherwise, I’m going to reply to SFC. A 'number' and a 'number with precision' are two different data models that serve two different purposes. For some applications, you need a number, for some applications, you need a number with precision. It’s not the case that some object with the data model of a number is 'secretly' a number with precision, but missing the precision. That’s not how it works. There’s nothing inherently wrong with Decimal having the data model of a number and then having some other object to represent a number with precision. As for propagation of precision, that depends also very strongly on your application. Before I worked in software development, I worked in physics where I can think of, like, two different procedures for error propagation right off the bat. Which one you use depended on what you were using it for. I believe neither of them were the same as the one in the IEEE scheme that Jesse presented in the slides. So I think for me the most useful thing here would be to have something with the data model of a number which you can then add precision to, tailored to your own application. + +WH: Quickly addressing a few of the prior points: This very much is IEEE 754 spec. It’s just that we don’t expose all of the IEEE methods, and we never did, not even for Numbers. IEEE has global mutable state in it — global flags and rounding modes. We never exposed those, and that’s fine; that’s a reasonable thing for a language to do. + +WH: Now, I’ve done yet another extensive pass over the Decimal proposal and identified a few dozen issues. Some of them will require a bit of work, but none of them are controversial. But my main impression is that this is now *so much simpler* than the proposal was when it had support for quantums. The proposal is way shorter, it’s way simpler. Weeks or months of work and arguing about edge cases has just completely gone away. So I’m really happy with the simplification. + +YSV: So I was the one who was adding stuff to the queue and then taking it off repeat lead because I wasn’t sure if I should say it or not. I was watching the slides and thinking, okay, I think they addressed this, okay, maybe not. We—the general standards community at Mozilla has been having some concerns about how much display logic Intl has been taking on. And potentially without enough coordination with other parts of the web platform. So when I saw that you had decoupled the display logic from the behavior, let’s call it that for now, of how decimal works, this was actually a positive in my perspective, and it allows JavaScript to, like, you know, the core of JavaScript to focus on making decimals available to really wide number of potential consumers. Now, Intl is currently limited to browsers, and we could also—well, not limited, but browsers are the main implementers Intl. It’s not necessarily available in other JavaScript runtimes. Having a relationship with that is better, all though, I would say that doing this work, we should be really coordinating more with standards bodies in W3C in order to make sure that we are taking into consideration the things that they’ve come across and the work that they’ve done making human readable and consumable data types available. Because there has been a lot of work in that space. + +YSV: I would say I would be opposed to this prose building if it tried to land human produced and human consumable numeric quantities as part of JavaScript core. Rather than as something that is sort of delegated to a space in between, because JavaScript is just available in so many different places. + +AKI: If you want help figure out how to make that communication happen. I can absolutely facilitate that. + +MM: I’ll speak for KKL since he cannot unmute. The proposal rightly decouples value from display, allowing a wide variety of propagation systems rather than making one default king. And to expand on that, there’s—and this also expands on the comment earlier about use of numbers with precisions in physics. Is there many different notions of propagating precision or error bars, and the—which one to choose really depends on what you’re using it for. So I think that first of all, I would not like to see any one number with precision be put into this proposal. I think that’s really too much coupling. So many of us have stated good reasons to keep it separate, but I think in addition, the fact—there’s a mechanism and policy separation. By keeping it separate, we allow multiplicity of those things, and we allow user definition of those things. So I don’t—so other than allowing the possibility, I don’t see a reason to couple them. + +SFC: Yeah, just to respond to YSV’s point and also some things other people have said. To be clear, I agree that I think that these are sort of, you know, two different concepts, like a numeric value on the number line versus, like, you know, an upgraded numeric with precision and other attributes on it, I agree throws are two different concepts, and those could be two separate objects in JavaScript. Like, that’s fine. I’m aligned on that, and I have been clear about that. You know, that’s my, like, you know, NRO’s, you know, Stage 0 proposal seems like, well, that’s maybe a reasonable direction to go in. It’s just, yeah, I mean, you know, I’m here because, you know, Intl is a 402 thing. It’s not a web WC3 thing. This is the body that standardizes these things. You know, to Temporal is, you know, a great example of, you know, we really engineered that to be the input type for, you know, Intl logic, and I think we’ve done a really good job with that. And the web platform currently lacks, you know, a number type, a numeric type that’s actually appropriate as an input for Intl, and we listed out the reasons on the thread. And, you know, this is the—like, this type, you know, has been, you know—we should have that type in the JavaScript language, and, you know, it’s not just a WC3 thing, although I do definitely do think we—although I think we should be having more collaboration, because it would be nice the, for example tie this with, you know, a number picker or thing or other things like that. It would be nice to tie the Temporal things with date inputs and things like that in WC3. So I do definitely think there should be more collaboration here, but I don’t think the argument of, oh, well, this should be a WC3 thing, therefore it shouldn’t be an ECMA thing really applies, because ECMA is Intl, Intl is ECMA. + +YSV: I didn’t say it shouldn’t be an Intl thing. I said we should be collaborating with WC3 more, because there hasn’t been enough collaboration. Two proses you listed, for example, Temporal, didn’t have any communication with the author of the time element in HTML, where we really should have done, that because that person also has a lot of knowledge and experience, and in addition, making those two—two interoperable will mean that people who aren’t using JavaScript will be able to benefit from the things that happen in Temporal. The same thing applies here. Like, the fact that there is no appropriate number element doesn’t mean we shouldn’t potentially taking some of our learns collaborate with W3C and get that into HTML. In fact, that’s a reason we should be doing that type of collaboration. By moving that logic into the content layer, that means it’s available to place no that are not just necessarily JavaScript based. + +CDA: Okay. That’s it for the queue. + +JMN: Okay, great, thank you. This is fantastic feedback that we will try to incorporate into our next presentation. I think many of these points can be addressed. For instance, I think probably this—when I think about the slide here that’s up, and the idea that we’re talking about producing human consumable, numeric quantities, I wonder if this is the best way of putting. I think perhaps more pressure should be put on the idea of exactness, so that is really a property of numbers here, which is probably an umbrella or a cover of what we’re trying to produce there in the first point. In any case, that’s my job to take in input. Thank you very much. Happy to discuss this further in the hallway tract if you like. You can take the conclusion slide or I can do that and put that in the notes. No need to do that here. Oops, there’s a new topic. + +SFC: Yeah, I just warranted to fully with what you just said Jesse. Stating the goals extremely clearly, which you have done in the readme, by the way. It’s super important here and sort of thinking about what proposal solves those use cases I think is super important. If it’s about exactness, great. Like, I think it’s extremely clear—I think it’s extremely important to be clear about that. And I think, you know, I’ve noted previously that something like numerics with precision has a very easy path to be able to represent exactness. If that’s really the goal. So, you know, like, I think, you know, we need to be—I think it’s really important to take a step back and sort of say, well, what are the goals here? We've been focusing—I feel like we’ve been focusing very narrowly on decimal128 as, like, the only way to solve this problem, but there’s other ways to solve the problems too. And we really need to be focused on what’s driving this and not just on this one solution we want it to move forward. Thank you. + +### Speaker's Summary of Key Points + +- Decimal remains at stage 1 +- Support for IEEE 754 quantum/precision has been removed (mathematically equal IEEE 754 Decimal128 values will behave entirely the same way, though with different object identities) +- The Decimal champions propose a spinoff proposal for “numeric value with precision” that can address different ways of encoding precision +- The decimal champion group believes it is close to asking for stage 2 + +### Conclusion + +- Feedback was expressed that it is unclear whether the current design, with canonicalized decimals, meets the stated needs of the proposal. +- The champion group will iterate on the current design and clarify the relationship between it and the newly proposed [measure](https://github.com/tc39-transfer/proposal-measure) proposal. + +## ArrayBuffer construction step order + +Presenter: Richard Gibson (RGN) + +- [proposal](https://github.com/tc39/ecma262/issues/3388) +- (no slides) + +RGN: So it turns out that ArrayBuffer construction is surprisingly intricate in how it deals with two arguments and a bit of NewTarget constructor information. The algorithm walks through the ToIndex consumption of byteLength, ToIndex consumption of maxByteLength (when provided), and then goes through some construction steps before revisiting the byte length and the max byte length for final validation. So by my count, we end up with at least eight possibilities for abrupt completions that terminate the operation with a thrown error. ToIndex really obviously when the type doesn’t match or toString/valueOf/Symbol.toPrimitive throw for both byte length and max byte length, and for maxByteLength the options object may throw on Get. So this is early argument processing of the sort that I think we generally like, although when I went looking for the past conclusions in how-we-work, I couldn’t find them. I don’t remember exactly where that effort fell. + +RGN: But proceeding on. In the AllocateArrayBuffer construction step, it checks that the maxByteLength if it exists must be greater than or equal to byteLength. So this is still early argument validation that throws if something doesn’t work out. Then we get into OrdinaryCreateFromConstructor which starts to deal with things like “can I Get a prototype property off the constructor? Is it a revoked proxy?”. Errors here can come up in the case of subclasses or Reflect.construct. Then assuming OrdinaryCreateFromConstructor succeeds, now we revisit byte length and max byte length to validate that there’s enough memory on the system and available to the process in order to support them. So if byteLength is too big to instantiate a Data Block, we get a RangeError way down here at the bottom of the algorithm. And then likewise for maxByteLength. So we’re processing the arguments up front, then jumping into some constructor-oriented steps, and then down at the bottom again processing arguments. This is how it is specified. I think it’s a little bit messy. But if we want to go with it, that’s fine. However, implementation reality is that we’re mostly *not* going with it. So I will switch now and look at actual behavior which is basically that only SpiderMonkey got it right at time of writing, since expanded to include GraalJS and Moddable XS. + +RGN: Graal, XS, and SpiderMonkey get it right. JavaScriptCore and V8 at different points are reading maxByteLength when they should be throwing a RangeError about byteLength. Hermes is throwing the wrong kind of error from checking NewTarget before byteLength, and engine 262 and QuickJS are not reading maxByteLength when they ought to. So we have five different cohorts of behavior, of which only one matches the specification. One thing that I’ve worked on in order to produce this data is a push for better ability to cover it in Test262. I’ve been working in that group for quite a while in trying to sort out how we get through these sometimes complicated algorithms in order to validate that we are actually covering all of the normative requirements including step ordering. That will be coming at some point in the future. And this is kind of highlighting the need. But it’s also an opportunity in the case of the specific algorithm to revisit—are there any normative conventions that we can apply? Are we happy with this? Do we want to perform it a little bit differently? And just in general, is this a stepping stone to those kinds of conventions that we really want? So I will go back to the browser window again. And basically open it up to the queue for commentary. I’m okay if there’s not a firm conclusion on this. But I did want to draw attention to the specific example of a possibly general problem that we’re thinking about in Test262. I see that SYG is up first. + +SYG: So I just got lost in the example that you showed with JSC and V8 from this numbered example. Which step do they deviate from? + +RGN: All right. So this is unfortunately going to be hard to read and I apologize. But let me see what I can do. Actually maybe I can copy this into a web editor, that might be the way to go. All right. Bear with me for just a minute. + +SYG: I think it will be more helpful for me to see the actual test. Like, I don’t know what is printing those things? + +RGN: It is not as mature as it ought to be yet. So what I’ve done is create a test using collections of cases ordered by the errors they should cause. You know, what are the expected points of user code execution and then what is the expected result? So first, byte length iterates through a sequence of invalid values and then valid ones. Likewise for the requested max byte length and then as the test runs through the values it captures a log and the resulting error. This is the kind of pattern that I want to introduce into Test262. And the result is that as we’re iterating through those values, we end up seeing where implementations jump the gun. So in the case of V8, when we provide as byteLength a value with a `Symbol.toPrimitive` function that returns a number too big for an index and as options a maxByteLength getter that throws and as NewTarget an object for which getting prototype throws, we expect that we see a Get of the `Symbol.toPrimitive` from byteLength followed by a RangeError from that function returning a value that is too big to be considered an index. Per the spec algorithm, we shouldn’t get to any interaction with maxByteLength. But what we see in V8 is that after reading the `symbol.toPrimitive` function for byteLength, rather than immediately invoking it and validating its output, there’s a read of the maxByteLength property off the options object. Does that make sense? + +SYG: I see. So this is showing that the range check in to index is not throwing as early as the spec says it should throw? + +RGN: Correct. + +SYG: Got it, thanks. + +RGN: And then in JSC, the first failing case, when the byte length is a `symbol.toPrimitive` that returns negative infinity and the max byte length and new target are throwing, what we expect to see is actually exactly the same situation, the negative infinity return by `Symbol.toPrimitive` should result in a RangeError because it’s not a valid index, but instead we’re reading maxByteLength first. So they don’t fail at exactly the same place, but it is the same kind of failure. And this is something in the engine internals that we can investigate and I want to see this coverage in Test262. But having the discrepancy allows us to address the algorithm... I want to see argument processing up front and put 7 and 8 after 4 or potentially before 4 just to have a general pattern where we deal with arguments completely before moving downstream. But maybe this existing spec really is the way we want it to be. We do have three implementations that conform with it. + +CDA: We are nearly at time. MLS, can you be brief? I see you’re unmuted but we cannot hear you. We still can’t hear you. + +RGN: The topic is actually—I’m going to pre-emptively answer and then if we get MLS, so much the better. + +RGN: JSC and V8 deviate in similar ways but not identical ways. I don’t remember which one comes first between these two test cases, the returning negative infinity versus returning too big. One gets it right and one does not. But then farther down the line, the other also gets it wrong. Similar but not identical. + +CDA: All right. We are at time. I don’t know if you noticed in the queue OMT is asking for the JavaScript file. So maybe you can follow up with him. + +RGN: I will be posting this as a Test262 PR with this case in mind. + +### Speaker's Summary of Key Points + +Implementations differ in the ordering of property access, type casting, and bounds checking for the ArrayBuffer construction algorithm, as reported by [ECMA 262 issue #3388](https://github.com/tc39/ecma262/issues/3388)—although some of them do accurately follow the spec. This divergence was not caught by Test262, due to general gaps in the patterns used to verify step ordering. The committee declined to change the specified behavior, and will instead push for implementations to conform with the existing algorithm. + +### Conclusion + +The semantics of the ArrayBuffer constructor remain as currently specified. Test262 maintainers will consider new patterns for increasing coverage of such cases. + +## `Promise.try` for Stage 4 + +Presenter: Jordan Harband (JHD) + +- [proposal](https://github.com/tc39/proposal-promise-try/issues/15) + +JHD: All right. So as a reminder, this is `Promise.try` and takes the callback function and variable list of arguments that are optional. It just invokes the function and then if the function throws, it gives you a rejected promise. If the function returns something, then it promise-resolves that. This proposal has an approved spec PR, it is implemented in Cloudflare Workers and bun and node and Chrome and it is in Firefox behind a flag in version 132 but in 133 and 134 I believe will be unflagged. LibJS and Boa and Kiesel have it implemented and WebKit has it is behind a flag, and has removed the flag. Hopefully we can get Stage 4. + +CDA: The queue is empty. + +JHD: Anyone want to affirmatively support this? + +CDA: Lots of plus ones with NRO and OMT and YSV is plus one. + +JHD: I will take it. + +CDA: LGH says biased after doing three of those impls but plus one. And RKG with a “whee” and MF with “meets the requirements”. + +JHD: Thank you. + +CDA: KM with the plus one. I support this as well. Do we have any more whoop or yeehaw or? We have WH plus one. KM. Tom as well with a plus one. All right. I mean, pretty close. But you got over the line. + +### Conclusion + +Congratulations on Stage 4. + +## Explicit compile hints + +Presenter: Marja Hölttä (MHA) + +- [explainer](https://github.com/explainers-by-googlers/explicit-javascript-compile-hints-file-based) +- [slides](https://docs.google.com/presentation/d/1X1k-rzMlqLS3MC-pgogVJZqPTIfUyLIo7H4TOFhCStg/edit?usp=sharing) +- [spec draft](https://explainers-by-googlers.github.io/explicit-javascript-compile-hints-file-based/) + +MHA: Hi everybody. I’m MHA from Google and will talk about compile hints and they are in the JS code and which functions would be eager and lazy. I will tell you in a minute what it means. This is part of the real world web performance work and one insight on the work is parse and compile times are a significant proportion of the cold page load time, when the JavaScript file is not in the cache. Not seeking to standardize this in TC39 as it doesn’t change JS semantics. But I wanted to inform you all, this is something we plan to do. So some background information about lazy and eager compilation. + +MHA: The idea of lazy parsing / compilation is to do minimal work until the function is called. When we are compiling the script, we need to preparse to find the function end, and this is kind of nontrivial in JavaScript and we can’t count parens or anything like that. We need to parse the full syntax. And in V8 that is done on the background thread and interleaved with the network load. If the function is then called, then we need to do the work and parse the function and compile the function and we need to execute the function and all of that has to happen in the foreground thread because the foreground thread is waiting for the function to be called. At that point we can’t really do much else. And eager functions are the opposite where we do the work up front so when we are compiling the script, we actually parse and compile the function. Again in V8 this happens in the background thread interleaved with the network load and then when we call the function we only need to execute. That still has to be in the the thread; we don’t have background JSexecution. The default in all browsers is the functions are lazy. This is because websites ship a lot of code that is not needed at least not during the initial page load. It might need it later, iIf the user does a certain interaction, or it could be completely unneeded. So this is kind of the web reality. The browsers have to be lazy or otherwise the performers would be really bad and all browsers do that by default. + +MHA: And it actually matters which function we peek for eager compilation. I did an experiment and loaded the Web page and recorded which of these lazy functions were actually called. Then I load the page again but this time I compile those functions eagerly and then I measure parse and compile times, foreground and background. I did this with 20 popular Web pages. There’s the link in the slides if you want to read more. And almost all, 17 out of 20 pages improved, the average improvement being 630 ms which is 74%. of the foreground parse and compile time. We add time in the background and but is propositional. We don't want to decrease foreground time by adding an unreasonable amount of background time, but this is not the case here. Two pages are so dynamic that compile hints don't help. They use patterns where the JavaScript they generate is somehow unique and the JavaScript filenames are unique. We just don’t recognize any functions during the second load. This doesn’t mean that compile hints wouldn’t work for those pages. This means this experiment set didn’t work. You might ask what is the remaining page? The remaining page is the page that doesn’t have that much JavaScript in it. It was a popular Web page so I included it. + +MHA: We also did a prototype experiment with Google docs where they just select the core file for eager compilation, just the whole file. They report their user page load metrics improve around 5%. They have many different metrics and percentiles of the metrics. I wanted to add one number of the slide that is representative. And that’s like 5%. These are userland loading metrics. They are related to when the web page is sort of loaded enough or visibly enough and things like that. So the previous experiment was about selecting functions and this prototype is selecting the file, the whole file. + +Another useful piece of background information is the PIFE heuristic. We have the immediately invoked function expression where we have a paren, the function body and the call. This function is immediately invoked. PIFE is a potentially invoked function. We have seen the parent before the function. And at that point we need to make the decision: do we eager parse this or lazy parse this? We don’t know whether it’s actually called. We saw PIFE potentially invoked. Chrome and Firefox treat PIFE as eager and eager parse or compile it. Safari doesn’t. And developers are sometimes using this heuristic for forcing eager compilation, there is precedence of doing this. This is not great. In particular we can’t add parens to methods. And it forces using function expressions rather than function declarations. If you want to not call the function but want to have it available, you need to do let A equals and then the function. This assignment typically has the run time cost and we need to do the assignment during run time. And really just a huge hack to use PIFE to force eager compilation. Still developers are doing that. + +MHA: Question at this point why don’t we just cache better? The caching we mean caching the compiled JavaScript code. All engines do this. Some of the benefits of this project can be achieved with better caching but not all. In particular the cold loads when the script is loaded and we don’t have it in the cache we can’t do anything. A relevant amount of script loads are cold. Depending on how you measure, this is around 15%. Some Web pages update frequently. Some update once a day or more than once a day and that means the user gets the code version at least daily or more. Some use uncachable code patterns and generate the code on the server side. And also caching inline scripts is hard. So if you have your JavaScript embedded in the HTML page and something changes in the page, it’s hard to reason about whether the script changed and if the cache is still valid or not. It’s just difficult. It’s also difficult to find optimal time for creating the cache. We might cache really early and the cache would be smaller but not contain as much functions needed for potential user interactions. Or we might cache really late so that it’s more comprehensive and bigger and contains things that the user just did during that page load but doesn’t do during the next visit, so it might contain unnecessary things. + +MHA: And one interesting thing we found working on this was that Google docs was telling us that many people load the doc in the background tab. And might be difficult just to write the cache in that case because the background tab loading might be a different code path. So if we have the background tab, Google docs load and try to generate the cache, it might not be representative if the user then loads it in the foreground the next time. And this is a really interesting insight because this is something that we would never have figured out based on benchmarks. Like, it would never occur to anybody to write the benchmark to exercise this scenario unless the web developers told us this is what happens in the real world. + +MHA: So before this, we worked on minimizing the cost when we get the eager and lazy decision wrong. There’s the blog about it if you want to read more about how we did it. We also improved caching. In particular we create compile hints so we cache information about which functions were called and then use the information. But we couldn’t quite get the same benefit as we could with explicit compile hints approach. So at this point we think it would be useful to have a developer signal telling us which functions should be treated eagerly and which not. + +MHA: And the plan is not set in stone, is it’s more like we want to do something along these lines, to do it with magic comments in the JavaScript file. First get started with the version that is marking the whole file for eager compilation because web developers tell us this is easy to get started with and might have a natural core file they should mark already for example. But in the future, I want to be able to add the hints per function basis. So here the plan to have something that encodes the position information for the eager functions. For example, add in some encoded format in the magic comments but the exact format is not planned basically at all at this point. We also considered alternatives to doing this information as the magic comments. So use eager, we could have used an eager directive at the top level or before a function. But the down side of this approach is the source code bloat is pretty big especially in the function version. For the network transfer would not be so bad because it’s basically compressed away. We still need to parse it and process it in the parser. That’s not really doable. We also considered transmitting this data in HTTP header or script tag and then the feedback about this approach was that this gets out of sync with the source code easier than if actually in the source code. + +MHA: So we assumed this works so that the web developers use profile guided optimization to figure out which functions are needed. Select for eager compilation and then for file based compile hints might construct a core file and mark that for compilation. With per function compile hints, they can add the per-function data directly. So for the web platform, this is probably going to be a chromium only feature first and just want to keep it generic so that other browsers can adopt it later if they want to. We got good feedback and discussions with other browsers during TC39. We don’t have a plan to create a standard for it and it should be always allowed to ignore the compile hints. They are just hints and don’t enforce anything. It’s really hard to standardize anything like that because lazy and eager are just like implementation detail. It can’t really easily standardize that the browser has to do background compilation for example or compile in a certain way. Also removing compile hints from the web platform is compatible. At some point we don’t want this anymore, the browsers can stop following the hints and then web developers will stop using it. + +MHA: Obvious risk is that the web developers are over using this just like the user sort of eager compilation directive to everything which is not a good idea. And at that point like the risk is the engines might need to start ignoring it if they can’t trust it. It’s an interesting question, should we prevent people from shooting themselves in the foot with this or not? Philosophically and practically? Another question is will there be optimal set of hints that apply to all browsers? Will the developers in the position to hint for certain set of functions for Chrome and Firefox? Might not be the case we want to be in. All apply to PIFE hack and we have precedence with Web page that wrap everything in parentheses. And arguably easier to do this than wrapping all the functions in parentheses. So this was just like just to inform you that we are planning to do this. + +MHA: There is not super much time for discussion. There is some. We are ahead of time, I guess. But there is like you can reach me via email and link to explainer and we have the proposal in WICG also. We have time for comments here also. I can’t see the queue. + +WH: I read the spec and I’m a bit worried that this just introduced yet another pass — per the spec you have to do a full pass to search for the magic comments before you start parsing anything. I assume it’s not what we want to do here, right? + +MHA: Yeah, definitely not. It’s just like specked that way because it’s easier. But practically, it would be done like as we parse. Like, definitely the intention is not that if there is like a comment somewhere at the bottom of the file that the top would be eager. We can’t do that in V8 with streaming parse. We haven't received the bytes and the whole file from the network that we need to parse. We need to spec it in a way that those already suggested to spec it in the way that it has to be at the top of the file. But, yes, definitely the intention is not that it applies to the part above the comment. + +WH: It currently sets state derived from the comment regardless of where the comment is in the file and then the state applies to parsing everything in the file. I’m glad that’s not the intention. + +MHA: That’s not the intention. I guess my intention was to write it in so it has to be the first thing and then I just got it wrong. + +MM: I think this is fast. You seem to strongly be implying and I want to verify explicitly that the intent is the hints cause zero observable difference? + +MHA: Yes, exactly. + +YSV: I’ll just bring up a conversation that Marja and I had yesterday that not everyone heard yet. When we looked at this, our thought was what would be better is to have facts stated rather than actions. So eager compilation might be more beneficial for Chrome for some functions than for a different browser whereas if we have facts and we can understand those, we can optimize differently depending on what we’re doing internally. + +MHA: My reply to that was that it’s compatible. To express things as facts are not directions is fundamentally compatible with my idea and work together to make it so. + +KM: I guess to the point of—I have to go back to the queue. Caught on first load instead of eager compilation. Another thing that I brought up which we already talked about is that—I want to relay to the rest of the committee right now allows slash slash at and slash slash pound. I think we should just pick one. Seems like slash slash pound was the agreed upon thing. I want to make sure that everyone is aware of. + +MHA: Yeah, exactly. The intention is only to allow //# and //@ was left over from source map where this was copied from. + +NRO: This is not strictly related to this, but I don’t know if you seen it the third proposal that is deferred execution of modules and I would expect that in this kind of lazy compilation and parsing would imply that? If I brought the fair module you do your pre-parse phase and then skip everything else? I guess even if it contains some comments about eagerly compelling some functions inside of it? I wonder if talking about how the proposal fits into this. + +MHA: I don’t know that much about the deferred module proposal, but I think it’s a fair assumption that it will be—it will imply lazy compilation because lazy compilation is the default anyway. + +YSV: I also agree. I had the same thought this might be interesting to see how it combines with import defer because import defer is also a hint to try to delay work if it’s not being used right away. You can undermined that attempt by in line immediately using the function in the main body of the script, but the idea is you won’t execute until that function is actually—you won’t execute the module until the function is actually called. It doesn’t work for classic scripts so it’s more limited than this. + +MHA: Right. We need to think how these combine and put the eager directive inside of the module, what will happen? Is it allowed? + +CDA: JKP noted that also deprecated //@ in hidden source maps. + +NRO: Just for context about that, it was just historically supported in the first original implementation of source maps. But then was quickly removed like quickly to hash because the at didn’t work I think. And nobody in practice uses the at anymore. You might decide to consider not repeat user of at for magic comments. + +CDA: Oliver is plus 1 starting conversation with magic comments. + +RPR: So I’m a big fan of this proposal and exploring space. I think that there may have been an assumption on one of the first slides that because this doesn’t impact the precise run time functionality, therefore it was not in scope of TC39. I think we have a very broad scope when it comes to JavaScript and the related ecosystem and in particular with TG-4 and the source maps already defining some of this category of comments. I think this could be relevant within TC39 if that was of interest. + +MHA: There is some precedence in like standardizing which are not mandatory to be followed in other standards. + +CDA: There is plus one from PFC. Plus 1 on TC39 from KM. Do you want to elaborate on that? + +KM: I don’t know. I feel like as a developer of a JS limitation and having it as part of, you know, whether it’s normative or not, part of the TC39 ecosystem is beneficial. That said, obviously since it’s not observable, there’s no requirement to do that. But I think it keeps everything consistent for people looking up and trying to understand the ecosystem and stuff. And I would hope also to collab on that since non observer I would hope that the bar for adding it as a nonnormative note is much lower and reduces the friction. + +DE: I agree with others that this is in scope for TC39 if people want to propose standardizing it here. I wanted to ask, when should JavaScript programmers or tool developers write this hint? It’s not just the “go fast” button. Should tools use some sort of heuristic, to figure out when to apply it? + +MHA: I didn’t quite hear the question. You were asking when they should write it or how should they write it? + +DE: Yeah. Like how should you decide whether to use this hint? What advice are you giving to JavaScript developers? + +MHA: Right. So with the PGO, I would assume that it’s something like you figure out relevant workload and maybe the relevant workload is loading the Web page or loading the Web page and doing common interaction with it and then figure out which part of the functions are called during the workload and basically just adding all of them is beneficial. Because those are the functions that are anyway going to be called. So they anyway need to be parsed and compiled. Better to do it eagerly. That is sort of my rough idea how to do this in the first step. + +YSV: This is basically just what Marja said and go with facts instead this makes it obvious to do tooling and it reports facts – + +MHA: Important point. We don’t expect developers to write things by hand. For the ease of the use and just to make the project easier to get started with, we have the thing to mark the core file with a per-file comment and go with the per function version and there the developers need to use tooling to go at the functions. Typically in the order of thousands of functions that should be annotated on the Web page. We don’t expect performance benefits if people hand annotate them, so they should be using PGO and tooling. + +### Speaker's Summary of Key Points + +MHA presented a planned Chromium feature where we enable web developers to attach directions about which functions should be eager-compiled by the JavaScript engine. The feature doesn't change JS semantics and is not observable. The presentation contained performance measurements to motivate the need for this feature. + +Discussion: WHT pointed out a spec bug; the intention is not to add a preparse pass for finding the comment if it occurs anywhere in the file. YSV suggested formulating the information as "facts" instead of directions. KM pointed out allowing //@ (which is unintentional, only //# should be allowed). Various people expressed that this would be in scope for TC39. Various people expressed interest in the feature. + +## `Atomics.pause` for stage 3 + +Presenter: Shu-yu Guo (SYG) + +- [proposal](https://github.com/tc39/proposal-atomics-microwait) +- (no slides, just spec) + +SYG: This is a very short in the theme of hints this is basically a method that checks if the input is on the fine or integer (?) but quickly go over the motivation and go over where the current spec draft is given the discussion we had last time about this hint parameter and how to control it for different kinds of spin weight loops. + +SYG: The motivation for `Atomics.pause` is that it’s basically emscripten and could benefit anything that does a busy loop but there is a very big user of busy loops on the web today and that is emscripten. Because we made the choice to not have blocking weights on the main thread, in order for emscripten to implement pthreads and mutexes… + +SYG: Emscripten is a tool chain that takes C++ and implements Wasm. The Wasm doesn’t do anything by itself like JS by itself and to have IO you have to hook it up to stuff. When you hook it up to the web platform emscripten has a standard library and basically implements pause stuff and pthread mutex and pthread condition variables and that kind of stuff and as part of that implementation, Mutex may block when waiting to be acquired and on the main thread we can’t put the main thread to sleep because that would be bad for UI check and instead what it does is it does a busy loop. Currently does this literally busy loop of a forever loop. + +SYG: This forever loop is bad for performance on some architectures. I don’t know the details of many architectures. I believe Apple folks last time told me this doesn’t really matter that much on modern Apple chips. But it matters on X city (?) chips and the manual recommends if you need to do a busy loop while waiting for a value to change like, “is my lock acquired?”, you should hint the CPU that you you are in such a busy loop so that certain parts of the underlying CPU can be better unlocked basically to check that—if that bit has changed. It goes down to how the CPU is designed and reads memory and that kind of thing. So many languages expose the ability to hint the CPU that you are in a busy loop via some kind of intrinsic. + +SYG: I’m proposing the same here for JavaScript in order to make the emscripten use case in particular more efficient and faster. Other folks have told me that they would also use this if it existed react for some reason, I don’t remember what, but I guess they also have a busy loop or something, game engines would benefit from this. + +SYG: So this has no observable behavior other than checking its input parameter that I will get into soon. But the purpose of this method is that it hints the CPU that “hey, I’m in a busy loop”. Or otherwise has no observable behavior and always returns on the find. It ought to pause for an exceedingly short amount of time, like, not in the microseconds. A few cycles basically, a few CPU cycles. It is not really a blocking weight in a realistic Sense, so it can be called on the main thread. And it takes input parameter instead of nothing because spin loops often have a backoff algorithm built in. What kind of backoff algorithm you want to use depends on the use case. You might have exponential backoff in positive or negative direction. Meaning you might want to wait progressively longer and longer and start with the longest way and wait progressively shorter and shorter. In a language like C or C++ the way to do it is just emit more calls to the pause intrinsic. That is not a good idea for JavaScript because the amount of time is so short and the performance characteristic of calling this method in the interpreter is so different interest when in inlined in the JITs and we always—basically because the execution time of calling `Atomics.pause` is so variable depending on which execution tier you’re in, it doesn’t work as well to spec this method to take no parameters. Like, that works for C++ because basically a function call takes the same time. There are no multiple JIT tiers in C++, and in JS VMs we do. Instead of emitting multiple calls to pause depending on how you want the back off behavior to go per iteration of the loop + +SYG: I’m proposing that we pass an optional parameter that says “what iteration of the spin loop I am in”? And the idea here is that by passing in an iteration number, then the engine can take advantage of whether you are in the optimizing tier JIT or interpreter to decide to use that number to emit the optimal number of pause instructions on the underlying CPU. So that’s why this parameter exists. To encode the various different kinds of back off that you might want to express, there’s this sentence here. So the idea is that when you pass in a loop number, that is not undefined, because it’s optional, if you pass something in, it determines how long it ought to wait. Of course, this can’t be unbounded. The idea is that the underlying implementation will cap it at something. Again, an exceedingly short amount of time. At most like tens or hundreds of cycles in the CPU. And the larger the number that you pass in, the longer time it waits. So if you want to do linear back off, you can start off with zero and then pass one, two, three, four, five, et cetera. And if you do exponentially you pass in exponentially growing loop numbers. You can also pass in negative numbers, meaning if you want to start with the longest time and you want decreasing amount of wait per loop, you start at zero and then you pass like negative one, negative two, et cetera to get smaller and smaller numbers. You can go either direction. + +SYG: There’s one catch here which is that if you start say with one and go down to zero and negative one, there is no good relation between one and negative one. So you can’t really like cross signs to implement your backoff algorithm. Of course you can pass it, it doesn’t really do anything. But underlying the implementation is probably not going to do the back off algorithm that you think it does. So that’s basically the only catch. And, yeah, this is all the spec text. The only observable step is the first one which is the argument validation that throws if it is either undefined or an Int number. Or otherwise returns—it does the pause thing and returns undefined. + +SYG: That’s it. I have tests in Test262. I think as JHD mentioned on the first day during the Test262 update. So with that, I would like to go to the queue before asking for Stage 3. + +RPR: Queue is open for business, if anyone would like to go in it. + +DE: Really glad that you’re bringing this proposal through TC39. You’ve iterated on the editorial aspects, and we all seem to agree on intent so this seems good to go. + +SYG: Thank you. + +RBN: I admit to having some interest in this due to its very useful for implementing spin waiting and lock free algorithms that come in handy with construct shared proposal and very interested to see it advance and I very much support Stage 3. + +SYG: Seeing empty queue, I would like to formally ask for Stage 3. + +RPR: We heard two messages of support. Question from KM. + +KM: Sorry, yes, how does negative work? So the idea is that negative N would be the shortest possible time? + +SYG: That negative like the smallest integer. + +KM: Suppose my engine – + +SYG: The shortest time. + +KM: N is the upper bound on the wait time. Do you pass negative N? + +SYG: Yes. + +KM: Are you expecting that to be like the longest backoff or the shortest backoff? + +SYG: The shortest backoff. It’s always larger the N, the longer the wait. But the expected use case is that—like, it’s harder to count—okay. How did we get here? How we got here was when we talk about the counting down case last time, I had naively felt like you first pass in the largest possible input and count down from there. That was deemed to be really silly. So this was reframed to let you start always counting at zero. If you want to get smaller, you go negative. If you want to go—if you want to wait shorter you go negative. If you want to wait longer, you go positive. You can always start at zero. + +KM: If you start at zero that would be the expectation that I just want—is equivalent to undefined, then that’s not the shortest possible wait, that’s some middle wait? + +SYG: Yes, that is a good point. It is like—so it’s a heuristic. So I think the very first, maybe the guidance is better you start counting at one or negative one. But my intention was not the very first wait is kind of hard for the engine to know which way you’re going next. The first wait is unhinted. And then afterwards, you will know the direction. If it can’t figure out a direction because you’re passing like bouncing numbers (?) or something, then that’s not going to really help either. + +KM: I have to think about that. Maybe there’s something we can reasonably come up with there. I’m not overwhelmingly concerned about it. But I was just a little confused what the expectation was. + +SYG: Let’s work on the—yeah, let’s talk offline. + +RPR: All right. So we heard two messages of support. Is there any opposition to Stage 3? I’m not hearing any opposition. So congratulations, SYG, you have Stage 3. + +### Conclusion + +Stage 3 + +## Discard (void) Bindings update + +Presenter: Ron Buckton (RBN) + +- [proposal](https://github.com/tc39/proposal-discard-binding) +- [slides](https://1drv.ms/p/s!AjgWTO11Fk-Tkr1WMJUD3IiTYZINVQ?e=fyNJaA) + +RBN: So I intend to keep this brief. This is intended to be a short update based on some recent discussions within the champions group in relation to discard bindings and some investigation I’ve been doing into some of the options that we discussed prior and part of the Stage 2 advancement. Just give a brief discard is idea is non-named placeholder for a variable and allows you to elide unnecessary variable names in certain contents with using and await using declarations and function and method parameters and object and array destructuring patterns and extractor and pattern matching. It proposes use of void keyword for place of binding identifier and prior art is C and C++ and Python and Rust and underscore. They all serve in most cases similar use cases of being able to declare something that you don’t actually need the name for it. This proposal is currently at Stage 2. This is mostly motivated by the need for—the ability to have the side effects of a declaration without the actual binding tooling often warn or error on unused variables so if you’re using temporary variable names you often have to use comments to disable lint rules or underscore-prefixed names. There’s several existing single purpose solutions to this. + +RBN: So array destructuring elision and bindingless-catch and this is single purpose for the syntaxes and there is no single use and problematic for declarations. In some cases we might try using empty object patterns. Those are insufficient because they throw if you pass null or defined on the right hand side which really wouldn’t work for using and in addition using itself doesn’t allow binding patterns and shrimp elission and not sufficient because using legal is syntax and you can’t have elission in object destructuring, or let/const. `let =` is legal in nonstrict mode. + +RBN: The proposal again seeks to use the void keyword for a discard binding. So examples of this show in using void equals and binding patterns it would be void on the assignment or declaration side. In assignment patterns again, where you might in array destructuring potentially use elission and use this in the array destructing and parameters would allow to have the parameter name and unnamed pattern can be in extractors and pattern matching. We did have a discussion around the time we were proposing it and advancing to Stage 2 which was we could consider underscore or void and majority of other languages that have discards for pattern matching it use underscore. It has pros and cons. In many of the languages of underscore sometimes have to resort to work arounds and potentially hack-y and unreliable ways of making in work or had to have a new version of the language that deprecates under score that allowed to use as identifier that could be problematic. One way to do it if we wanted to have similar workarounds to use underscore would be to weaken the current restrictions on duplicated identifiers which is the approach that we discussed the last time that this was presented. So basically rather than just caring about underscore, instead it would be any identifier could potentially be redeclared but then in strict mode cases but then be illegal to actually reference. It would allow us to kind of mirror what we can do in non-strict mode with bindings or far declarations and repeat a binding. + +RBN: But this isn’t without its own issues. Unfortunately duplicate identifiers or underscore any duplicate identifier would have issues that void itself was not. One you can’t use underscore in an assignment pattern safely and might be overriding something or declaring global or introducing error if this is strict code. This doesn’t exist. You would have to declare it. It can be leaky in the global script where you might have a declaration like a var declaration to use underscore as discard and then leaks into the global. So the main thing that we’re seeking to do is to definitively state this proposal despite the advantages. Similarly in the language and use underscore this proposal is seeking to advance or not to advance but seeking to continue to use void as the primary mechanism for discards and covers all cases declaration and assignment patterns and pattern matching cleanly. We could potentially also still pursue duplicate loosening the restriction on duplicate identifiers as the convenience mechanism but not the primary mechanism and if so, the open question is whether or not we should push this to the follow up proposal or continue it as a side feature of this proposal? More likely to say if we still want to have this, we should consider it as either a follow one proposal or even just a needed consensus PR rather than tacking it on to void discards. So that’s kind of where things stand right now with the proposal. I will go to the queue and we can discuss. + +NRO: I was very strongly pushing for underscore. I still think it looks nicer from scratch we should use underscore. However, I’ve been convinced that the different—with constructing declarations and not in the expressions is variable, it risks for example trigger in the scope and didn’t notice it’s happening. And it’s just too easy to make errors if we make this thing behave differently in the declarations and expressions. So I don’t think underscore would look better. I just don’t think there is a reliable solution. I’m happy with the proposal and happy with void. + +JRL: To be a counter to that, I continue to think that underscore is best choice and don’t like void. I don’t understand why void is better than the current status quo than unused variable because the unused variable names are just `_` or `_a` or `_b` and shorter and less wordy—less tokens in the program. The whole point for the discard binding is to be something that you can scan over and doesn’t distract from the actual intention of the program. The void keyword besides appearing like a full word that looks like an identifier in my program, and now takes up more space than the thing that is the status quo. I don’t think void is the correct choice for this. I understand there are cases where underscore behaves badly. But no one does an assignment destructor. Ok, some people in here do it. Not many people. They don’t understand to wrap it in parentheses to make it assignment destruction rather than a block scope with label on it. If we have global variables, no one is doing it anymore. I continue to think we should continue to pursue underscore as the correct solution for this? + +RBN: Again, even if it’s not assignment pattern and simple assignments if you’re saying underscore or I have seen plenty of assignment pattern usage of array destructing and in those cases you could elission and you run into the corner case of the trailing element if you fail with the extra comma because trailing commas can be problematic. My biggest concern with underscore in the assignment pattern case becomes you’re going to see people write examples in block posts and things that might contain some of the syntax and just use underscore and if they say that is just the discard it is easy to use the context if you’re doing this in the assignment pattern case you need to have a free var underscore somewhere in the code that this gets written to and if you don’t do that at whatever scope that you’re in, whatever function block you’re in, that variable could leak and if you have other references to it in other places it could potentially be held on to because you created a closure scope and in the branch with the function closure have the thing and you’re holding to memory. It has the corner cases and side effects that make it fragile and one of the main reasons why I think it’s not really a viable solution. + +MF: Happy with the proposal and the choice of void. I agree with everything RBN just said. + +WH: I don’t want two solutions for this. So if we do `void`, we should not do `_` or permit multiple definitions of `_` in a scope. + +CM: I’m happy with void. I think I understand the esthetic appeal of underscore. But I’m wondering if the gymnastics to make that work wouldn’t interfere with some existing practices of just using underscore as the name of a variable. Which some people do! + +RBN: One of the reasons why—so this proposal if we had wanted to consider this alternative solution, it was not just underscore. It could literally be any identifier would be allowed to be duplicated and issue with underscore and using underscore and `import * as underscore` from the underscore package or from lodash or something like that you could use double underscore or underscore A and tempt and that would be fine. It would be every in the documentation would be the discard you would have the recommendation that underscore is to be using. + +CM: That defeats the purpose of having a standard way to say this is the discard. + +RBN: This is like many other things to shoehorn in underscore as discard with weird corner cases that don’t quite work the way you expect or don’t quite look the way you expect. + +CZW: I was in favor of the underscore but given with the issues presented with the underscore that void can be a pragmatic choice to use as a discard binding. + +SFC: Yeah, so I have two comments here. One is not the motivations, you know, this is the common theme when I ask—make a comment about proposals including decimal but also here with the discard bindings. I look at the slide that says motivation and the first bullet point says need for declaration side effects without the variable binding. So I’m trying to parse what the bullet point means. I want to skip to the second one here that is tooling often warns/errors on unused variables. This seems like something we can discuss here. So the motivation is that while the status quo is unused variables. And that causes elission to complain about unused variables. There could be other ways that we could like hint ESLINT this is discard binding. I will use rust as an example underscore with the name variable`_a`, then the default rust lint will not complain about the variable and assume you started with under score and that’s discard biding. That style is explicit and sometimes I like using the style in code because if I’m doing a discard binding sometimes I like to actually name this is the variable being discarded. I name the variable and then also put a prefix on it and underscore on the prefix variable name to show this is the variable that is being discarded here, right? So like there’s other ways to solve the second bullet point than just this proposal here. And then as far as the other three bullet points, are those motivations? Those are more like these other things don’t work very well. So that’s not really a motivation for the problem that we’re trying to solve. That’s more motivation why this solution is something to consider. What is the actual problem we’re trying to solve? To close, if there’s actually like a—so what I can imagine here is that if it is expensive for engines to like – if there’s a variable that is unused that, you know, during like a JIT time or whatever, engines can be like we’re not going to allocate space for this variable and make the program more efficient, like, that would be like motivation I think why we want a void binding. If such a case exists, I would love to hear about it and see it. And then I would say this is very well motivated proposal because it actually has concrete performance improvements on your program. If this is just a stylistic thing, it is not clear this proposal is not much better than the status quo. I agree with what Justin was saying earlier where it’s like I’m already doing discard bindings in the code. If the only problem I’m trying to solve is that ESLint is too noisy let’s solve that instead. + +RBN: Let me back up. Historically when the resource management proposal was at Stage 1, when initially introduced and up until it reached Stage 3. It had a mechanism for allocating and tracking a resource without declaring a binding. And that ended up being removed and pulled off to the separate proposal because it specifically had cross cutting concerns with other places that need it. Pattern matching needs a discard mechanism. And it seems somewhat less motivated until pattern matching comes in a way because having a discard mechanism is profoundly necessary in the pattern matching. You need to be able to match on a property exists but I don’t care about its type. To do that without any discard mechanism is nearly – shouldn’t say impossible but requires you to do like a disjunction of two things that are the opposite of each other or requires you to again declare a variable that you don’t actually use that is more code that you have to write and requires the mental effort of thinking about how do I—what do I name this thing? And performance doesn’t need to be the only reason we consider adding things to the language. Developer experience is still very important. When we looked at bindingless-catch that was practically a shoe-in. People didn’t like naming catch variable e over and over again because they didn’t use in many cases and delay the catch. This is more than delay the catch. You have to write out characters, yes. It means you don’t have to name these things. I don’t want to have to litter my code with `const _a`, `_b`, and `_c` for various things just to allied the things I don’t need with multiple times with multiple declarations with the same scope that have these types of elisions. It is not really beneficial in those cases. For cases like array destructuring if you need to exhaust something and move past the end or skip an element that is easy to be tripped up by the trailing comma in array destructing that if you expect a live last element and A comma B comma the last comma is not progressing and ignored and easy to forget with the explicit marker for the discard is much more effective for elission. All the way back to the top there and say the need for declaration side effects without the variable binding using is the primary use case and using needed a way to use the resource that already exists in scope in many cases, so I’ve passed in a resource and now I need to take ownership of the thing and control it for the scope of the body. But I don’t need to give it a new name. I already have a name for it. I just need to enlist it in the current scope. To be able to do that is important and useful for things like the Mutex that is and the unlock token part of the shared struct proposal and often have a block where I need to lock the mutex within the certain scope. I don’t need to give the thing a name. It doesn’t need a name. I will never reference it. Underscore can be useful for this but once again you can run into the case if you also need to lock multiple things or create one lock to create a log in resource none of these things need names and now I’m having to give them names and having to differentiate between them and create the bindings and the effort the user doesn’t have to go to if they don’t need to. This is primarily meant to be or up until you get to the pattern-matching case, this is primarily meant to be the convenience in the DX improvement over having to constantly name things and then once you get to pattern matching it has much more value. + +JRL: To respond to the part of pattern matching needs discard binding. Pattern matching doesn’t need discard binding but just to match the structure X and Y if you’re already doing everything. If you don’t need X, it’s an unused variable. + +RBN: That is incorrect. Pattern matching regularly needs to be—this example at the end and match object with X void and Y void is currently the only way to check the structure. If you wanted to check that X exists but don’t care about the valueOf X because I’m checking the shape of the thing is incorrect and plan to pass it to somewhere else I need the way to test that the property exists without having to write a complex match pattern and the actual valueOf the thing doesn’t matter. Being able to test structure is paramount in the cases and you don’t want to dig further in you need a way to test it. + +JRL: You are digging in further in to test the pattern structure. Rust has the way to do it to test the object and the thing inside of the object. + +RBN: You can do that as well and rust can use underscore for this. + +JRL: Correct. + +RBN: This exists in Rust. It just uses underscore. We really can’t. + +JRL: When you use underscore in Rust, it doesn’t mean you captured the object. The case that you outlined to capture the outer object and assert the structure of the inner object is that correct? + +RBN: No. I’m saying in this branch I’m testing that the object has the properties X and Y and the only thing it is testing. It is not X colon number and saying must have number and must have X and be a number. It is testing X and Y and that’s all it is testing. + +JRL: Can’t do `when { X: X, Y: Y }`. + +RBN: What does that mean? If you said in the current proposal `x: x` means `x` must match something in the outer score and a reference in the identifier. If you were not going to use the thing, you have to say X colon let underscore or let X to declare a variable declaration that you not use. + +JRL: Pattern matching proposal doesn’t allow from the object structure. + +RBN: It does using let and the variable name and you’re once again back to I’m introducing a variable declaration I don’t need. + +When we change when to let this will—change when to let and remove the voids this will work correctly to match the case. + +RBN: You can’t change—the when key doesn’t change in the current proposal we’re making changes to it. You have to—so when you are matching something, the right-hand side of the match will set X colon something, that something is a reference to something in scope. It’s not a binding. You actually have to declare the binding and X colon let X to say matching X to some pattern and that pattern is a let binding that introducing a new variable and introducing a new variable in scope. This is a variable you don’t need. You’re just adding more variable declarations to your function body. + +JRL: It is radically different than what Rust has done and causing some confusion for me. + +RBN: Rust has advantage of having a type system and knowing whether something is a new declaration or a reference to something else. + +JRL: I think we need to talk offline. I continue to think that we don’t need to introduce void to handle this case for pattern matching and there’s another way to do it and rust has shown there is a way to do this. + +RBN: I think we need to talk offline and rust has issues with—there’s discrepancies between what rust can do and what JavaScript can do. But I would appreciate if you could potentially join one of the upcoming pattern matching calls and discuss that further on the pattern matching proposal side. + +RPR: We have five minutes left on this. A few clarifying questions. + +NRO: Just hard to follow the discussion about code with braces and colons and everything. If you could write in the queue the example how the pattern matching case would work without void. + +RBN: I think this is something we can potentially follow up with offline. I’m not seeking advancement for anything right now. I’m mostly setting where the direction is kind of going based on feedback and investigations that we have been doing that—to continue with that. If we can continue that offline, I think that would be fine. + +RPR: Good move. + +JWK: Hello. I have posted some examples for what works in pattern matching if you want to do things like test a property but don’t care the actual value. You can see the matrix. + +```javascript +when { x: let _ }: // unused _ when { let x }: // unused x when has 'x': // test 'x' in subject (a possibly extension) +``` + +SFC: Yeah, so if we do move forward with this proposal, I do think that like a more proposal with fewer edge cases IE the void keyword seems better. It’s just I’m not convinced by the motivation. If this is really necessary for pattern matching, then like you know that should be clear like this is part of pattern matching. When it comes to, you know, proposals and we definitely take this position a lot in TG2 which is that like a proposal needs to be motivated by itself like not because of some future proposal that’s coming along. This sometimes made it hard to make progress in TG2 but I think it’s ultimately the right call. Because if this is only useful for pattern matching, it should be just part of the pattern matching proposal, question mark? + +RBN: It is not. This is part of the research management proposal. I pulled it out for two reasons: One to simplify the resource management proposal somewhat and it had cross cutting concerns with mat tern matching and other things to discard. Not narrowly focused to one case. It is a very small thin feature that is intent to broadly cross cut several different things. + +SFC: So since we’re low on time, I’m sorry, but I can see the examples in pattern matching for why we actually need void here. It’s very hard to do the thing without void. But in terms of using void and things like that, just say `using _handler =` or something like that. You know, like it seems fine using void equals. Like, makes me have to look up what does using void equals mean. When I use reading underscore handler equals I know this is a handler going to be discarded at the end of the scope because I have seen that before. Using void equals is like what the heck is going on here? So I don’t necessarily think it’s necessarily an improvement. But again my point here is if it’s narrow, it seems harmless to have this. It’s just I’m not convinced of the motivation. If the rest of the delegates think so, that’s fine. I mean, I’m just – + +RBN: I will say that having started writing code that makes fairly heavy use of the using declaration and I had more than one occasion with three or four things that I don’t need to name. It’s very inconvenient to write `using _a`, and then `using _1`, and `using _2`, or `_a` +and `_b`, and you have to look up in scope and what to use and use underscore B and I have a type checker I already used that. And I have to name it something else. This is developer experience improvement that is supposed to save you from having to do all of this work. + +RPR: One minute remaining, Ron, could you summarize the points that we have heard? + +### Speaker's Summary of Key Points + +RBN: So the primary purpose of this was to provide an update and kind of set expectations for where things stand with the proposal. The main change here is that we are seeking to narrow in on primarily just using void. The underscore or duly cat identifier solution has a number of very rough edge cases that make it difficult to be able to use consistently. Other than the suggestions we have been having about, like, motivation versus just writing things with_this is a developer experience improvement, which it has additional cases where pattern matching is deployed + +## AsyncContext updates + +Presenter: Andreu Botella (ABO) + +- [proposal](https://github.com/tc39/proposal-async-context/) +- [slides](https://docs.google.com/presentation/d/1DZXqdY-J4DBNCBfEmTNUbkhm2iSgRZj1ez3MM6dReKc/edit?usp=sharing) + +ABO: Okay. So I will be giving some updates on the web integration of AsyncContext. That is specifically what these updates are about. The web integration is a requirement before Stage 2.7 because some of the delegates would block without it. So I will be giving an overview of the current status of things. + +ABO: So the whole point of the proposal is to have some state that gets preserved across awaits, like a thread-local storage but for async-await. And the thing is, with things like setTimeout here, this is a very similar case where you would expect setTimeout to preserve the context from when setTimeout is called to the callback. So here, you expect `asyncVar.get()` to return 42. + +ABO: But this is because in both browsers and runtimes, promises are not the only source of asynchronicity. And things like setTimeout need to be accounted for. + +ABO: So the basic idea for the web integration is that most of the web APIs that take a callback will then run it later, at a point where no context is active—where the JavaScript stack is empty. So there is only one relevant context, the one that was active when the scheduling API gets called. If you are familiar with the proposal, you know that the contexts are strictly scoped. And so if there is no active JavaScript stack, if there’s nothing on the stack, the context is empty. So we’re not dismissing that, that empty context. + +ABO: And for these APIs that take a callback, we store that context when the API that schedules it is called. And we restore it at the time of running the callback. And this is equivalent to wrapping the callback in `AsyncContext.Snapshot.wrap`. This can be implemented in WebIDL, which is part of how browsers implement web APIs and how they are defined in the specs. And this would work automatically, with no work needed at all for future spec additions and engine implementations. This is the case for `setTimeout`, `queueMicrotask`, `requestIdleCallback`, `schedule.postTask`, all those APIs. + +ABO: Not all APIs like that, and there’s some more complexity. So observers are a common pattern in the web platform, things like `IntersectionObserver`, `MutationObserver`... These classes take a callback on the constructor, which will be called to notify of some observed change. You can register things to observe and so on. + +ABO: The thing is, the callback might be invoked once for multiple observations—like, the observations can be queued and then the callback gets called once per event loop turn or whatever. This is different from FinalizationRegistry, where the callback is called once per observation. And so the context in which the callback is called cannot be the context of any of the observers. + +ABO: This actually affected the behavior we proposed for FinalizationRegistry with AsyncContext. So that is an interaction that affects the 262 spec. + +ABO: So here, the only reasonable option for the context in which to call the callback is to do it in the context in which the observer is constructed. But a use case for PerformanceObserver is that you might want the observation context. And those might be exposed as an `AsyncContext.Snapshot` object that is exposed in PerformanceEntry—in each of the observation entries. + +ABO: And events are a more complicated case. There are one or two possible contexts for each event dispatch, or rather for every time an event listener gets called. There is the registration context, which is the context that is active when `addEventListener` is called. And there is always one because you need to call `addEventListener` or set `onClick` or something like that. So there’s always a registration context. And then depending on the event, there might or might not be a dispatch context, which is the context active when the JavaScript source that caused this event is called. This can be synchronous, like the `click()` method on HTML elements fires a click event synchronously. It can be asynchronous, like if an event gets fired after a request has completed or something. And it can be missing for things like user clicks or notifications from the browser/runtime, when there is no JavaScript source for the event. + +ABO: For events that have a JavaScript source that is asynchronous, in the web platform currently there is no way to track that context automatically from the source to the final event. And there is no way to distinguish that from events that don’t have a JavaScript source. So that is not something that is built into the platform currently. + +ABO: If we start tracking that context manually, it would mean changing a bunch of web specs and a bunch of browser implementations to track them manually. There is also another possibility, which is defining an automatic way to propagate the context through the spec-internal and browser-internal operations. Which is complicated. And if the context is tracked manually, adding new events in the future might not be trivial for web spec authors + +ABO: So with this as the background, our initial proposal for events was that the context active when an event listener is called is the registration context. So the context that is active when `addEventListener` or `onClick` is called. + +ABO: And then the dispatch context can also be important for some use cases. And it can be exposed as a property on the event object with an `AsyncContext.Snapshot` value, same as for observers. + +ABO: Now, since it might need to be tracked manually, what we initially said is that this property will not be present in all event subclasses. It would only be present in some. It could be a nullable property, in others to allow an incremental rollout. And we would only initially expose this property for just a few events, like the `error` and `unhandledrejection` events. + +ABO: Some web platform features need a similar kind of context propagation as AsyncContext. Like, incumbent realm tracking, soft navigation, `schedule.yield`. Some of these are long-standing web platform features, some are things that are being added recently or as we speak. And these could be implemented as browser-internal `AsyncContext.Variable`s. + +ABO: It is not clear that the semantics for all of the features match the web integration that we’re proposing. For some of them, it’s possible that they match. For some there are differences, but it’s possible they could change to have this behavior. For others, it could be integrated with this behavior by hooking at a lower layer, still within the browser engine, not the JavaScript engine. But these are still things that need to be figured out. + +ABO: And so what about the server-side? We have been talking about web integration, but this is because it was a requirement for Stage 2.7. But I think that considering Node.js and other server-side runtimes is important, and I think that the same general approach that we use for the web should also work for them. And it would be bad if server-side runtimes had completely different semantics. Especially with WinterCG and runtimes that support web platform APIs, having these different semantics depending on where the API is defined is not good. So if this is unworkable for the server-side, we want to know, and we should try to find a different way. + +ABO: So current status. We have a PR for the web integration. There are some minor issues that I should have found time to fix, but I didn’t. But this is the main blocker for Stage 2.7 right now, as I said. The majority of the semantics are clear and uncontroversial. Except for events because they don’t seem to work for everyone. And we heard from some of the framework folks at Google, and it seems like different use cases seem to have different and conflicting requirements. So we need to figure out how to fix that. + +ABO: So we will work through various alternatives, various ways that we can resolve this conflict. If this affects you, we want to hear from you and consider all relevant use cases. And we are also looking for feedback from browser engine implementers and web platform spec authors. This is not just JavaScript engine implementers, because all of this will not affect JavaScript engines. + +ABO: So get involved, I guess. We have meetings every two weeks. Previous slide decks about AsyncContext didn’t mention you should ask us to add to the invite because I think it’s on the TC39—private. I don’t know. Just ask us. And we have a matrix channel. + +SYG: So I guess mostly clarifying questions, but I don’t have all this paged in. Bear with me. About the events, specifically the two contexts, the registration context and the dispatch context, like what does it mean to have two? Like, are you just saying that you need to make a decision on which context the event listener runs? Or are you saying, like, there’s a way—proposing a way to thread both contexts so the event listener itself can choose which context? + +ABO: In the PR as it is—like, the state of the world currently, it’s like this slide says that there are two possible contexts that could be used. Like there’s always a registration context. Some events would have a dispatch context, and our initial proposal was that the listener would always run in the registration context. And then for specific types of events, you would pass the dispatch context as a property of the event. + +SYG: What is the value of that property? It’s like the value of the async variable? + +ABO: No. It’s `AsyncContext.Snapshot` value. + +SYG: I see. That’s what you mean with an AsyncContext. So the property would have an `AsyncContext.Snapshot` if it originated from, like, JS user code, otherwise it’s null. + +ABO: Or if the event has an async source—like it originated from JS, but not synchronously. + +SYG: I see. + +ABO: In that case, we’re not initially going to propagate that. We might set it to null in most places, and possibly, incrementally, modify that for individual events over time. + +SYG: So I am not really an expert on the DOM side of things. Like, I assume there are a lot of events. Is it okay to add a property on all events like this? + +ABO: In the proposal that we had, the idea would be like just incrementally – + +SYG: Not event subclasses. But event instances. During the time of an application of a page running, there’s a lot of events, I imagine. And we’re adding a property in every one of those with a possible snapshot? Like, I guess, you know, the snapshot will only be present if the page itself is using AsyncContext. I don’t know. Generic, seems like somehow high cost. I would like to understand it better, but I am not an expert, so we should get more engine people. Like, DOM-side engine people. + +ABO: If the page itself does not use AsyncContext at all, then the way I imagine is that you would always store the registration context. But those would point to the same internal mapping, which would be an internal object in the engines. And whether you create a JavaScript object from that—there was a possibility that you would have that property be a getter actually and only create the JavaScript object if that is needed. + +NRO: So for context I help with the proposal, this proposal is now mostly blocked on figuring out exactly how events should work. When we proposed the idea of running the callback in the context where the event listener was registered, we received feedback that in some cases it’s desirable to run the callback in the context where the event was dispatched, if there is any. Mostly from framework authors, but also because this is how, for example, task attribution works in Chrome or how the `console.createTask` API works in Chrome. However, we believe one of the reasons we went with the registration time context was because it is easier to implement. Because you just need to capture that without propagating the snapshot through async steps in engines. + +NRO: So I know we cannot get an answer for this now, but it would be great if the browser people here could talk with their DOM colleagues to understand the feasibility of actually propagating the dispatch time snapshot for async events. For example, if I have an `XMLHttpRequest`, how feasible is it to propagate the context from when `.send()` is called to when the event listener is run? Because once we figure out what to do exactly with the events and once we have an answer people are happy with, then we can advance the proposal to 2.7. + +DE: I want to emphasize what NRO was saying, that this proposal is done, except for the web integration. The thing is, there are other web APIs like `scheduler.yield` that are being developed and in some places even shipping, which are solving the same problem of how to do context propagation. Some of these don’t have completely formal specifications for how they propagate the context. And as part of this project, we want to define and formalize that. In my opinion, we should propagate context in the same way across AsyncContext as well as the context tracked by these other features. Otherwise, there’s the risk that events would have to propagate multiple different context variables, which could be expensive at runtime. I would be hesitant to ship things incrementally unless we’re comfortable with AsyncContext. This is a debate I hope occurs on the web side. And I would appreciate the help of people who work in browsers to think through this, this sort of existence of multiple things that need the same context propagated and what the semantics of that should be. + +ABO: I want to mention that, well, I think I mentioned some of this before. For soft navigation tracking, this is something where it seems feasible that the semantics that are currently being used in the Chrome implementation could change to be aligned with the web integration semantics we are proposing. It’s not fully clear to the folks working on it whether that is possible, but they are willing to see, to try. `scheduler.yield` is a slightly more complicated case, but it would not require a completely different—like, having two separate mappings that need to be propagated in parallel. It just—it would have to be hooked in the browser engine’s implementation of the web integration. + +DE: So I think it’s important that people consider these interactions, whether those hooks apply in all browsers before they get too far in shipping the web features. There might be a misunderstanding that the web is moving faster and is more dynamic, and that TC39 is slow. In this case, the slowness is not on TC39 holding back things, but rather building consensus with web stakeholders on the web semantics. That’s what I want people to come home with with this presentation: that browsers should work to solve this same (or closely related) problem. + +ABO: Yeah. I am working on a document, like for the exact interactions between these web platform features and the exact semantics they need. I didn’t have it ready for today, but I will try to finish it and make it up soon so that anyone can take a look at it and check. + +SYG: I have another question—just not about any of the particulars, but I want people familiar with AsyncContext to see if this characterization is a fair characterization. My understanding of the difficulty in pinning down the semantics with all these different contexts, that it’s come up with AsyncContext but not for soft navigation tracking and `scheduler.yield` is because soft navigation tracking and `scheduler.yield` are concrete, narrow APIs for like one particular concrete use case like scheduling priority and tracking soft navigation. And because they are narrow enough, they can just choose which context is the right one to propagate. Whereas, for AsyncContext, because it is programmatically controlled by user code, you have to consider all possible things, all possible contexts that might be reasonable to propagate, and it’s difficult to nail that down because you don’t know the intent of user code. Is that a fair characterization? + +ABO: I think so, yeah. And I know that soft navigation has changed things and shipped things incrementally because the results are not exposed to the user. Except as a timing measurement that would be hard to influence as a developer. + +SYG: Okay. Outside of implementation reasons, like complexity reasons in Blink and WebKit itself, why is it important in your opinion, to explain the propagation of soft navigation tracking and `scheduler.yield` via AsyncContext? + +ABO: The main reason is that we don’t have different features at various different levels of the platform, including JavaScript, that basically do the same thing except with slightly different semantics. And for the implementation in Blink and V8 in particular, those features—well, not incumbent context, but soft navigation tracking and `scheduler.yield`, are currently using `ContinuationPreservedEmbedderData`, which we were hoping to replace with AsyncContext. It might end up that is—that at that replacing is fully doable, but I think it would—like, it would offer a lot more, I guess, + +SYG: I am confused by the layering. CPED is not a spec. Is it? + +ABO: No. + +DE: The fundamental issue is that implementations don’t want to carry around multiple variables to understand which context is propagated. Right now you can root everything in one thing, one extra pointer that the JavaScript engine threads through. It would have additional cost to thread two different pointers through. This will affect all implementations, not just V8. + +DE: A difference between `scheduler.yield` and soft navigation on one hand, and AsyncContext on the other hand, is that you can be sloppy with propagation of yield/soft navigation. And you will get approximate results that are okay enough, as long as you don’t care about consistent, reliable results across browsers. The main difference about AsyncContext is not quite that it is user programmable. But we have to nail down the precise semantics because there is so much more readily observable. If something doesn’t propagate the priority the same way by accident, you kind of recover okay. + +SYG: I still see it as a thing of a user programmability, because you don’t know what—like if you propagate the wrong thing, you don’t know if it is acceptable enough. For `scheduler.yield`, you know because it’s not user-programmable that it is acceptable. For `AsyncContext.Variable`s, you can’t know, so you have to assume the conservative thing– + +DE: If we want interop across browsers, then you also want the same sort of very strong well-defined semantics for `scheduler.yield`. Same thing as AsyncContext. If you don’t care if browsers have different propagation semantics for yield or different measurements for soft navigation, then you wouldn’t care about this as much. That’s the main difference. So if you just have a single browser, you know, experimenting and going ahead then it’s okay to be somewhat vague and unstable, but probably when multiple browsers want to coordinate and adopt semantics with reliable interoperability, that solution should work for AsyncContext as well. + +CDA: We are past time. I would like to at least get to Mark’s comment before we finish. + +MM: I really like your terminology of registration context and dispatch context. There are two conflicting interpretations of least surprise that pull you in opposite directions, and I very much appreciate the choice you made. Let’s make sure I understand the consistency of this. I want to use `Promise.then` and then a `Promise.all` with the `Promise.then` on the result of `Promise.all` as examples in the language, obviously outside the web. But a thing that is caused often has multiple causes. + +MM: And your observation is, there is always exactly one registration context clause. And there are some arbitrary number of dispatch clauses. So, for example, if you do a `Promise.all` on a number of promises, and then you do `Promise.then` and a `Promise.then`, the two clauses are the registration, and the settling of the promise you’re observing. So `Promise.then` is a degenerate case of observing things, and triggering the `Promise.then` is a degenerate case of the logic of triggering an observer, notifying an observer. + +MM: By choosing the registration context, you’re not saying it’s more important and the others are less important. You are choosing a rule such that you always have a predictable model that the programmer can plan around, or programmer can coordinate on so that even when it’s not the one the programmer actually wants, they still know what it is and they can expect that it won’t be the one they want. And then you’ve provided these separate snapshots where the programmer knows they want something else and they can get it. + +MM: The issue—this goes to SYG’s question about the specific semantics of what is going on, it will differ from case to case about what the programmer wants. The real issue is not giving the programmer what they want in all cases implicitly. The issue is, if you provide one thing implicitly and other things explicitly, less conveniently, they have the one that’s implicitly provided be predictable. So the programmer can know when they need to turn and get the other one explicitly because that’s the one they really want. + +MM: Does that all make sense? I am asking SYG as well. Okay. + +ABO: Yeah. That was part of the motivation for this initial proposal. But what we saw is that like some of the feedback that we got is that sometimes the implicit context is important because if you are a third-party library, and the code in the listener is first-party code, the first-party code wouldn't know to propagate those values, wouldn't know to pass things around. And with this model, you would only be able to make available to the third party code the registration context. It seems like some use cases need that. We are investigating other solutions. And so we’re looking at this problem and seeing if there is a way to propagate both things implicitly in some way. + +MM: You can’t propagate both things implicitly. In every case, there’s one thing to propagate because of the nature of the dynamic scoping you’re providing. + +MM: Well, okay. Justin suggested we talk in the hallway. I agreed. All right. + +### Speaker's Summary of Key Points + +ABO: We have a proposal for web integration. There is some feedback that the proposal doesn’t cover all use cases and we will continue to look into it. And this is the only point that is missing before Stage 2.7. + +## `Array.zip` for stage 1, or 2, or 2.7 + +Presenter: Jordan Harband (JHD) + +- [proposal](https://github.com/ljharb/proposal-array-zip) +- no slides presented + +JHD: All right. So we talked about `Iterator.zip` and `zipKeyed` in previous plenaries, and agreed to have it proceed only producing an iterator. So I should make a separate proposal for arrays if I want that,and that is this proposal. + +JHD: It’s basically exactly the same as the iterator functions. But instead of running an iterator, it returns a collected array. This is the spec for joint iteration. These are the only steps that are different from the joint iteration. The same steps in both zip and zipKeyed. It just takes—I could have replaced the first 16 steps of this with an direct invocation of the originality `Iterator.zip` for example. And it just makes an array out of it. + +JHD: This is spec text and could be different. It’s only 8 lines different from the existing Stage 2.7 spec text of joint iteration. And so I put Stage 1, 2 or 2.7 because I am comfortable with any of the advancements. But I wanted to see what the room’s thought were on solving the problem, which is Stage 1; on having the solution exactly match joint iteration which would be for Stage 2 because it has the spec text already; and I'm assuming it does need to match joint iteration, then I think it would be eligible for 2.7 because there’s no more design decisions. + +MF: Yeah. So we talked about `Array.zip` when talking about joint iteration. We didn’t include it in the joint iteration proposal as we didn’t want its questionable motivation to comprise the joint iteration proposal. So we dropped it saying that, if we find good motivations, nothing wrong with having this, but we want to not add something for no reason. + +MF: So the reason why I was suspicious of motivation is that we think all of the use cases of joint iteration are followed immediately by some other operation other than like toArray. And if we find that it is common for users of joint iteration to be immediately collecting into an array, this would be motivated. As you say, you would avoid the iteration protocol there. But so I think that—I would be fine with Stage 1 for this proposal in that we are investigating if there’s a need for something like this. And maybe it doesn’t look like `Array.zip` in the end. But until we get some usage data from joint iteration, meaning that it’s reached Stage 3, out there in the browsers and see the people are actually doing that, like doing Iterator.zip and toArray on it or something, it’s not convincing enough motivation for specifically this moving forward past Stage 1. + +JHD: Okay. But I mean, so that makes sense. I am fine with it going to Stage 1 for today. I have a hard time conceiving any possible design changes it could make. Although, like what I am hearing you say is that either it deserves to be added to the language or not. If it deserves to, usage data from `Iterator.zip` should show that + +MF: It would probably look like this. If people are doing Iterator.zip and toArray, that would be like a drop in requirement for that. + +JHD: Yeah + +MF: But it’s possible. Maybe we identify other similar patterns. Do something like that instead. That’s like—that’s a little bit too – + +JHD: Right + +MF: It is definitely still worth keeping in mind that we have a possibility for `Array.zip` if it becomes common and people can improve their code using it. But for now, I don’t think it warrants more than Stage 1. + +YSV: This is +1 to MF's comment. This is something good to see as a library first, before we talk about getting it into the language. So I am fine with Stage 1. But I think we need more data to back up the motivation. + +JHD: Well, then I am asking for Stage 1. + +SYG: I have a question. So one of the statements on the proposal repo is that one can put an iterator into array, useful but not ergonomic. I personally agree with the non-ergonomic. Why is that less ergonomic than array.zip? + +JHD: So for—if you have two iterables that you want to zip together, and then make an array, you have to use `Iterator.zip` to get the iterator and do `.toArray` on it and that’s fine. It’s not unergonomic in a blatant way; it’s not that bad. It’s having to do it in two steps, when a frequent use case will be both of those steps. + +JHD: It doesn’t feel clean and elegant to me. I hear you - it’s a fair point, when we talk about ergonomic benefits, the win is often much larger than I think you will get from here. + +SYG: It’s also like here is—here is a slightly different incantation for exactly the same thing. And I—I don’t know why we would have that. But like if we iterator.zip, the cost of adding this is also pretty small. So all in all, I am not against it. But I—I am just unconvinced on the ergonomic part of it. Dot to array is pretty short. It’s method-chaining. It meets the check marks of, like, ergonomic.programming. I personally think it’s really good. + +JHD: That’s fair feedback. So yeah. I would like to ask for Stage 1. + +RPR: Any objections? Thumbs up from CM. Any objections to Stage 1? + +RPR: No? All right. You have Stage 1 + +JHD: Just to clarify, for—in order to meet the—some of the expressed bar for Stage 2, we need to see some usafe of the `Iterator.zip` functions to hopefully demonstrate one way or the other whether it’s common to immediately turn them into an array. + +MF: No cheating by putting them in a bunch of your libraries. + +JHD: Yeah. Obviously, my own usage wouldn’t count. And I hear SYG’s feedback that the ergonomic win is arguably non-existent. + +JHD: Thank you. + +RPR: Thank you, JHD. + +### Conclusion + +Stage 1 + +## Iterator chunking for Stage 2 + +Presenter: Michael Ficarra (MF) + +- [proposal](https://github.com/tc39/proposal-iterator-chunking) +- [slides](https://docs.google.com/presentation/d/1V2pFMn0s6UIdrjbfaBlfdu9XE4v3u6qD2gBwLRycVr8) + +MF: Okay. Iterator chunking. I will be looking for Stage 2. The problem we’re trying to solve with iterator chunking. We want to consume iterators in one of two ways. Either through overlapping subsequences of a given size or non-overlapping sequences of a given size. I will show you with the illustrations I have made. Let’s say you have an iterator of the digits 0 through 9. You have those numbers in order. Chunking, which is the common name for consuming non-overlapping subsequences, would produce an iterator—if you chunk by 3, produce an iterator of array 012, array 345, array 678, and then finally a singleton array for the final 9. If you were to do windowing, consuming overlapping subsequences, done by the size of 4, the array 0123 would be yielded, then the array 1234 would be yielded, and so on until 4567. Those are the two problems that we’re trying to solve. + +MF: We discussed the basic shape of the solution during the Stage 1 advancement. But now we have landed on this. I have spec text, it looks like chunks methods, each taking a size parameter. I have highlighted the boxes, the places they differ, but they are otherwise the same. In fact, we pointed in Stage 1, how chunks could have been implemented in terms of windows if windows had a configurable step parameter. But we decided not to do that and just have two separate methods. + +MF: So I am going to go over the design decisions and things we have considered through the process. We decided to have two methods, as in the Stage 1 presentation. As I said, you could have defined chunks in terms of windows. We don’t want to have a step parameter for windows, not because we don’t need to define chunks in terms of it, but windows with a step of not-1 is not a thing people really do. Also, if you end up doing that, if you have a step other than 1, you have to figure out how you like to truncate or pad the final windows. Things just end up getting weird and complicated. And it doesn’t seem worth it + +MF: For chunking, we saw in the illustration that when I chunked by 3, I had that last chunk, size 1. We could have decided to only yield chunks that were fully populated. But it’s better to—and more common in the things we looked at, to yield an undersized chunk instead of trying to, like, pad. If you want to do something like that, since you know the chunk size that you are chunking by, you can check the chunk and treat the last one— not of the appropriate length, specially however you want. That seems like the most widely applicable choice here. The most flexible. + +MF: We had considered having a default size for both chunks and windows of 2. Kind of arbitrary, there are some libraries we saw that do have a default size. I guess if there was a default size, 2 would be the choice. But it doesn’t seem worth it, if you want—you want to chunk by 2, it doesn’t help anything. And if you are not familiar with what the second parameter is, it makes it harder for a reader to know what is going on. + +MF: We decided to forbid 0 as a chunk or window size. Not because it wouldn’t make sense to do. And a lot of things allow it. But a surprising consequence of chunking by zero, you can turn a finite iterator into an infinite iterator and that would surprise some people who expected that any iterator transform you do, on a finite iterator would still result in a finite iterator. It’s not a safe assumption, but at least in this case for chunking, it would be. We needed to forbid zero and all negative values and the maximum chunk size is cannot larger than the array so that gives an upper bound as well. + +MF: In the Stage 1 presentation, we discussed whether we should have a sliding window that slides in and slides out past the ends. We would have to fill it with some padding element. This is less commonly used. I had given some use cases that do require it and found some code that does use it, but not only is it less common than sliding windows generally that don’t slide in and out, but you can also now, given we have iterator sequencing, you can recover this behavior very easily by just sequencing a couple of, you know, padding elements to the beginning and end of the iterator that you’re doing a sliding window on. So because you can recover it so easily, no need to provide that. + +MF: And lastly, this is something we never considered, but an option. You could yield iterators instead of arrays. But these are generally small, fixed length things. You may as well use an array. That was always the right choice. + +MF: So I think that’s everything I have. I would like to ask for Stage 2. And if there’s anything in the queue. + +MF: I don’t see anything on the queue. + +RPR: Any questions for MF? YSV + +YSV: When reviewing this, we were somewhat unsure about the motivation and somehow widespread, the use case, it needs to be implemented as a built in method. Do you have—you have libraries or something you are referring to? + +MF: Yeah. I didn’t include that information in this presentation because I had given all of that in the Stage 1 presentation. But in the repo, in the README, there should be at least all of the prior art we looked at in the Stage 1 presentation, I can pull it up and share it. I went through GitHub code search and looked at—tried to classify all the different kinds of use cases that there were. Let me pull it up + +YSV: I remember pagination + +MF: There’s carousels and – + +YSV: One question we had is how much does this benefit from being built into engines as opposed as being shipped as a library? + +MF: I am quantifying that. + +MF: So there are libraries that do this. The benefit is going to be more in convenience than anything. I don’t think performance is going to be any different than the user land implementation. Yeah. You know, it will be handy to have around on iterators that people are using + +YSV: So could we say that the benefit here is from the standardization coordination point and is that it’s centralized from us having it from a library and + +MF: I wouldn’t call that a coordination point. It would mean the decision—the design decisions have been made, and people aren’t doing chunking with slightly different edge cases. It’s not needed as a coordination point like other things we have as a coordination point. It’s not performance. It’s just convenient. I don’t want to oversell it. I put up the—I have the use cases for chunking there. The use cases for windowing here. + +YSV: Okay. It’s not an opposition to Stage 2. I want to think why we are moving things into implementations + +MF: Hopefully I have made that clear. + +RPR: SYG + +SYG: In terms of how to quantity these things, for things that are convenient, I think one .… it’s true to figure out how better caching of standard libraries and, if you move something in the standard libraries, it has instant loading, unless you like lazy loading behind the screens does not make it true, but compared to network. So one thing you could do is look at how widespread are use of these that’s very difficult. I don’t know how you find that. But if you could find some number, widespread page takings libraries are things that are used. There could be arguments for improving loading performance. But yeah. Otherwise, like it’s hard on pure convenience to really decide—yeah. + +MF: I would just say that, like for these iterator follow-on proposals, the ones I have taken up are the ones that are—I have seen most commonly in the ecosystem implementations. This is mostly through GitHub code search, because it’s very common to call these things both chunks and windows, it was easy to find lots and lots of occurrences of that and that’s how I did this classification that is on the screen. + +RPR: ACE? + +ACE: Happy to be one of the spec reviewers. I like this proposal. + +RPR: JMN has volunteered as a reviewer. + +MF: We are jumping the gun a little bit. But I would like to get Stage 2 first. + +RPR: All right. So any objections to Stage 2? Congratulations,MF. You have Stage 2. + +RPR: There’s also a + 1 from JHD, also happy to work with you. + +MF: Thank you. Let’s make sure we document in the note that is Ashley, JHD, and Jesse have all volunteered – + +RPR: Could you read a summary of the main point please + +### Speaker's Summary of Key Points + +Iterator chunking has two methods: chunks and windows taking a size parameter. It has advanced to Stage 2. Mozilla has expressed some uncertainty about—I wouldn’t say motivation. I will reach out to Mozilla between now and the presentation for Stage 2.7 to see if we have the right trade-off there. + +### Conclusion + +Stage 2 reviewers: ACE, JHD, JMN + +## `Error.isError` for stage 2.7 + +Presenter: Jordan Harband (JHD) + +- [proposal](https://github.com/tc39/proposal-is-error/issues/7) +- (no slides) + +RPR: All right. We are doing well in terms of getting through the agenda. Next is JHD. If you are ready to present on `Error.isError`, Stage 2.7. + +JHD: All right. So `Error.isError` is very simple. It contains a single AO with three lines. There was editorial feedback that perhaps the AO should be inlined and that’s fine. The stack proposal needs an AO, but that refactoring can also be done later. The spec text is straightforward: if it’s not an object, return false. If it has the internal slot, return true. Otherwise, return false. + +JHD: So the proposal with Stage 2, we decided to remove any special behavior around proxies. Which I think relieves most people. There is an HTML integration pull request which has not been officially stamped, but directionally approved. In other words, two of the editors of HTML spec are happy with the PR, and so this adds the error data slot to DOMExceptions so that they are true subclasses of error. That is also a relatively small change. I added the slot here, and add a little note about it here. And then the spec reviewers and editors have all signed off. So I would like to ask for 2.7 + +MM: Yeah. So, first of all, I approve for 2.7. The reason I wanted to say something, this is violating a general rule that I’ve been very forceful about, enforcing. And I want to explain why I am approving it anyway, and what we should interpret this as being a precedent for and interpret this is his not being a precedent for. + +MM: The rule that it violates is that it’s a built-in that looks at internal properties of a non-this argument. In a way that’s observably different than some object that might be in its place that tries to emulate it and because it doesn’t punch through proxies, it has the cost that we’re trying it avoid by imposing that, which is a simple proxy whose target is an array object that tries to emulate the array—sorry the error object, tries to emulate the error object, fails to emulate the error object with regard to this test. + +MM: The reason that I am okay with this is that with regard to practical proxy transparency, having written—practical membrane transparency, the behavior that I would recommend on practical membranes for error objects is to reconstruct a genuine error object on the other side, which has a transparency cost, which is a mutable property on the newError object is not automatically reflected into the original. There’s a disconnection of side effects. The reason why this is okay is because in practice, people just don’t do that with error objects, so breaking that side effect connection. On the error object, communicating back to the origin by mutating the property is arguably enough within the practical membrane design world constraints. + +JHD: Thank you, MM. + +RPR: + 1 from Chris. + +RPR: Any .… + 1 from Devon. + 1 from Chip. Do you want to speak? Chip has wanted this for years. All right. Just as to check, do we have any objections to Stage 2.7? Congratulations, JHD. You have Stage 2.7. + +### Speaker's Summary of Key Points + +- List +- of +- things + +### Conclusion + +- List +- of +- things + +## Restricting subclassing of built-ins + +Presenter: Yulia Startsev (YSV) + +- [proposal](https://github.com/tc39/proposal-rm-builtin-subclassing) +- [slides](https://docs.google.com/presentation/d/1J0xct8EHUC90P6QpggISxuAp_5L_qaDa3SZ9gQaB0kA/edit#slide=id.p) + +RPR: Let’s see. Is Yulia prepared to talk next? The topic, I guess it’s the restricting can you be class and built in methods. Is that appropriate now? We have got 23 minutes left and yours is a 20-minute time slot. + +DE: Yes. Why don’t you fill time with a joke. + +RPR: My wife, actually, she generally tries to avoid my gravy, and she was going to make her own. But to her credit, she stuck with me through thick and thin. We need to create another win as a committee. + +RPR: You know, a bear, even when it's wearing socks and shoes, still has bare feet. + +YSV: Okay. I will go quickly with this. So you folks probably remember equate some time ago, we introduced the first proposal to remove functionality rather than add it. Going through the proposal, phases. And one of the requirements for moving forward with that, so the proposal was to restrict subclassing support in built in methods. + +YSV: So Shu did an analysis. Here is the minimum support where you can simply extend an existing built-in and then, you know, you get back the type A when you make a new A. There is the next kind of subclassing, which when you use iterator—built in methods, they still return the subclass A. Then you can have this thing called symbol species, which allows you to override the type being returned from these—from these methods. + +YSV: And then finally, the most complex version of this is subclass type IV. I forgot the details of this. But we can go over it in more detail. One comment here is that type 3 and 4 are actually linked. You can’t remove type 4 and have a meaningful impact without removing type 3. That’s an important thing to keep in mind. + +YSV: Okay. So we were hoping in the best case, to take away support for type 2, 3 and 4 subclassing and only keep type 1. The hypothesis is here. We believe that this was both rare and—we had evidence of harm because we had a number of zero days and we also saw there was low uptake of this very powerful capability in the language. + +YSV: Firefox went ahead and landed teletri-to validate whether the feel this was in the being used was actually true. And this presentation is a result of what we found out. + +YSV: Our previous results show that the compatibility risk of unshipping species is real. Chrome counters estimated 2% of pages modify the constructor or species on an array RegExp and promise. Those are both undercount and overcount because it doesn’t count modifying.constructor. But it does count what is going on in shims. Which might not be a real use of those things. + +YSV: Manual inspection pointed to the library core JS. Which unconditionally installs things as the species getter if it doesn’t detect species. I believe that’s how it worked. And out of 1,000 sampled sites, 400,000 all 26 traps and 26 traps from core JS. + +YSV: Any questions? + +YSV: No. Unfortunately, not. I do have the types, a breakout of the types. Okay. Here are our results of per built-in and per type. What the breakdown was of the teletri-report. We were surprised by how heavily it was used on array. And regular Reg and promise. It’s higher than we expected. That's really high. + +YSV: Happily, it’s pretty share—it’s not non-existence on ArrayBuffers and shared array buffers and rare on TypedArrays. But you will also notice that this is for type 2 and type 3. So like in terms of the intervention we could do here, we can’t even remove the worst offender, the species that type 3. + +.YSV: So it validates the previous results. And when we did a dive into what is actually going on from these results, we did see that it’s corresponding JS doing future detection. We did consider running a second teletri-to separate out corresponding JS from the teletri-. But this is going to be quite resource intensive risk and given how high the numbers are, more than 10% for most of these, and this is really millions of pages load per day, we are not sure this is only core JS. We think this is a real problem on websites. + +YSV: And we can now unlike the previous version of the experiment, we that the paths that are are concretely paths that the ones that are hit. We say which built in are active and show that promise array and RegExp are surprisingly frequent. Whereas the other two I called out earlier, ArrayBuffers and TypedArrays are very infrequent. + +YSV: So what do we do next? + +YSV: We can abandon the work of trying to remove species. That’s one option. We can alternatively remove subclassing support from built-ins that are not impacted, array buffers and TypedArrays and shared array buffers. From our perspective, we think it is worthwhile. Shared array buffers are expense active. We don’t want to make that more sensitive than necessary. So we recommend removing from shared array buffers. We also suggest adopting a rationale for new built-ins that we don’t support built in subclassing beyond type 1 from now on. + +YSV: Yeah. So more on the topic of doing careful design. On shipping it’s extremely hard. We have been trying and the results are not promising. And it’s very difficult to fix mistakes after the fact. + +And it’s open to questions. + +RPR: Shu? + +SYG: Clarification the percentages are from all page loads? Okay. + +RPR: JHD? + +JHD: Yeah. I mean, it’s hard to—because I haven’t done those teasers apart from core-js, there’s no evidence to support an expectation in any direction. And I think that my personal instinct is you may be underestimating how many people have core JS on the site, it’s the default choice for babel since forever. When I suspected this was a problem, but I didn’t get the telemetry separated either, was when I was trying to make that change to `RegExp.match` (so that the match symbol wasn’t special). I am convinced completely that 100% of the triggers of that counter were from core-js. But nobody wanted to run the research and separate it out. + +YSV: So we have currently removed our counters, but we have the code for the counters. If someone is willing to invest the time to tease out the results, I can see about getting them back in. We need someone to volunteer to do that analysis + +JHD: When you say resource intensive - do you mean not having the counter collecting data, but rather doing the work to modify the counter code? + +YSV: And then go through all of the instances to figure out what the actual percentage of what core-js really is. + +JHD: Okay. Thank you. + +RPR: Justin? + +JRL: Another approach, can we—can we approach core-js to see if they'll remove the bad type things? + +JRL: I understand some of the sites will not pick up new code. But some of the sites might be massive and they might update for core-JS or babel and get the new things and the next deploy won’t have the awful typed stuff + +YSV: In fact, I believe if I remember correctly and I might not be remembering it correctly, what core JS is checking for the presence of species, if it doesn’t exist, it installs it. And we have no way in our teletri-code to distinguish what it’s doing with species. We know its species gets hit. Right? So it could be that we disable species. And we just revert back to type 1 as an experiment and see how much stuff breaks. But this is a horrible experiment to run on real users or get the list of websites and have a runner execute the site as a disabled version after the browser to see if everything still works. But even there, you know, it’s difficult to say that it’s working correctly. Because you have to be familiar with how the site works. + +RPR: Shu? + +SYG: I think this reply to Justin. That was covered by Yulia. Even if we—even if we update a core JS and got people to update to the core JS, we still have to run the telemetry again because the whole point is to—okay. For Yulia, I don’t follow the implications from—looked like core JS to now looks real because we got a lot of hits. What caused the additional now it looks real? Just because of the how large the numbers are + +YSV: Yes. We are much more nervous about—like, for example, what could potentially be a next step, 2%, what Chrome counters had done, we might say. Okay. Let’s see what happens if we run a beta and turn species off. When it’s 15% of the web that is happening in a given deployment, that makes us nervous of doing such an experiment. + +RPR: Nicolo? + +NRO: Yeah. You said that core JS hits the path because it checks the species and polyfills it. So like given that I don’t understand why we’re concerned about core JS. The polytriggers, so web with old versions of correspond JS would still have that + +YSV: So the problem comes from the fact that we are not certain about how much this is actually core JS and because we’re talking about millions of page views, this is going to be a lot to check by hand. In order to validate that things are not going to break. It’s a massive amount of work we’re talking about. + +RPR: Justin? + +JRL: To put my guesses into this, the type 3 for promise here is how you polyfill async context. You turn all your async functions into promise functions. And then you patch promises in order to capture contexts. So I think this is actually being used by real websites. RegExp, to me, seems like core JS, I can’t verify that and without something else to say otherwise, that seems risky. Array, though, in particular, I have written array extensions and used species and have my own custom array stuff. That seems like something you are taught in computer science. I fully expect web developers to have done this and have running code. Whether or not that site matters, I don’t know. Maybe it’s not performance, especially for large sites, but that seems like something everybody web dev + +SYG: What is taught in computer science + +JRL: Extending an array class. In Java, you extend the array to do whatever you want. That seems like a basic case that is taught in a computer science class, how you extend something, array is the default thing to extend. + +SYG: Why did you need to use species? + +JRL: Sorry. Not using species for that. Using array and then expecting it to return the instanceof my thing. + +SYG: Okay. I misunderstood. + +JRL: Okay. + +RPR: The queue is currently empty. Any more questions for Yulia? + +SYG: I would like to thank Mozilla for doing the legwork here. + +RPR: Would you like to continue, Yulia? + +YSV: What do we think about the proposed way forward? That we disallow it on ArrayBuffers and TypedArrays. And maybe if you were interested we can take a look at the RegExp to see if that’s fake or real. I don’t know how much benefit we get from doing that in terms of complexity and engines, but we can certainly take a look. And then the second part of the recommendation is to adopt a rationale to no longer introduce subclassing beyond type 1 for new built-ins. For example, anything that we add to the language going forward. If there is agreement that this is a good direction, we can make a pull request, removing the subclassing capability from array buffers and shared array buffers and TypedArrays. + +RPR: There is a lot of agreement. Mark? + +MM: Yeah. I didn’t really need to speak. Yeah. Let’s remove what we can, which is ArrayBuffers, SharedArrayBuffers and TypedArrays, and fix the future to not make the mistake again. + +RPR: Shu + +SYG: I don’t need to speak. Same. + +RPR: So + 1 to rationale, also willing to try for the array buffers. Jack Works? Supports. Justin supports. Ross supports. All right. We are at the end of your slides. Would you like to read out a summary? + +### Summary + +Mozilla gathered telemetry on real-world use of species. We were surprised that the number of pages using species could be as high as 26% for some of the builtins. Every case we investigated was CoreJS, but given that this is millions of sites, we are not confident that this is potentially web compatible to remove. Discussion focused on whether or not the use cases appeared real, conclusion was that they are for Array and Promise, probably not for RegExp, but we can move forward with the conservative step of removing it from TypedArray, Array Buffer, and Shared Array Buffer, as the usage was close to 0 for those builtins. + +### Conclusion + +- TC39 reached consensus on removing subclassing from typedArray, AB, and SAB. +- Possibly investigate regexp further. +- Add rationale for future built in subclasses not be included in the built-ins. diff --git a/meetings/2024-10/october-10.md b/meetings/2024-10/october-10.md new file mode 100644 index 00000000..d5e3ae25 --- /dev/null +++ b/meetings/2024-10/october-10.md @@ -0,0 +1,1189 @@ +# 104th TC39 Meeting | 10th October 2024 + +----- + +**Attendees:** + +| Name | Abbreviation | Organization | +|-------------------|--------------|----------------| +| Yusuke Suzuki | YSZ | Apple | +| Keith Miller | KM | Apple | +| Michael Saboff | MLS | Apple | +| Waldemar Horwat | WH | Invited Expert | +| Jonathan KUperman | JKP | Bloomberg | +| Ujjwal Sharma | USA | Igalia | +| Yulia Startsev | YSV | Mozilla | +| Ross Kirsling | RKG | Sony | +| Linus Groh | LGH | Bloomberg | +| Chris de Almeida | CDA | IBM | +| Andreu Botella | ABO | Igalia | +| Richard Gibson | RGN | Agoric | +| Chengzhong Wu | CZW | Bloomberg | +| Marja Hölttä | MHA | Google | +| Oliver Medhurst | OMT | IE (Porffor) | +| Devin Rousso | DRO | Invited Expert | +| Chip Morningstar | CM | Consensys | +| Daniel Minor | DLM | Mozilla | +| Justin Ridgewel | JRL | Google | +| Ashley Claymore | ACE | Bloomberg | +| Ron Buckton | RBN | Microsoft | +| Jordan Harband | JHD | HeroDevs | +| Dmitry Makhnev | DJM | JetBrains | +| Jesse Alama | JMN | Igalia | +| Istvan Sebestyen | IS | Ecma | +| Shane F Carr | SFC | Google | +| Justin Grant | JGT | Invited Expert | +| Eli Grey | EG | Invited Expert | +| Jack Works | JWK | Sujitech | +| Daniel Ehrenberg | DE | Bloomberg | +| | | | + +## Immutable ArrayBuffers for Stage 1 + +Presenter: Mark Miller (MM) + +- [proposal](https://github.com/Agoric/tc39-proposal-immutable-arraybuffer) +- [slides](https://github.com/Agoric/tc39-proposal-immutable-arraybuffer/blob/main/immu-arrayBuffers.pdf) + +USA: Good morning everyone. Welcome to the third day and final day of the 104th TC39 meeting. The first topic of the immutable ArrayBuffers for Stage 1. We can see you screen. Take it away when you can. + +MM: Before I get started, I would like ask everyone’s permission on line and in the room if I can record the presentation using keynote record that is the slide climbing plus the audio. I will turn the recording off when the discussion after the presentation. The discussion happens during the presentation would be by default be recorded although I will be happy to edit out anything that discussed during the presentation by request. So does anybody object to my recording during the presentation? Wonderful. + +MM: Today I am presenting immutable ArrayBuffers for Stage 1. I’m Mark Miller of Agoric and the cochampions are Peter Hoddie of Moddable and richard Gibson of Agoric and Jack Works here at the meeting in general but not yet in the room. As a Stage 1 proposal, the focus should be on a problem to be solved rather than the specifics of how we plan to solve it although explaining the specifics as an example certainly good for illustration. So the problem statement is the need for immutable bulk binary data. And this has several motivations. + +MM: Moddable, well, embedded JavaScript, JavaScript for devices such as standardized TC39 and implemented by moddable in the XS engine is in a situation where ROM is much cheaper and plentiful than RAM in many target embedded devices so there’s a big need for embedded devices if that you have a bunch of data like images, for example, a bunch of data that in fact is not going to change runtime, you would like to be able to place it in ROM without having to do crazy bookkeeping to pretend that it’s mutable. Right now, in fact, people doing JavaScript for embedded place data in ROM making it, in fact, immutable but they do it by cheating, by deviating from the spec. We want to heal that by enabling them to do that within the conformant spec language. We’ve heard from several people I believe including kevin Gibbons but heard from several people that on the web, there’s—we’ve observed various patterns of defensive copying that one API receives a bunch of data and because they need it to be stable and can’t count on the provider of the data not mutating it, that the receiver then makes their own copy. And especially when the data in question is bulk binary data, that can be a very expensive copy. It would be good to be able to get the same level of defensiveness without needing to do defensive copying. + +MM: "Agent" is the JavaScript name for a separate unit of concurrency like worker. The general term is agent. And by normal agent here, I want to distinguish agents that are allowed to engage in shared memory multi-threading allowed to use shared ArrayBuffers to communicate through shared memory versus normal agents that are only able to do communicating loops and communicate through post message. On the web at least, we’re forcing an opt in for an agent to be able to engage in shared memory multi-threading so the common case will remain normal agents as I’m calling them here. We would like to enable them to share bulk binary data at zero copy without the expense of a copy and because it’s immutable, there’s no concurrency problem in sharing the data. And in fact, depending on how the JavaScript engine co-exists with the operating system, they might be able to use the MMU to get MMU protection for the actual data itself, they might be able to map the data into multiple agent address spaces if they’re in different address spaces. If they’re in the same address space, that goes away. Similarly agents within the machine, there’s communication of agents on different machines. Agoric is participating with Spritely and Cap'n Proto the standardization of the network protocol messaging protocol called OCapN and intended to be inter language standard and not JavaScript specific standard. That one contains both bulk strings as things passed by copy as well as bulk binary data that is a big array of bites passed by copy. The language biding, things that are passed by copy over the network, it’s very nice to have the local representation be immutable so that it doesn’t—so that the local copy does not diverge from copies that have been made remotely after the fact. + +MM: For strings we have that perfectly in JavaScript with JavaScript strings. We would like to have something similar for bulk binary data. And the result would be something that looks very much like the previous case of sending bulk binary data that’s immutable of post message to share with agents within an agent cluster. And the surface form that we generally find that we want, we often find we want, this isn’t speaking for everyone, but speaking for the cases we have encountered in our own API taste is we generally find that we want at the end of the road to see this stuff through a frozen TypedArray. And because TypedArrays themselves don’t have their own data, but rather they’re windows on to ArrayBuffers, all the data is in the ArrayBuffer and currently there’s no way to create an immutable ArrayBuffer be this—has the implied restriction that there’s no way to create a frozen TypedArray. And there’s no way to shim it. There’s no practical way to shim it. + +USA: On the queue we have a clarifying question by WH. + +WH: I see a bunch of proposed ways to implement this on the slide, but don’t we already also have this functionality in the form of strings with just a thin shim to access strings as though they were immutable byte arrays? + +MM: You can certainly store bulk binary data in strings and with the understanding that the strings underlying representation typical enI didn’t know is the array of 16 byte values. You can even arrange to do it and that those values are not constrained to represent characters. You can certainly do that. That’s a good point. I retract the statement that there’s no way to practically shim it. We would still like a more natural way to do this and like to be able to use genuine TypedArrays and freeze them. But you are correct. + +WH: I was just wondering what am I missing here. + +MM: No, that’s a good point. And since right now we actually have a need for shimming it, that’s actually a good practical suggestion for our immediate needs. We should think about that. + +WH: Thank you. + +MM: In any case, there’s various places to start. One is just the JavaScript normal arrays, this seems like an unsuitable place to start because of all of the object-like natures of normal arrays including things like holes and that any one of the properties might be an accessor property, et cetera, et cetera. That’s not where we chose to start, but as struct arrays and SYG presented them as part of the struct proposal and struct arrays mature maybe we could revisit whether struct arrays might be a good place to start. That’s not what we explored to date. TypedArrays itself is normally what we want as I mentioned, but not a suitable place to start, because the TypedArray has a backing buffer and the backing buffer, if it’s mutable, the TypedArray cannot be frozen. DataViews have the same problem. They don’t have the data but just view on an ArrayBuffer. Web array has extraction called the Blob. Right now it's just part of the web and not part of JavaScript. We can consider adopting it. However, it has extra features that are well motivated for the web and for the uses made of Blobs like mind types that are really besides the point for what we’re asking for. So JWK does have a previous proposal. + +MM: There’s jack. JWK does have a previous—let me mention. I’m recording the presentation itself by permission. I’ll turn the recording off before the discussion at the end. But by default, discussion during the presentation will be recorded. But I can certainly edit things out by request. + +MM: Okay. So JWK does have a previous proposal where he started where we’re starting which is ArrayBuffers in his proposal called limited ArrayBuffers. And that proposal for various reasons did not proceed beyond Stage 1 but it did get to Stage 1. And we believe that the current proposal that jack is a coach champion of avoids the impediments that cause the previous one not to proceed beyond Stage 1. So today, we’re again starting with ArrayBuffers as our place to start. + +MM: This is a realistic piece of code authored by RGN of using the abstractions we’re proposing as part of doing realistic interaction with the protocol. In this case communicating binary data between processes. The important expression in this piece of code is just this one. This is the one that does all the interesting work. So this is kind of a preview of the API we’re proposing. `buf` is a normal TypedArray into which the data that we want to transmit has been placed. `transferToImmutable` does a transfer analogous to the existing transfer and transfer to fix size operations. Does the transfer to immutable creating a new immutable ArrayBuffer detaching the original and the only reason for that is so that we can create a TypedArray on it that we can freeze. And then the rest of this just makes use of the fact that now we have a stable frozen TypedArray. + +MM: So until recently, this was the entirety of the ArrayBuffer API. And it has a slice method for extracting out a sub range by copy into a new mutable ArrayBuffer. And it has a getter only accessor for getting the length of the array. Since then, we’ve added these three methods. `transfer`, `transferToFixedLength`, and `resize`. And we’ve added these additional getter-only accessors. And what these transfer things do is they create other—well, the transfer we started with transfer, it just creates a new ArrayBuffer that’s like the original detaching the original and the reason it detaches the original is so that it can create the new one with the same data without having to copy it. So that’s a zero copy transfer-exclusive access to the data. Transfer toFixed length is similar but creates a flavour of a non-resizable ArrayBuffer which is one that is less mutable than the original if the original was resizable but contains all the same data. For a resizable one, the resize operation, and then these first two getters on the accessors detached and resizable lets you look at an ArrayBuffer and see what flavour you have got. I’ll step through the flavours. This API already enables three flavours of ArrayBuffer. There’s the resizable ArrayBuffer for which the interesting operation is `resize`. It’s identified with the resizable being true and it has—it’s interestingly has both a byte length and a max byte length that can be different in that the max byte length that you preallocate where the byte length can grow up to that limit. Transfer toFixed length creates something that is less mutable than a resizable ArrayBuffer in that the size is no longer mutable. And in this one, the resized method is useless. If you call it on a non-resizable ArrayBuffer, I believe it throws. And the getter only accessor identifies this case by the resizable accessor returning false. And then all of the transfer operations, what they leave behind is a detached ArrayBuffer in which the detached accessor returns true to identify it. And all of the other members, both methods and accessors are useless. + +MM: The reason I went through the existing API is we’re very happy to have fit the functionality that we’re looking for very naturally as an extension of the API we already have so as to minimize conceptual cost, minimize—have it be a very natural additional API to learn giving that you’ve learned the existing API. So like transfer toFixed length reduces the mutability of—creates a new ArrayBuffer detaching the original where the new ArrayBuffer is less mutable. Transfer to immutable is similar, it detaches the original and creates a new ArrayBuffer with the same data zero copy, but the new ArrayBuffer is much more—has much more reduced mutability in particular all of the data is now immutable. And there’s a corresponding getter-only accessor immutable that identifies this. So this API enables the following additional flavour which is the immutable query for this one returns true and slice continues to work because that’s just a query. It doesn’t modify anything. But since the thing is immutable, you also cannot detach it. So none of the other operation, resize or any of the transfers work on this flavour. And an important thing to know, and this is a difference with the previous limited ArrayBuffers is with this API, an ArrayBuffer that’s immutable is born immutable and forever immutable. And ArrayBuffer that is not immutable is born not immutable and is forever not mutable. So there’s no making it an immutable in place. And that’s similar with not being non-resizable. You don’t make a resizable ArrayBuffer not resizable. You make a new one not resizable and born not resizable and stays not resizable forever or until it’s detached. + +MM: To enable frozen TypedArrays as a consequence with immutable ArrayBuffers we’re proposing to modify the TypedArray specification in a way that fits very well with what SYG presented earlier as the prevent extensions modification of the TypedArray specification which is right now TypedArrays, all of the indexed properties of a TypedArray are dealt with by this block—by this special handling block in the specification, the special case for TypedArrays for indexed properties for anything other than indexed property just falls back to the normal get only. For index properties, what we have been doing until now is they always describe themselves as configurable writable. And they needed to do that because for both resizing and detaching purposes, they have to be configurable in order to account for the future changes that might happen and also because the data value can be changed. They have to be writable. What we did for prevent extensions is we allowed them for non-resizable ArrayBuffers but prohibited for resizable ArrayBuffers for a similar philosophy in order to account for what mutations are possible in the future or if those mutations are not possible to enable the TypedArray to visibly commit to stability that it will actually have. So with this particular specification does is it says if you make a TypedArray on an immutable ArrayBuffer, then all of the indexed properties of that TypedArray are born non-configurable, non-writable data properties and therefore forever that way and once again it’s important to point out a TypedArray made on an ArrayBuffer is forever a TypedArray on that ArrayBuffer. There’s also no ability anywhere in the language to change what the TypedArray is a window on. + +MM: So to sum up that point, when you make a TypedArray on the various forms of buffer, if you make it on a resizable one you cannot prevent extensions, if you make it on non-resizable one you can. If you make TypedArray on the buffer or TypedArray detached then the TypedArray is useless. Finally a TypedArray on an immutable ArrayBuffer is TypedArray that you can freeze. There’s also the issue that’s outside the JavaScript specification that is increasingly part of JavaScript on more and more hosts including non-browser hosts and structure clone started as a creature as a browser in web APIs, but it’s in general a common way to account for the semantics of communicating between agents, for example, by post message. And structuredClone, the API lets you say which things you want to transfer and which things you want to copy. If you don’t say you want to transfer, then you’re implicitly copying. So on a resizable ArrayBuffer or a non-resizable ArrayBuffer, it would copy it with the overhead of copying because the original and new one are both mutable and they can—and they then mutate separately. Or you can do a zero copy transfer leaving behind a detached one. Doing structured clone on a detached ArrayBuffer is useless. Doing structured clone, here is the interesting part, doing structured clone on immutable ArrayBuffer can do zero copy sharing of the data. It will create a new ArrayBuffer as a new object with a new object identity but it can directly share the same underlying data if the implementation is able to share that data between the communicating parties. And because it can’t be detached, you can’t transfer it if we maintain the semantics, the implied semantics, the transfer means that the original party loses it. + +MM: So the open questions in the proposal as we have written it down is transfer and transferToFixedLength both have a new length property—sorry, a new length argument, an optional new length argument. There’s two stances one can take with regard to the transferToImmutable. One is orthogonality. You can transfer to immutable if you’re about to detach any way, there’s no particular way to the no resize to the way you want orthogonality that the other two have the parameter and it would be pleased surprise to have the parameter here as well. That’s an open question. + +MM: The slice operation because it creates a new mutable ArrayBuffer cannot be zero copy even if the original is an immutable ArrayBuffer and that seems like a missed opportunity or would be a missed opportunity particularly missed opportunity to do what jack works did in his earlier limited ArrayBuffer proposal which is to be able to create a window, narrow window on the ArrayBuffer and do it at zero copy cost and then you have TypedArrays that reveal only that window of data rather than revealing the entirety of the data. That’s especially relevant potentially in the case of where the underlying original thing is some large flat address space like you have in a wasm-like world. So sliceToImmutable could be like slice but producing an immutable flavour ArrayBuffer and original and new one are immutable then obviously you can do that at zero copy overhead. + +MM: RGN’s presentation on order of operation on constructing ArrayBuffers and where you get the errors reported, there’s similar questions for the implied failures to mutate. And in particular, because of the history behind ArrayBuffers, if there are cases right now where a failure to mutate such as by assignment to an index property on a detached ArrayBuffer is silent which is quite annoying, but that’s what we have got, so there’s an issue going forward of now that we have new reasons for failing to mutate, whether we report them with an error and if so which error and what’s the order of operation that determines which error is reported if multiple things are wrong? Also, I had not mentioned SharedArrayBuffer or atomics in the talk so far, and we really haven’t thought about it much. Atomics, as I understand it, you can do all the operations on the normal ArrayBuffer, it just doesn’t mean very much. So being able to do them on an immutable ArrayBuffer in the query part seems to make sense. But in any case, as I said, we really haven’t examined that, so examining the relationship to shared ArrayBuffer in atomics is something to do. The status is that we’ve got spec text that talks about both the ArrayBuffers and the TypedArrays as I mentioned and also the DataViews. We have a partial shim. Partial because of the constraints on shims I mentioned. It shims the immutable TypedArray and does it securely in the sense that the virtual TypedArray, the emulated TypedArray that it presents is encapsulates a genuine TypedArray exclusively so that it really does prevent the underlying data from being mutated, but because the encapsulating wrapper is not a built in ArrayBuffer, there’s no practical way to create a TypedArray on the emulated immutable ArrayBuffer. And that is our status. And I would like to ask for Stage 1. And that leads to discussion for which by agreement, I will now turn off the recording. + +USA: All right. Thank you MM. Going with the queue, first on the queue we have OMT. + +OMT: I've used strings as array before and they can be used but this would be how for me at least. + +MM: Sorry. I couldn’t quite hear that. Can you say that again? + +OMT: I've used strings before as byte arrays and they are useful but they are not nice to use. This would be very nice. + +MM: Okay. So that’s along the lines of WH’s suggestion and good suggestion. Since like I said, we currently have a practical need to shim it if we can, we should take a look at that. I didn’t think about it until WH suggested it. + +JHD: If this achieves Stage 1, will the limited ArrayBuffer proposal be withdrawn as part of that, or would they coexist? + +MM: I will turn that question over to JWK. Somebody hand—I took your microphone. I’m sorry. + +JWK: This proposal solves half of the problem that the limited ArrayBuffer intended to work out as MM just mentioned, there is also a limited slice of an ArrayBuffer in case the ArrayBuffer provider want to let someone else to see only limited slice and cannot get extended to the whole ArrayBuffer, for example, when the ArrayBuffer is a wasm memory. If this proposal advances, I think I may change the old one to remove the frozen part and maybe represent to the committee in the future, and let’s see if the remaining part is still convincing to the committe. + +CM: So my question is probably most easily visualized in the case of ROM but more general with JavaScript not having static data declaration (perhaps with the exception of strings): if you get one of these by invoking transferToImmutable on a regular ArrayBuffer, in the case of something like a ROM how do the bits get in there? + +MM: Great question. So I won’t speak for TC53 because I’m more familiar with the specifics of XS than I am with the TC53 standard and I know it conforms to TC53 standard and it is represented somehow in TC53. In XS the execution model is basically two phases. There’s a first phase of execution that happens in development on the developer’s machine and then there’s a snapshot event that looks very similar to the hardened JavaScript lockdown that also XS has and I’m not sure I can speak to the relationship between snapshot and lockdowns. I won’t try to speak to that. But snapshot basically says take a snapshot of the state of the machine. Basically of the state of the heap. And turn it into something that in which all of the mutable parts are now in ROM and then you can continue execution from there on the device. It’s very clever and works very well. + +CM: I’m familiar with the XS snapshot mechanism. So there would be some annotation that the engine itself would understand knowing this over here is immutable and the stuff over there has to be maintained in the mutable portion of memory. + +MM: Right now what they do is before snapshotting is they make the tremendous number of objects transitively frozen and they have inside the engine a purity test. Are these things pure meaning there’s no hidden mutable state and no hidden I/O and for the things pure and frozen ArrayBuffer and frozen immutable ArrayBuffer and a frozen TypedArray, those things would in fact be pure, and if you make them pure before you snapshot then all of those things would just go into ROM and then would continue execution on the device as data in ROM. + +CM: That satisfies the ROM case very well. It doesn’t solve more general question of how do I get a big blob of static immutable data into my code. That’s a separate problem, I guess. + +MM: I mean, it’s what you already understand which is that the JavaScript initialization would need to be expressed as put a whole bunch of data into a normal ArrayBuffer and then do a transferToImmutable on it. So it would be, you know, module initialized time computation. And whether we want to introduce some more declarative form of just here is an immutable ArrayBuffer pre-populated with data, that’s something we can examine. Certainly not something we want to make as part of this. + +CM: I imagine you could do something with typed imports or something like that. But that’s clearly out of the scope for this. + +MM: Yeah, that’s an interesting one. The same thing we can import other flavours of module-like importing JSON. Importing just, you know, a bunch of bytes from a file into an immutable ArrayBuffer is something that one can imagine is another type of import. Once again, not something that I would like to see in the proposal but can be another follow-on proposals. + +MM: Can someone project the queue? + +KM: I will ask my question. And this isn’t block Stage 1 at all. + +MM: I’m sorry. Your audio is muffled. I didn’t hear the beginning what you said. + +KM: Sorry. This isn’t a block of Stage 1. I’m curious if you thought about instead of transfer to immutable just having an option on transfer toFixed that says I want immutable buffer back? It’s not one way or the other but… + +MM: That one actually I never thought of that. I never heard anybody else mention that. That’s an interesting suggestion. Yeah, I’d like to—let’s consider that an open question. It certainly fits all the constraints. I would be curious what the feedback is from the room about which API feels more parsimonious. I did see the queue projected for a moment. And then it blanked out. + +USA: Next on the queue we have SYG. + +SYG: I have some clarifying questions on the freezing and the structured clone. For freezing, remind me again what did you propose to be freezable? + +MM: TypedArrays. + +SYG: TypedArrays that are backed by – + +MM: Immutable. + +SYG: I see, okay. + +MM: Just to be very, very pedantic, a TypedArray that’s backed by a fixed size empty ArrayBuffer is already freezable. The reason why one that’s backed by nonempty ArrayBuffer is not freezable is because of the non-configurable indexed data properties. And if it’s backed by a mutable ArrayBuffer, then the specification change that enables the freezability is only that the properties, the indexed data properties are described as non-configurable and non-writable display properties and that combined with the change we agreed on to prevent extensions has the emergent effect that now you can freeze these. + +SYG: Second clarifying question on structured clone your intended semantics for structured clone is if you structured clone an immutable ArrayBuffer, you get a new ArrayBuffer with the same backing store that is a new object but the old ArrayBuffer, the source is not detached because there’s no need to? Is that the extent of it? + +MM: It’s both because there’s no need to and would violate immutability to detach it. And we’re saying immutable flavour cannot be detached and transfer are disabled. + +SYG: And it would violate the immutability to detach? That’s the operation on the ArrayBuffer object itself not operation on the underlying bytes? + +MM: That’s a good point. It depends on what we intend to mean by immutable. So we could consider that an open question during phase 1. + +MAH: We cannot consider that an open question. That would violate the freezability of TypedArrays. + +MM: Yeah. Exactly. Thank you MAH. If the underlying ArrayBuffer could be detached, than a TypedArray on the ArrayBuffer could not be frozen. + +SYG: I see, okay. So it’s to make the invariants work. So I prefer your proposed semantics. I don’t think we should detach it. I’m trying to think through if there are any problems with that. Like, given that you still get a new ArrayBuffer object. You can freeze an immutable—like, okay when you create—when you get a new immutable ArrayBuffer back, is the ArrayBuffer object itself born frozen? + +MM: It is born immutable and not resizable. The bytes are immutable and not resizable and not detachable. But it still as an object, it’s not a frozen object, it’s not even a non-extensible object but you can make it non-extensable and you can freeze it. + +SYG: That answers my clarifying questions. + +KM: Seems like there’s not really any value add to allowing a transfer any way in the sense that you can’t write the bytes. You don’t want the bytes on your old like worker or agent. You would just throw it away and let the garbage collector clean it up? + +MM: I’m sorry. I don’t understand. + +KM: I guess I’m agreeing with the current semantics. The proposed semantics because I don’t see any value add to allowing a transfer independent of that. Because if you don’t want your bytes anymore, you just let the object get garbage collected. + +MM: Right. So good. Thank you. So if you want the bytes and a new agent and you simply know that the bytes are now irrelevant in the original agent, you could—the existing abstractions are perfectly good for that where you’re detaching it, or you could use this new immutable ArrayBuffer, share it and then drop it and let it get garbage collected. But, yeah, if it’s not needed in the original, then this proposal doesn’t add any value to what we have now, because detaching is already a zero copy transfer by detaching it in the original. It’s for some lightweight concurrency use cases among agents that are not authorized to use shared memory multi-threading, it’s very nice to be able to have the same underlying bulk data, for example, images, whatever, accessible simultaneously for multiple agents since you can do it by sharing without copying over it. + +DE: I think we’ve partly started to discuss this. But I like this design. This seems like a very simple design and very in line with what we established with ArrayBuffer transfer. But because a lot of uses within a particular process might be in terms of a single ArrayBuffer going through different modes where it’s potentially read by multiple things in immutable way and then is written to and at that point immutable things have to be kind of invalidated, but that might be read potentially by multiple things again, a reader writer lock would be able to encapsulate that pattern? There’s an API design proposal by Jake from Shopify and I think this is an interesting thing to consider in this space. + +MM: So let me make sure I’m understanding. The reader writer lock would be specifically in order to do concurrency control around SharedArrayBuffers. + +DE: So I’m using the term reader writer lock potentially loosely. It’s not necessarily about concurrency as such. But you have a sort of holder for the ArrayBuffer and then from it you can either get out one, you know, writable thing or you can call it multiple times and get these, you know, read views to it. + +MM: So this is interesting. So given that we’re not talking about SharedArrayBuffers in autonomous and just talking about the Rust-like ownership and borrowing model to do reader writer locks. You can do half of it, zero copy within this model. You cannot do the other half zero copy which is given a read write ArrayBuffer not SharedArrayBuffer is therefore necessarily exclusive to one agent, you can transfer it to an immutable ArrayBuffer zero copy that you can then share, but you can’t go the other way. You can’t go the other way zero copy. There’s no way to sort of collect all of the references to the immutable one, invalidate all of them and make a new exclusively held mutable one at zero copy. And I don’t see a way to accommodate that. I mean, if there is a way to accommodate that, if this is something that we should be trying to consider, we could consider it an open problem. But I don’t see a way to approach that from where we’re starting. + +DE: Shouldn’t we make them all be detached? I feel like I’m missing something. + +MM: I’m sorry. Say that again. + +DE: Shouldn’t we make them all be detached? + +MM: We could if we allowed them to be detached. So that goes back to SYG’s question. So let’s go ahead and continue to consider that an open question. But once again, if you allow an immutable ArrayBuffer to become detached, IE, if the non-detachment is not a permanent commitment, then you cannot within the object invariance freeze a TypedArray on that immutable ArrayBuffer. And we really do want to be able to create frozen TypedArrays. + +DE: I see. That makes sense. Yeah, I agree that the reader writer lock doesn’t make sense if you never give up the reader privileges. Thanks. + +MM: You’re welcome. That’s a good question. I haven’t thought about the reader writer issue in this context. I see a clarifying question from KM. Is this considered a cooperative multi-threading environment or adversarial? So that’s a great question. Actually I’m sorry before I answer the question, KM, did you have more to say on the question than the title of the question? + +KM: Sorry. I’m not talking about the reader writer lock situation. You have a cooperative one, you can just agree on some SharedArrayBuffer and cooperatively have a lock on your thing. In an adversarial environment then obviously that doesn’t work. + +MM: But even in a cooperative environment, because of the constraints here, there’s no zero copy way to make the transition back to a mutable ArrayBuffer within this proposal. + +KM: I think there would be that you would wrap your ArrayBuffer in an abstraction that like, I don’t know, strong person proxy and try to write when it’s in the shared state is in the read-only mode, like, the reader lock is held, then it would trap. + +MM: Putting a proxy in front of it and trapping is something that I think is—would be, you’re an engine guy, high performance engine guy and I’m not. So you’re the one who is more authoritative on this. I would think that would be from the performance perspective, just contradict the performance goals of dealing with bulk binary data. If you wanted to support that kind of thing, you have to move something like a copy on write support directly into the engine which Mathieu has suggested in other contexts, in other very closely related contexts. We’re not proposing that in this proposal. And I don’t think—I don’t see a way to do it practically within the performance goals of bulk binary data without something like that. + +KM: Yeah, that’s a fair point. Not arguing as a performance thing. I was arguing that it was potential possible independent of the proposal. + +MAH: So from what I understood from DE’s question is more to do with views than with the buffer itself. And I think that is what the proposal from JWK is more about being able to create limited views that are read only, that are on a subset of the data and things although that I can while being able to maintain a writer view to that ArrayBuffer. But the ArrayBuffer itself would remain mutable. So it’s possible that there is still a way to integrate both proposals, but I see this as oring thattal problem. How do you provide limited views on an ArrayBuffer that is mutable? + +MM: Okay. So that makes sense to me. And I endorse what JWK said earlier which is given that this proposal advances to Stage 1, which we’ll see by the end of this, then he could rephrase the limited ArrayBuffer proposal to be the differential assuming this proposal and so I want to endorse that, there is that further functionality that can be explored. But I also want to endorse keeping it out of this proposal itself in order to keep this proposal really very limited and minimal. + +MM: WH says you can do this in this proposal. I don’t know what this is. + +WH: This is musing about DE's question. If you have some kind of a notion of copy-on-write, then you can use *slice* and *sliceToImmutable* to go both ways between mutable and immutable versions of an ArrayBuffer. I don’t like that very much because if you get it wrong and references don’t get GC’d quickly enough, you silently end up with multiple copies, which is not ideal. + +MM: Okay. So WH, are you satisfied that the previous answer applies to this question as well, that I endorse the exploration of copy—what you could do with copy on write, but want to keep the things that would imply such an optimization out of this proposal itself to keep this proposal minimal? + +WH: Yes, I agree. I want to keep these things out of this proposal. It would rapidly get complicated and brittle. + +MM: Good. I see another reply from MAH. + +MAH: Yeah, just clarifying that copy on write is an optimization that implementation could do today that doesn’t require any changes to the spec implementation as SYG mentions remains skeptical that it is doable safely. I remain skeptical that is actual case because I don’t find it different than detached. But I don’t know enough about implementations. But any way, I don’t think it relates much to this proposal in general. This is more about potentially views or ability to share—yeah, anyway. + +MM: It’s true that the existing slice operation creates a new mutable one that starts on the same state as the original and then both of them can be mutated to diverge from there. So copy on write if it were—if some engine chooses to implement it as an optimization could optimize the existing slice method without any particular relevance to what is being proposed today. It would just be existing and optimization API we already have. That brings us to SYG’s skepticism that is his reply. SYG. + +SYG: I don't really need to go into it, but we have talked about transparent copy on write optimization of ArrayBuffers when doing the size ArrayBuffer stuff and yes the language would admit transparent and copy on write and remains the case that V8 doesn’t want to do this for security risk reasons. + +MM: Let me just confirm with you that the—what’s being proposed here by emitting anything that would imply a copy on write optimization does not raise any of those objections. + +SYG: I haven’t checked with the security team. But as far as I can see, this is—I like this proposal very much. It doesn’t raise any security flags to me. + +MM: Right. + +SYG: The risk around copy on write is the on write part. + +MM: Good. + +USA: SYG again. + +SYG: Certainly not a Stage 1 blocker. I don’t think—it might not be any stage blocker. I want to just note because Wasm linear memory is exposed as ArrayBuffers that are currently design space of memory control. They want more complex memory control than this and want to remap certain sub regions than that. They want more than what this is proposing. I checked with some of the Wasm team members on V8 and the feeling is maybe this is just orthogonal to anything that you ask them. Maybe it is just fine. In particular Wasm memory can’t be detached. You just can’t use this. So my concern is if they also come up with an independent immutability machinery, it would be a shame that we just have a second different thing that’s different but not really. But I just want to register that they are thinking about the space but I don’t think it really holds up anything here. + +MM: Okay. Great. So thank you for that. We’ll keep an eye on that. But I’m glad that it seems unlikely. CA point of order, two minutes remaining. Is there a use case for slice? SYG again. Let’s see if we can cover that and I can ask for Stage 1 within the remaining two minutes. Is that possible? + +SYG: Yeah. So quick question: The slice to immutable thing sounded like the motivation is ticking a box. We have slice and why not slice to immutable? We asked methods to aArrayBuffer and TypedArray and sort the bytes in the TypedArray. Slice on ArrayBuffers themselves is also like—if there is actual use case from moddable or whatever that seems find. If not, I don’t want to keep proliferating new methods that don’t have any use. + +MM: Okay. I will just note that I included the slice to immutable in the open questions part of the presentation, which I think it sounds like it’s compatible to consider it an open question. With all of that, I would like to ask for Stage 1. + +MM: I’m seeing a thumbs up from CM and support for JWK and DLM and WH. Do I have any objections? I didn’t hear any objections. I think I have Stage 1. + +USA: Great, congratulations MM. + +MM: Thank you everyone. + +USA: Yeah, I guess I need to figure out how to calculate Waldemar’s point [Waldemar wrote –i²], but, yeah, thanks everyone for this topic. Before — oh, yes, so, MM, would you make any concluding remarks or give a summary of the discussion. + +### Speaker's Summary of Key Points + +MM: With the Stage 1 problem statement of finding good ways to support immutable bulk binary data and with the immutable ArrayBuffers as proposed with spec text as at least an example—an ill US active example of how we are currently thinking of addressing that problem statement, we have achieved Stage 1 with, I think it’s fair to say, with no objections to the proposal part of this and with open discussions and notes with regard to things considered open questions with some additional open questions that I trust have been capture in the notes. But the Stage 1 is just the Stage 1 for, you know, for this. That was very rambling and incoherent, but I hope it makes sense. + +## DataView get/set Uint8Clamped methods for stage 2 + +Presenter: Jordan Harband (JHD) + +- [proposal](https://github.com/tc39/proposal-dataview-get-set-uint8clamped) +- (no slides) + +JHD: Okay. So I’ve presented this in the past. Basically, we have this nice little matrix with only two little ugly marks on them. `Uint8ClampedArray` does not have the `DataView` get or set method for it, and that’s the only kind of aberration in terms of consistency here. The contention that was talked about during the last discussion was that there may be, I think generally everyone’s okay with the idea of the setter, but some folks are not happy about the fact that the getter is identical to the one for Uint8Array. So it either does the exact same thing or it’s a literal `===` reference to the exact same function, and my argument is I think we should have that for consistency, and the counterargument is if it doesn’t do anything, why should we have it. It occurred to me this morning that Sets have keys, values, and entries, and why? The only point of that is consistency with the same triple of methods elsewhere. Empirically, we have decided that it is valuable to have consistent methods - API interfaces - even if they don’t actually have much use case. In this case, that’s what I would like to do, is I don’t personally care whether this is a distinct method here or whether it is just an alias to `getUint8` - I’m happy with either one. But I would like to advance to Stage 2 with both methods. Let’s discuss that. + +KG: Sorry, I didn’t have time to get on the queue and I have to leave, like, five minutes ago,. + +JHD: That’s okay. + +KG: But there is utility in having map have keys and values and set also having keys and values, which is it lets you write code that it doesn’t care which of those two things you’re passing in. + +JHD: That’s what I want these for. + +KG: That works because those things have the same name, and here you are proposing APIs with different names. So, like, you’re not -- + +JHD: So I discovered this problem because I wrote code that was trying to do exactly that, not care which type of TypedArray I was dealing with, and dynamically dispatch to the appropriate `DataView` method based on the type of TypedArray. This is one special case I had to hard code in there. If we patch those methods, then in theory, I wouldn’t have to do that and my generic code would be nice and clean and elegant and just work, exactly as someone would want to do with maps and sets and treat them agnostically. + +KG: I think that the sort of dynamic dispatch based on the names lining up is not a sufficiently good motivation to advance a proposal, but I’m not going to block, and I have to go, bye. + +USA: All right, thank you, Kevin. First on the queue, we have MF. + +MF: Yeah, I want to preface this with I am also not planning to block this, but, yeah, I am—I remain unconvinced about this motivation. The important thing I want to state, and I think I’ve stated this but I want to make it clear, I don’t want this to set any precedent for us to trying to fill in every square of a matrix. That is not something we should even attempt to do. It could maybe be used as a way to inspire us and say, oh, I found something useful now, but I don’t think we’ve done that in this case. But the reason I’m not opposed to this is because there are people who do say that it’s useful. I’ve not been convinced, but I think that’s fine. So I’ll let it proceed. + +JHD: And I will say that I agree that none of us should be hunting out inconsistencies and fixing them on principle. I discovered this with an actual use case and I made the matrix to attempt to bolster my case. If I wasn’t actually using both the getter and the setter, I would probably not care so much about having the matrix fully filled in. It’s that I actually need both of these methods. + +USA: Next on the queue, we have CM. + +CM: Yeah, I support this. Weird non-orthogonalities are confusing to users, even if you can come up with rationalizations for why they’re OK. In this particular case, it’s just one more bit of confusing detail that people have to learn about or perhaps stub their toe on, even if it’s something minor like this. And I agree it’s probably not worth trying to go back and patch the vast horde of historical stuff like that. But when we have the opportunity to kind of pick up the trash, we should do it. + +USA: Thank you, CM. Next we have PHE. + +PHE: Hey, yeah, I mean—I don’t really feel like this is very strongly motivated. There’s plenty of inconsistencies in the language, and you don’t actually have to look very far from this proposal to find them. We have Uint8Clamped arrays, but we don’t have Uint 16 clamped arrays, for example, why not? We have had TypedArrays for almost the entire history of the language that were completely consistent, and as a committee, we decided that Uint8Array would no longer—would differ from that by adding the hex and Base64 methods only to the Uint8Array. So not only are we not fix all inconsistencies, but we are content to create them. And so given how unbelievably obscure Uint8Array camped cases in the first place, I don’t think it’s really necessary to fix this. I understand it’s a small change, but I think it—it sets the committee in the wrong direction. So I really don’t think the motivation works for me. Thank you. + +USA: Next on the queue, we have SYG. + +SYG: I wanted to understand more of the use case. So, JHD, your use case was you said you had some TypedArrays and you wanted to use the constructor name of the TypedArrays to dispatch on data view methods? + +JHD: Not the constructor name. So this is—the use case specifically is I am adding in polyfills for some TypedArray methods. I understand that for performance reasons, most people wouldn’t want to use them, but I’m doing it anyway. And in the course of doing that, I had to implement some abstract operations in my library for that. And the spec is written in such a way that it dispatches based on the name slot or whatever of the TypedArray. So I’m reading that and I’m pivoting on that to be able to do those operations. + +SYG: But what’s the use case for that polyfill? Like, what is the polyfill? The use case—the polyfill doesn’t seem like a use case to me, but if user of the polyfill would be a use case. + +JHD: Yeah, yeah, so this was—so I’m happy to answer the question. But the more I do, like, this isn’t something that—I think everyone is aware that I have—that I like have compatibility farther back than I think most people care about. And that requires me to write polyfills for environments that are similarly old or niche or less used. + +SYG: Sorry, okay, to cut down on that, I’ll ask it another way. The alternative without this is that you would branch on Uint8ClampedArrays,. + +JHD: Yes. + +SYG: And that- okay. + +JHD: For the getter. I mean, so to be clear, I’m hoping that all this not strongly motivated stuff is talking about the getter only. Because the setter, that logic doesn’t exist anywhere else, so I had to manually guess at it and recruit it. So even if there are people blocking the getter, I would hope that I can proceed with just the setter. But for the getter, yeah, that’s what it’s doing. It’s branching on it and it’s just literally using the getUInt8 when it’s Uint8Clamped. + +SYG: That clarifies. Thanks. + +USA: All right. That was the entirety—oh, no, we have a reply by YSV. + +YSV: Yeah, if it’s for polyfilling an old browser, adding this to current implementations, wouldn’t you still need to do the branching? + +JHD: I would be able to use the polyfill in those old browsers. + +YSV: But you’re writing the polyfill and you need this for the polyfill, right? So then you would still need -- + +JHD: Oh, yeah, I see what you’re saying, yeah. The branching code still will need to exist somewhere, but I’ll be able to move that complexity so that the consuming code can just pretend it’s already there. But, yeah, I mean, I hear you that, like, this is why I haven’t gone into detail about the polyfilling use case to motivate this, because I understand that that’s not going to be compelling for most of the people in the room. And as you’ve said, it doesn’t allow me to actually delete my hacky code. It just lets me put it in a cleaner place, which is not a compelling reason for this room. It was more that I have the use case where I’m trying to dynamically dispatch on TypedArrays and this was a gap here. + +YSV: Okay, so it does make it quite—because I’m trying to think of when else you would want to do something like, when you would really want to dynamically dispatch on every type of TypedArray and this branch would be a problem. But actually polyfilling would be a case, but I don’t think this solves that case, introducing a new feature to the language won’t solve it for older browsers. Because it doesn't actually solve your use case, it really makes the motivation weak for me. + +JHD: Okay. + +USA: Next on the queue, we have Peter. + +JHD: I see an EOM. He’s clarifying that he’s talking about the whole proposal, not just the getter. + +PFC: So I’m not particularly opposed or in favor of this. I’m neutral. But I just want to point out that earlier this year, we heard that built-in functions are a big resource bottleneck for V8, and we removed a bunch of them from Temporal. I think some of them were genuinely convenient, but there was a slightly more inconvenient way do it that didn’t involve adding a built-in function. And I would say I find at least the withPlainDate methods more useful than get and setUint8Clamped. So, I don’t particularly have an opinion about this, but I’m kind of confused why it’s not too much to add these built-in functions? + +JHD: If there’s any implementers who have thoughts on that. + +USA: Yeah, there is new topic on the queue by YSV. + +YSV: I don’t have a lot of thoughts about this as an implementer. It’s a pretty trivial proposal, but it kind do have comes down to what are we actually solving here. Several people have raised that we shouldn’t just be filling in a Matrix. But because the stated use case wouldn’t actually be solved by this proposal, I would need to see a use case that would be solved by this proposal in order for it to move to Stage 2, because at Stage 2, we accept that the problem statement is real, and that it’s solving a real world problem. And that’s what we’re missing here. + +JHD: Okay. + +USA: Next on the queue, we have OMT. They also have a statement that says, "are there any examples of people publicly wanting this"? + +JHD: Not that I’m aware of. + +JHD: And then own the queue, we have Shu. + +SYG: To answer PFC’s question, I think it’s just, like, a slowly boiling pot kind of thing. Temporal is huge. It crossed the threshold by itself, so that’s the concerted effort to cut that down. If Temporal was, like, 500 proposals, this probably would not have happened until you crossed the threshold at that point, so, like, because this proposal is pretty trivial, I as the V8 delegate have decided to not really expend much effort in blocking it on, like, data size concerns. But, like, that’s just because it is very small in itself. If it were bigger, then—yeah, it’s a usual problem of, like, how do you draw the line for what is too much in aggregate for the language, and that’s difficult. But certainly huge proposals get that scrutiny. And I don’t think that’s a principled answer, but that is, I think, the real answer. + +JHD: So what I’m hearing is in particular, the comments about needing use cases that will be completely solved by the proposal so it’s not just filling in a matrix, that’s a fair point. I will have to put more thought into that. If the harder problem that I had run into other than not being able to dynamically dispatch was reconstructing the clamping logic, so it would be helpful for me to have the setter or some separate method that took in a number and spit out a clamped version of it without having to actually make a TypedArray and run it through there. So depending on the other use cases I find, I may suggest as an alternative path just adding a single, like, helper somewhere that gives me that clamping logic. But based on YSV’s feedback, in either case I would need to have use cases that it would solve that are compelling for the room. + +YSV: I don’t, I think everything looks reasonable. It’s small. I think what we’re just missing here is taking the approach of how are we solving real world problems, and once we have a clearly defined real world problem, then… + +USA: We also have a topic by PHE. + +PHE: Yeah, the topic is just a question, which JHD, you mentioned that clamping logic has to be reproduced. I haven’t reread the ex in a while. How complex is the clamping lodge snick. + +JHD: In the spec it’s not described with algorithm steps in a way that was easy for me to intuit, so the logic itself ends up not being complex at all. It’s more that it’s hard to get right and I would love to not have to think about it. + +PHE: Seems like a great project for editors to improve the spec on that, no? That would help a lot, right? + +JHD: I mean, that would be—that would help me reimplement it more easily. That doesn’t solve me not having to think about it, it just makes the journey slightly easier. + +SYG: There’s like a one-line for the delegates chat. + +JHD: Cool. Confirm that it matches what I had already come up with. + +USA: And that was it for the queue? + +JHD: Okay, well, then I guess I will just call it there, not requesting advancement, and I’ll either bring it back with more evidence or at some point, I will withdraw it completely. + +USA: All right. Thank you, JHD. Now, I believe we have some time to fill. + +### Speaker's Summary of Key Points + +In order to advance, champion needs to provide use cases that are completely solved by the proposal; polyfilling is not fully solved by it. Current proposal is get + set; based on future presented use cases, other alternatives may include “just set”, or “just a clamp helper” + +### Conclusion + +The proposal does not advance to Stage 2 and remains at Stage 1. + +## Discard (void) Bindings: Request for Stage 3 reviewers + +Presenter: Ron Buckton (RBN) + +- [proposal](https://github.com/tc39/proposal-discard-binding) + +CDA: I think Ron is looking for Stage 3 reviewers for discard void bind,. + +RBN: I realize that in, it was June, I did not—I failed to ask for reviewers at that time. I did have it on my slides for the presentation I gave yesterday, but we did not get to that bullet point because of the overflow or time box expired, so I’d like to quickly ask if anyone would be willing to volunteer as Stage 3 reviewer for discards proposal. + +USA: Let’s see. Let me know about any volunteers in the room. On the queue, we have ACE, who is happy to review. Can we get, please, one more person. Okay, OMT on the queue says it’s their first time, but they’d be happy to review. I can remind you, OMT, there’s a number of folks around who would be happy to help you out. But, yeah, great. And anybody else who would like to also volunteer can reach out to RBN. + +RBN: Thank you very much. + +USA: And that’s it. Would you put in a quick summary, Ron, in the notes. + +### Speaker's Summary of Key Points + +ACE and OMT will review discard bindings for Stage 3. + +## Smart Units stage 1 update + +Presenter: Ben Allen (BAN) + +- [proposal](https://github.com/ben-allen/proposal-smart-unit-preferences) +- [slides](https://docs.google.com/presentation/d/1WIinmXmkOvMvOUi4zAnH8qVNXPSE67uQEP6ptBXZLZM/edit#slide=id.p1) + +BAN: This is just a smart units update, though it’s—I don’t know, we’re at a point where we’re getting feedback where what to include is very, very valuable, so I’m very much looking forward to doing this presentation. All right. I’m having slight problems sharing my screen. Okay, here. Give me just one second. It doesn’t want to share the individual… You can do it. You can share the slide show. I’m largely talking to Teams here, but there we go. Will it actually share in slide show mode? + +BAN: Yes, fantastic. Okay. So you may recall that the last time we talked talked about smart units was in Helsinki, that was two plenaries ago, and it’s been put on the back burner and I’m ridiculously excited about. I think the people I talked about it with in TC39 and out in the world, this is a proposal and it’s really fun to talk about for technical reasons, sort of, but also primarily for sort of social reasons, figuring out what sort of things are important for us to include here. So the idea behind this proposal is, well, often when you’re localizing content, localizing measurements is really important because, well, the sort of, like, first step that we’re going to go a lot deeper into this, the first step is that the sort of intuitive one, if someone from anywhere else in the world is looking at temperatures on a weather site in the United States, they’re faced with the annoying problem of converting from Fahrenheit to Celsius in their head. Additionally, like, metric versus imperial, well, we kind of think about in terms of, oh, well, the United States uses this weird imperial system and it’s used sometimes in Great Britain and then the rest of the world uses metric. But as we’re going to get into, it’s a little bit more nuanced than that. And this proposal addresses those, like, interesting, cool nuanced things. + +BAN: Okay, I promise that every single time I present on smart units, let’s say until it gets to Stage 4, I will be including there flowchart. I enjoy it a great deal. Hopefully it—if not useful, at least entertaining and at least handy for all of you. I have heard from actual Canadians that this flowchart for how to measure things like a Canadian is in fact very simple, and if you want to properly be a Canadian, there’s many, many other thing us have to decide when considering whether or not to give a measurement in metric or imperial. So Canada is maybe sort of—it’s the poster child for this phenomenon, but it happens in a lot of places. A lot of places where there are differences in what measurement scale you use to represent a measurement. that go beyond just, like, oh, we use metric rather than imperial. My favourite line in this is "is it the pool’s temperature?". + +BAN: Okay, so as discussed at previous time, preferred measurement system and scale in a given locale converted based on, well, the locale, but it can also vary based on the type of thing measured, so it’s common in many locales for—more typically use metric use centimeters and millimeters for the metric system for measurements to give the height of people, using feet and inches. So there’s that special case. And the special case can occur based on the type of thing that’s being measured. But also bracketing off the sort of, like, immediately apparent metric versus imperial thing, measurement—the measurement scale used to represent a measurement can vary based on the value of the measurement itself. And this sort of—the most intuitive—the sort of most obvious place that this shows up is if you’re giving the distance to something by a road, if you’re giving the distance to something, you will tend to use different units based on that distance. So in the United States, if I’m giving the distance of something that’s, let’s say, less than a third of a mile away, I will tip equally or it’s customary to give that measurement in feet rather than miles. In shows up in other contexts too. To steal the box from the how to measure like a Canadian flowchart, it is not rare to switch measurement scales in day-to-day life based on—for math based on the mass itself. There’s also the context in which the measurement is used. These are some of the funniest ones in the how to measure like a Canadian flow chart. + +BAN: But so the things that we’re trying to support here are properly localizing measurements, to properly localize measurement values based on the type of thing that’s being measured, the value of the measurement itself, the -- in most cases, the context in which the measurement is used is note something we’re going to be concerned with here, as I’ll talk about at length. Basically, we don’t want a maximalist implementation of this. Sort of like more what’s the right word, parsimonious implementation is for reasons I’ll get into, is a very good idea. So we certainly aren’t concerned with—we have no particular interest in ore placing—in being able to do everything a Canadian can do. We’re not proposing that Intl support being a Canadian. That came out really weird. But in this case, we’re not going to be concerned with the more esoteric ones. The usages that we want to mark as localizable, as being localizable measurements, we’re going to use a very limited set. I’m going to the next slide. + +BAN: So to give some context on how we plan to do this. CLDR has some pretty extensive data on context and value or thing measured and value dependent locale measurement customs. This is from the file units.xml in common/supplemental. This is a relatively small file. I think this is, like, the—we’re only adding 8KB in we were including everything. But, so, yeah, this data exists in CLDR, and it’s fairly extensive or at least it’s extensive enough for our purposes. And the thing I want to really call out here, the reason why I want to lead talking about category lane usage person height, this one really well illustrates that this isn’t about purely, like, localization in terms of converting between metric and imperial. It’s not about just oh, there’s one special way of doing things in one specific locale or one specific region, and everything else does it differently. If we’re talking about the proper culturally respective of culture way to give the height of a person, it is most common for that person to give their height in centimeters, but in the United States, great Britain and a couple of former commonwealth countries, it’s so much more typical to give feet and inches, giving height in centimeters would be incomprehensible. And although it wouldn’t be incomprehensible to give a height in centimeters in any of these regions, it wouldn’t seem idiomatic. It is sort of of a localization fail. It would seem like it something from somewhere else rather than something from a local. So even if imperial didn’t exist, even if we could, like, snap our fingers and have the United States get with the rest of the world, we would still be concerned with things like this because of those differing preferences in regions that both use the metric system for certain types of measurement. + +BAN: Something I want to call out, and this is something that I’ll discuss at much greater length after the break, something you may notice is that, oh, yeah, often the cultural preferred way to give certain types of measurements is using mixed units. So not -- so if you’re giving the height of a person who is 3 feet tall, basically this is what is used for height of adults and this is what’s used for the length of infants and the height of small toddlers. You’re going to be using feet and inches instead of just inches. So in the measure proposal I’ll talk a lot about these units and I’m going to leave that. + +BAN: I promise to use the how to measure like a Canadian flowchart in every single presentation about smart units. I don’t promise to use this cartoon in every single presentation about smart units, but I’m tempted to. But I might be able to resist it. This joke is clearly written in American for reasons I’ll get into. But it can illustrate how incomprehensible measurements can be if they’re given in the—in wrong units. And especially, like, this third panel, which is kind of—because it shows that even if we’re not dealing with, oh, well, this locale uses imperial and this locale uses metric, even just inside regions that use imperial for most things or regions that use metric for most things, there’s still a need to localize measurements because often—and often and kind of often with the most commonly used sorts of measurements, there’s going to be a desire to put them in mixed units. + +BAN: The other thing that—the other reason I why like to use this cartoon when talking about the smart units proposal is, well, so far we’ve talked about properly localized—giving measurements in properly localized units and scales. But also precision is a key concern. The reason why I said oh, I’m pretty sure this was written by an American or at least someone from North America is that correct me if I’m wrong, it would be kind of weird in most places that give heights in centimeters to give two digits after the decimal point when giving your height. So in the case, I would argue that even though—I would argue that this is kind of not localized to anywhere, because even places that give heights in centimeters, well, typically don’t give heights with two digits after the decimal point. It’s far too precise. Okay, so what we want to do is we want to add to number format the ability to specify usage. I’ll get to the API, which is fairly simple, towards the end of this section—or this talk. + +BAN: We want to support a small subset of these usages that are locale specific variations in measurement scale, measurement unit used. We want to support a small subset of them by adding a usage option to number format. There are many reasons why we want to use a small subset. But one key one that came up in discussion both during the actual plenary session and just private discussions, one issue with this is discoverability. We don’t—there are many, many reasons why we don’t want to implement every usage that’s given in CLDR’s said XML, but one sort of primary one is, well, okay, say that your locale doesn’t have special handling for this type of usage. You actually have no reason to suspect that it’s going to be worthwhile to mark that measurement as being that type of usage. You’re simply not aware that, oh, in some locales, this is going to be a localization problem, so hear trying to focus on is a small set of units, because, well, the small for set of units is, the easier it is to find it. But also, this small set of units should be stuff that is used sort of as universally as possible. So, yeah, I’m going to in the next slides propose a set of units to set of usages to include them in this proposal. I have been composing this list, I’ve tried to err on the side of including too many, actually. I’ve included in this presentation more usages than I would necessarily, if I were just, like, willing this into existence myself, include. So one of the reasons—one of the sort of types of feedback that I’m most interested in is actually, player which ones of these can be removed? Which ones can be safely cut away, which ones should be cut away. That said, obviously, if there’s good arguments to include other sorts of usages, I would love to hear them. And they’re not on the slides, but there’s some sort of, like, hmm, just barely on the bubble usages that we considered and dismissed. But if there are other usages where, oh, actually, there are serious localization needs related to localizing these particular measurement and we should, you know—even know, they might want to prefer a small subset of parsimonious implementation or a parsimonious definition, there may be things that oh, yeah, we should add it. But my sort of, like, leaning is towards removing usages rather than adding them. + +BAN: Okay, so the units—the next few slides, and apologies for all of the pictures of text on this. The sort of first one that comes up, the first one we consider, the one that’s just, well, obviously in is something to include, is as in that cartoon, the heights of people. It’s a type of measurement. It’s a type of measurement that sort of comes up very frequently in people’s day-to-day lives, and it’s a type of measurement where it’s one of those ones where there’s variations other than just imperial versus metric. Most notably, most regions prefer to give heights in centimeters, but many regions prefer this mixed meters and centimeters unit. And, again, if you give—if you give someone from one of these regions content that gives people’s heights in centimeters, they’re going to understand it. But it’s going to be an annoyance, it’s going to seem not properly localized and seem that this content was not made with them in mind. + +BAN: Another one that is sort of a type of measurement that in so far as anything in this world is across all humans, across all type time, it’s a type of measurement 245 all humans across all time have been concerned about, which is the distance from one thing to another. If you’re travelling from one place to another, how far is it. And this is one where properly localizing it can be fairly complex, because it varies not just on the region, not just on the usage, but on the value itself. So as I said at the start, if I’m giving—so this GEQ attribute, in the case of threshold, so the CLDR has this set up is, okay, well, if you’re above this threshold, if you’re above one-half of a mile, then you give it in miles. If you’re above 100 meet and under half of a mile, it should be given in feet. And this is, well, feet, if you’re in the United States. Also, there’s this precision increment. So if you’re giving the distances from one place to another, if the places are very, very close, you’re likely to give a more precise measurement. If they’re thousands of miles away, you’re not typically concerned with the ones unit. There’s also—I have give myself permission to mention this second time even though it’s an interesting unit. It is actually—there are reasons to sometimes give these measurements in this—in this scale. I’ve—what I’ve read is that in Sweden, sometimes it’s used on some tax forms, but the Scandinavian mile is equal to 10 kilometers. It’s roughly equivalent to the Roman mile. On the English speaking Internet, when the Scandinavian mile is mentioned, it’s most often mentioned by angry and confused highers who think that they’re going on a 1 mile hike and actually going on a 10-kilometer hike. That’s one case where you can see the importance of giving measurements in a properly located localized scale. And in properly localized units, because although there are many cases where you might want to use the Scandinavian mile, you don’t want to be giving a Scandinavian mile to people outside Sweden, generally. + +BAN: Another type of measurement that we want to include or another usage that we want to include various speed measurements. CLDR’s fairly exhaustive, has fairly good data on how the measurement scale used for different types of speed varies from the one used for other sort of—other sort of speed. Most of them concern weather, which is something I’ll come back to. And, again, this is another one where it’s not just a matter of the United States or the United States and Great Britain versus the entire world. It is most common across regions to give one speed in kilometers per hour, but in many regions, including some very large regions with very large populations, it is more common the give it in meter per second, and unlike converts from measurements in centimeter to measurements in meters and centimeters, which is, you know, straightforward conversion that you can do in your head without thinking about it very much, converting from kilometres per hour to meters per second is a puzzle. If you’re given a value, if you’re looking up the wind speed if you go to a weather site, trying to figure out how windy it’s going to be, either because you’re flying a kite or you want to play a board game outside or you have a hurricane bearing down on you, you can get your—the wind speed in meters per second and expecting kilometres per hour, you’re not going to know what do with it. So this isn’t just a paper cut. This isn’t just a oh, this content seems like it’s not for they, it’s this content very much excludes me and I would very mump -- the thing I’d very much like to know about. + +BAN: Another one, that seems like the very, very, very high priority one to include is the weights of people. This one, this one is, how do I put it? I consider—let me go back. I considered this particular example (wind speed) as one of the better ones. This one (person weight) kind of just breaks down to metric versus imperial. That said, there is the extremely charming stone unit used for giving the weights of people in Great Britain. But that said, even though this is a less interesting, so this is—there’s less cultural complexity in this measurement, it’s also a measurement that you do in your day-to-day. It’s very important regular measurement, and you’re going to want that measurement. You’re going to want users of your content to receive that measurement in a way that they can understand and a way that they find welcoming. And the mass in general. Again, this is another one where it’s sort of metric versus imperial. It’s less interesting topic of conversation, but despite being a less interesting topic of conversation in practical terms, localizing measurements of mass, measurements of weight is, like—it’s very important for making your content properly localized. And the last one is just temperature. This is kind of the most boring one. It’s the one that breaks down most clearly into United States versus rest of world. + +BAN: So, yeah. Oh, yes, one more. CLDR also has a lot of information on volume. One of the ones where—this is one of the ones where it’s like, if I’m taking a hatchet to this and saying, we want two use a very, very small number of—allow for a very, very small number of usages to to be to be localized, then volume is one I would probably cut. But it’s also useful. I guess the reason why I feel like I’m biased against this is that even in the United States, for many uses, the old imperial system is falling out of favor. + +BAN: And finally, area. This is another one where it’s fairly timeless and universal. Everyone everywhere throughout all time has wanted to talk about area in some way. CLDR has data on the customary scales, the customary units for area in land. So farmland or what have you. And in area for floor plans. So, yes. So this sort of breaks down all the categories, measurements related to people’s bodies, these ones are the most high priority. If we end up only supporting these, it would be a fairly major win. Measurements related to weather. It’s something that comes up on a regular basis and something that a lot of people want to—it’s content that a lot of people are interested in and it’s content that’s important to localize correctly. Measurements related to travelling on roads. Again, incredibly common. Incredibly useful to get this properly localized. And just general temperature, volume, area, mass, sort of the fundamentals. + +BAN: Okay, so the proposed API is essentially unchanged from last time. As I said at the start, it’s fairly straightforward and fairly simple, and I realize I am leaving 7 minutes for questions so I’ll try to talk faster, so, yeah, it simply essentially we’re going to add a usage option to the options bag for number format indicating that, oh, this value I’m giving you isn’t just a bare value, it’s a measurement and more specifically the measurement of a person’s height. So drawing on that CLDR data, the person height will be represented in the appropriate units for that locale, be it centimeters, meters and centimetres, or feet and inches. The sort of intuitive names for the values of the usage option are, well, the ones from CLDR. If you look back, there’s another number of ones, those are the default is given and then the more specialized—the more specialized usages where that category are added. So for ones where the names are default, the value for usage should be the category. And when it’s not the default usage for that category, the name should be until there’s better names, the names should be value of the usage attribute from the CLDR data. + +BAN: The output and rounding will be determined by the locale, the usage, the input unit and the input value, CLDR fortunately has the data on precision. So this is what it could look like in use. + +BAN: Now, there is a big unanswered question, and that big unanswered question is one we’ll have to consider after lunch, so wait, a second, if we’re localizing measurements that means we’re going to convert values between measurement scales. Yeah. So fortunately, unit XML does contain the conversion constants for all of the relevant conversions. It is straightforward for our localization tool, for our thing that’s in 402, to draw on that data and do those conversions. However, and this is something that comes up very frequently with internationalization tools, if you have a tool that is for internationalization, but can be used as a pinch for things other than internationalization, users will do that, and there’s many horror stories of developers using internationalization tools to do things unrelated to internationalization, and finding that their sites, their applications work until they don’t. So the sort of—the big problem that’s blocked in is, oh, well, people are going to use it. So when we last presented it, one idea that came up was, well, what if we just provide outside of Intl, outside of 402, a measurement conversion tool that can do the things that the—specifically the things that internationalization, that smart units can so people can go use that instead of the internationalization tools. + +BAN: Also, I didn’t put this on the slide, so also representing those mixed—the measured proposal, which I’ll talk to you about after lunch, is also needed to do those mixed unit representations. So the measured proposal, which we’re going to get to, it’s an object to represent a measurement that can support mixed units, that can support specific precisions, and that can do the unit conversions that the Intl tool can use. One reason that we’ve split this out into its own thing, as we’ll discuss, making measurements proposal is inherently worthwhile because there’s demanding for this outside of the relatively narrow demands of smart units. I realize, that as predicted my slides took more time to get through than anticipated, but I’d like to go over to questions for relatively short questions, and we will almost certainly run out of time, but fortunately when we come back from break, there’s going to be the talk about the measurement proposal, which is kind of the second part of this talk. + +USA: All right. Let’s go to the quick—real quick. First we have a clarifying question by WH. + +WH: Early on in the presentation there was a slide with length measurements related to persons. There were two different ones: length in context of a “person” and length in context of a “person-height”. + +BAN: Yes. + +WH: I know what the length of a “person-height” is, but what does the length of a “person” measure? + +BAN: Thank you for catching that. Okay, so CLDR has data for person height that’s in the US, for example, feet and inches, and the second one for length—that’s person length. Person length is typically used—that’s, if I recall correctly, that is used to give the length of infants. So person height is, this is how we talk, this is the measurement scale we use for the height of this person you’re going to stand up, and just person length is the length of someone who is very small. And typically doesn’t stand up. + +WH: But you have that logic already encoded in length of a “person-height”: if it’s less than 3 feet, then it produces the result in pure inches. So what’s the difference? + +BAN: Oh, I’ll go look at it. Does somebody want to -- + +USA: Also there’s a response by SFC on the queue. If you’d like to go to that. + +SFC: Yeah, so the person—the person-height is a way for a CLDR to specify, for users to specify that they’re interested in person’s height as opposed to other measurements of a person, as opposed to waist size or wingspan or other ways you can measure a person. Person height specifies a height of a fern. Well person is if you don’t have the additional context, liking, this is the best measurement you can get there. When CLDR adding an extra subtag in the usage, it makes it more specific. + +BAN: SFC, thank you so much for that. + +USA: WH is on the queue again. + +WH: A question about the precedence of units on different lines. Is the first line that applies the one that’s chosen? Several of the presentation slides show examples, in the same countries, that you can measure mass in tons or kilograms or micrograms. How do you decide which line to use? + +BAN: So that’s a very good question. I’ll have—I believe the way the data is organized is that the entries farther down are the ones that are less commonly used. I’m not certain about that, though. I do not know if there’s CLDR data about, specific data about what is most commonly used, though. Like, I don’t—I kind of have to defer to people from different locales, for example, I’m not sure how common it is for the weights of people to be given in stone in the UK. So, yeah, that’s a good question, though. + +USA: All right. That was the entirety of the queue. I would say that we’re over time now. So unless you have to add something, let’s quickly summarize. And, yeah. + +### Speaker's Summary of Key Points + +I presented a list of usages that we’re considering having as part of smart units and received some very useful clarifying questions from WH on which units would be preferred for which usages in which regions when there’s multiple options. + +## Measure object stage 0 + +Presenter: Ben Allen (BAN) + +- [proposal](https://github.com/ben-allen/proposal-measure) +- [slides](https://docs.google.com/presentation/d/1yakKCsS3pR0T7eJcaJD2ZUzK6xG6FcoFnrmngkJUcEo) + +BAN: I apologize, some of this is fairly sketchy just because based on feedback related to decimal and number position, I found the need to make some last-minute changes to the proposed API. So what’s presented in the slides is fairly sketchy, and what’s currently on the explainer is inadequate for reasons that I’ll get into. But what we’re doing here is asking for Stage 0 for exploration, so hopefully that’s not too disastrous. + +BAN: Fantastic. Oh, before I get going, I have to apologize. I was confused by the questions at the end of the last session. So if you look at the specification for unit set XML, basically when provided with a unit, there’s an implicitly that GX attribute is one. Oh, how can I phrase this so that it’s clear? Basically, if someone has provided a value in stones, say, it would present it in stone. If there is no value given, it will be sort of like the previous one that’s in the specification. So, yeah, it’s not that they’re more common units or less common units. Those are the units used in that locale. And also for some units, there’s the GQ figure value that’s something other than one, so, for example, the road distance is the clearest one. If you’re given a value in miles, but that value is less than half a mile, the result will be in feet. + +BAN: So, okay, without further ado, I’m going to talk about the measure proposal. Again, a lot of this is influx. It’s influx based on things that I’ve talked about with the decimal people. But, so, this proposal is inspired by smart units. It’s inspired by the need to do unit conversions and smart units. And the need to avoid in so far as is possible people misusing the conversion tools provided in smart units to do things other than internationalization related conversions. That said, the reason why we split this out into its own proposal is that there’s—it’s applicable to a number of other contexts. It’s actually—it’s—even aside from the conversion stuff, it’s kind of generally useful because it makes it much easier to keep together units with values—values with the units that they’re denominated in, which is a very common task, and sometimes, like, with the current state of Intl tools, it’s fairly clumsy do that. Ideally a fully flushed out version of this proposal would extend number format method the take a measurement so that you don’t have to always set a separate currency property, for example, when formatting currencies. + +BAN: Okay, so as you may remember hearing from your introductory physics teacher when you were 13 or 14, measurements don’t make sense as value, it’s a value and a unit of measurement united, and as such, it’s relatively common the keep track of measurement units and objects, and we want to make this as easy as possible. This a fairly common problem, the value of a measurement and a measurement unit are fairly insufficient because you don’t want to imply that your measurement is more precise than it actually. So measurements will often need to come along with a precision. In most of Intl, the way that we handle precision is by using decimal strings. I don’t believe that’s done anywhere else in EMCAScript. So this would be sort of unit measurement, as I understand it, to use decimal strings to store values here, so this allows it being able to specify precision allows us to do that while still, like, storing values as just numbers. Okay, and, again, so this isn’t the most exciting need. I really like thinking about making it easier for users doing, I don’t know, physic simulations to keep together units and values. I like being able to more elegantly handle currency. And while I like it when people don’t abuse internationalization tools, I know that they will down the road get themselves in trouble if they do that. So there is a need to provide, if we’re going to have this smart units proposal go through, there will be a need to provide the conversion functionality that’s present in smart units outside of 402 so that people will use that instead of abusing the internationalization tools. I’m realizing I’m talking very negatively here, but also it is just objectively useful to be able do certain types of conversions. + +BAN: Okay, so one of the really key reasons why we split this up from smart units is it can be useful in contexts like MessageFormat. There are a lot of—pardon, there are a lot of contexts involved in internationalization and also outside internationalization where you need to keep track of currencies and the denominations together. It can be error prone and kind of, like, hackie, some of the things you have to do to properly—I’m more aware of the internationalization context, so to properly format currencies. But it goes beyond display. It goes beyond formatting. Yeah. Okay. So I’ve got a link here in the slide to a discussion that sort of, I think this discussion was the proximate cause of having the separate measure discussion. But, yeah, it would be very nice to be able to pass a value or currency code value in currency code pair into number format and receive properly formatted output, and outside of the context of displaying localized text, it’s just generally useful to be able to keep those two pieces of data, which don’t make sense without each other, in the same place. + +BAN: And as discussed at the end of the smart units proposal, it’s common for some measurements to be given in mixed units. And keeping track of mixed units is—it’s an imitation error, so we want to make this as easy as possible for users. As I said, I’ve been doing some last-minute revisions to the proposed API, so the proposed API, feel free to think of it as a placeholder. I’m not presenting this to advocate for a particular design. I’m presenting this to get a feeling of what design would be useful for people, and there’s a couple of decisions where it’s not immediately obvious what the right thing to do is, and that’s why I’m bringing this to plenary. + +BAN: Okay, so given that there needs—that unit exists for having an object that bundles together with the values for the additional information to make sense, we need an object to measure this. It needs to have more to the subsequent sides. It needs to have the value, it needs to have the unit, so the value could be a number, it could be a BigInt, it could be a decimal string potentially. We need the unit, a string, indicating what unit this measurement is in. And an optional value, the precision of the measurement. In terms of design, the reasons we’ll get into, it’s—my first thought is having the value in the precision in an options bag, for reasons we’ll get to in a second. But it’s more complicated than that because we need to handle mixed units. And mixed units, it’s an obvious need in the context of smart units. It also exists in other contexts. Again, the currency context is the one that sort of springs to mind most immediately. But I am absolutely certain that keeping track of mixed units is useful outside of either the context of smart units or localizing measurements and outside of the context of handling currencies. So the version that I was frantically scrapping right before the meeting, apologies for that, is a version with the following properties, a value, an optional minor value, a unit, a minor value unit, which these names are not necessarily good, and an optional precision. One—the thing that makes the most sense in the context of smart units would be to have, if we have a mixed unit—if we have a mixed unit measurement, to use the name of that mixed unit as the unit property. So unit could be feet, it could be inches, it could be meters, it could be centimeters and hyphenated feet and inches or hyphenated feet and centimeters. That might not be the most elegant way to do it for other contexts. We’ll get back to it. And you’ll see this assumes one question that I have is this assumes that if we have mixed units, there’s only going to be two units involved in the mixed unit. So feet and inches and meters and centimeters, but not kilometers and meters and centimeters. + +BAN: Measure would have a method `convert` that takes the unit and a precision and returns a new measure in that unit and at that precision. If for mixed units we were doing, say, an array of units, it would take in that array of units. And then for the original internationalization purpose, there would be additionally a locale convert method that takes a looks at how and usage. The usage could be optional and which returns a measure with the value converted to the locale appropriate usage appropriate scale. And the sort of secret that’s not a secret is that this method what are—like, this is from whence the entire proposal emerged. + +BAN: So the current, very sketchy design, given in the current version of explainer presupposed there are two units involved in mixed unit representations. It does that, because, well, it’s not a great reason, it does it because it covers all the reasons required for smart units and covers almost all situations involving currency. But off the top of my head, the only semi hypothetically commonly use case where there would be mixed units with three components I would can think of is old style United Kingdom money was the LSD version before they decimalized their currency. There are likely other more relevant cases, and in this case, the sort of most intuitive thing would be for measure to store values and units into arrays. One entry for each of the units for—and lack of a better name, one hinge that came up in discussion is a method just called splat that will return those arrays. + +BAN: A question beyond the API design question, question that gets to the more fundamental what should this thing be or what would it be appropriate for this thing to be, what conversions should we support? So my initial first impulse, but that impulse is sort of—that impulse is clearly coming from me thinking in the smart units context, is that, well, okay, this is a tool that we produce to avoid misuse of the internationalization tool. It’s something that makes sense in the language in the context of internationalization. JavaScript is kind of The lingua franca of the world, nearly as much as English is. And so internationalization is particularly important in the context of JavaScript. And being able to properly localize measurements is crucial for ensuring that content is localized in a way such that it is both comprehensible and sort of culturally appropriate for all users. But, okay, we will have to/get to add this measure object with conversion functionality for use outside of the context of internationalization. So if we’re thinking of this as, oh, okay, well, this is a thing that we’re sort of stingily adding because we know if we’re don’t, where we’re adding something else for people to use so they’re not tempted to use the internationalization thing for wrong purpose, well, then the conversions has to support precisely those conversions that are required for an internationalization tool. But there are other ways to approach it we can support all conversions that need to be made using CLDR, for example, so I’m looking forward to your feedback. The proposal as it stands is very, very Stage 0. We’re looking for the feedback from the committee on design decisions, but more than design decisions, I’m looking forward feedback on, well, what do people both in and outside of the internationalization context, what do people think something like this should be, what do they think something like this could be? So, yeah, I’m going to hand it over for questions now. I’ll also welcome questions about the earlier smart units proposal, earlier smart unit presentation, just because we were—we had a bit of a time crunch there. + +BAN: This is a move that I’ve done while teaching sometimes - I reserve the right to say something wrong so that people will dive in to give the right answer. I believe there’s an XKCD about that being an effective way to get the right answer on the Internet, I’m perfectly fine with discussion evolving away from me answering questions. Essentially I’m here to ask questions for you, and I don’t want to get in the way of your questions. + +SFC: Yeah, so I wanted to give some additional commentary on this proposal and the motivation and the high level roles for this proposal. So there’s been three different problems that we’ve seen that this proposal attempts to solve all three of those problems, so the first problem is the most concrete one and the one that BAN discussed in his presentation, which is the smart units proposal requires unit the conversion and by requiring—since it requires unit conversion, we would need to be able to, in order to prevent users—developers from, like, you know, basically abusing Intl API to get unit conversion behavior, we need give them a nicer way to do it so that they don’t abuse the Intl APIs. EAO talked about this principle multiple times before, and I talked about this principle before, so this general principle is if there’s any new type of functionality that is in an Intl API, it will be abused by web developers, and therefore, we need another way to let them do it. So that’s the first and most obvious one. But then there’s two more use cases here. One is, you know, working toward the message from my proposal. You know, which, you know, wants to be able to, you know, have a -- have a number with unit context in order to, you know, produce formatting for that unit. Now, you know, obviously the MessageFormat proposal is also still at Stage 1 and there’s still some open questions there, but just to get some context, that’s idea of how the fits in. And then I think the third and most important use case and why it’s I think for really important for us to discuss at this meeting and I would like to have additional time to—I would like to open this discussion with the group, I don’t think BAN was—BAN mentioned this a little bit in his presentation, I want to emphasize that, this is also a way for us to solve the, like -- what is the basically the input type to Intl formatter that carries a numeric value and precision that you’re formatting with and the currency. This is the like, canonical type for any numeric value with a dimension. One previous name we thought about this it’s a dimensioned value, right? I think YSV said yesterday something about maybe working more with HTML and see if there’s an HTML input to for this, and see if is that would be the foundational piece for that. This could be used to solve some of the use cases of the decimal proposal, right? Which is we want to be able to have a way to represent a currency value. If you want to represent, you know, $100.50 exactly, you could do that with this proposal that BAN is proposal right here for—he says it’s for Stage 0. I would kind of like it to be for Stage 1, but, yeah, this proposal that BAN is presenting here, so those are the three sort of big use cases. And I think all three of those are really important, and I think that, you know—this thing that makes me excited about this proposal is it sort of starts to lay is groundwork for how we look at numbers and numeric values in EMCAScript and perhaps more generally in the web platform. And, you know, I think this is a discussion that, you know, is really important for us to have and sort of have this big picture. I was talking with this yesterday with the decimal proposal. You know, we can take sort of narrow approaches here and there, but it’s an opportunity for us to have a big picture approach for us to solve this problem. + +BAN: SFC, thank you so much for highlighting the—you know, I alluded very briefly to having the format method of number format take one—take a measure. A lot of—I don’t know, from someone who has spent too much time at it in respect, having a thing—having the measurement or a dimension unit would make things clearer from the spec, writer perspective. That’s not the most important perspective, right, and being clear from the spec writer perspective is because it’s clear from the user perspective rather than passing in a bunch of sort of disparate options for—that are associated with the value, just being able to pass in the value with everything required to make the value make sense, it makes for cleaner code. + +WH: What is the story about measurement units which are not in base 10 such as small length measurements in the US? If I’m supposed to use a 7/32” drill and you give me that as the decimal fraction 0.21875”, it will be really weird. + +SFC: I can reply to that one, which is CLDR—there’s CLDR issues open about this topic. And basically this would be a formatting layer concern, right? So, like, after you pass your measure into Intl number format, then Intl number format would be like, oh, look, this unit in this locale, it’s more conventional to represent this as fraction, right? That can possibly be, like, you know, a display preference. So—but that’s definitely more of a formatting problem, you know, a display problem, because the actual quantity is the same. And, you know, I’ve had this sort of distinction before between what’s the quantity and what’s the quantity layer versus what’s a formatting layer. And this is definitely a problem more in the formatting layer. + +WH: Okay. We also have my next queued question, which is about the talk right before the break. The selection of units seemed fairly sparse. For example, for area units, you had hectares and such, but not square kilometers. In the US you had acres but not square miles. + +BAN: That is an artifact of the presentation rather than the data. If I recall correctly, and others correct me if I’m wrong, in the slide set, the sparse—what you’re describing as, like, the sparse set of options, those are for specific usages. There’s also a default usage for area that contains what you’re talking about. + +WH: What’s the threshold for switching from acres to square miles? Is it one square mile? Who gets to decide that? + +BAN: Yeah -- + +SFC: I can answer that, if you want. So the—so terms of acres and square miles that’s what the GEQ equal to parameter is there for. If it’s not explicitly printed it’s one. And UTS35 declares a fallback algorithm for—or declares the algorithm for how you select which one based on the magnitude of the input. + +WH: I understand how that works. My question is, who picks the thresholds? + +SFC: CLDR does. Mark Davis, as well as all the translators who give input into the CLDR. If we think that the thresholds are wrong, then we can open a CLDR issue and discuss that in that body. + +WH: My concern is not necessarily that the thresholds are wrong. It’s just that the thresholds might differ depending on your use case. + +SFC: Yeah, so I mean, that’s what the usages are for. So the thresholds can and do differ between usages and between locales. + +BAN: And correct me if I’m wrong, when they’re picking those thresholds, CLDR tends to privilege first-hand reports. + +WH: Okay. I’m also curious if any of these things measure area in football fields? + +BAN: They don’t measure in football fields and don’t measure in Rhode Islands and don’t measure in Luxembourgs either. + +WH: That’s good! + +YSV: Right. So SFC and I were talking about this earlier and also he just made the comment about that discussion where we were talking about the separation between data and view, and the discussion about this is in input, general input towards various types of formatters. I think that makes a lot of sense. I think something like that feels appropriate for something that lands into this sublibrary of the language, which is internationalization. A concern we’ve been having at Mozilla has been the amount of formatting and display work that’s been going on in Intl. So something like this makes a lot more sense to me, something that allows to do calculation or acting on values makes sense. And allowing to work with those values in a programmatic way really makes sense. This is part of a much big you are discussion we’re going to have directly with Intl, I want to say I do support this kind of direction about thinking about internationalization in the committee. + +SFC: Yeah, I put up another queue item, which is, you know, I think it would—you know, I know that BAN put this on the agenda as Stage 0, but I was sort of hoping that, you know—I would like to have some discussion from delegates other than just myself and BAN and WH and YSV about, you know, what we agree—what I think is the use—the most useful problem space, if you agree with the problem statement that I made earlier in the queue here or, you know, some of the things that Ben has put on his slides. You know, I would like to see us, you know, continue the basically actively engage. I mean, that’s what Stage 1 is for. So I guess, you know, I’m, you know, hoping that maybe we could achieve stage one on the problem statement, the problem statement being that wasn’t to design a—or we want to explore the space of having a type that is able to encapsulate both the numeric value as well as attributes of that numeric value, such as the currency, the unit, and/or the precision of that value, and then have that be fed into—and have that integrate with Intl formatting APIs, so, like, if that type of problem statement is, like, something that we think we should continue to spend time exploring as a committee, which I certainly hope it is, that seems like something we should ask for Stage 1 for. + +BAN: Yes. I’m regretting putting up for Stage 0 at this point just because I thought that there—like, I’m extremely pleased to find this that we’re hearing for having—having measure be something more than just internationalization. + +YSV: I mean, SFC just said you didn’t want to hear from me, but so we are discussing it and usually what I see is Stage 1 is as the barrier for Stage 1 is this is a problem area we need to explore further. Has this been presented already for Stage 1 and been rejected for some reason? + +BAN: It has not been. This has just been split off from smart units. + +SFC: Yeah, this is a—so when Ben gave the presentation in Helsinki, there was feedback from a couple of delegates that we should think about having us put a proposal, and that’s what this is. + +CDA: That’s it for the queue. + +BAN: All right. I’m going to ask an embarrassing presenter question. This was put on the agenda for Stage 0. There’s a sense I’m hearing from SFC and YSV that this is actually something more appropriate for Stage 1. And I’m guessing at this point it’s not possible to pivot and ask for stage Stage 1, but I feel compelled. + +CDA: It is possible. Somebody could block solely on missing the deadline, but if nobody wants to block on that, you can get stage one, you can ask for it. + +BAN: In that case, I have tremendous respect for anyone who blocks and I’m tremendously excited for people who don’t. I would like to ask for Stage 1 on measure. + +CDA: Do we have any support for Stage 1? Ross. + +RKG: Sure, I’ll just say what I wrote, is which is we’re already talking about it, we’re going to talk about it again. Let’s call it Stage 1. + +JHD: Yeah, echoing the same thing. If we’re ever going to hear about it again, it’s probably Stage 1. Let’s do that. + +YSV: Yeah I don’t have any objections to this, the exploration and the problem space, and I think it’s appropriate to ask for Stage 1. + +CDA: All right. Support for Stage 1. Are there any objections? Seeing nothing and nothing in the room, so congratulations, you have Stage 1. + +BAN: And I just want to say, thank you all so much. This -- + +BAN: I was just going to say, smart units and then, like, this and smart units, all the things that I’ve worked on with this committee, they’re ones I’m most personally excited in, so this—this is extremely happy making. Thank you all so much. + +CDA: Great. PHE is on the queue. Supports Stage 1. Interesting for embedded. Next we have Shane. + +SFC: Yeah, so I would like to, you know, continue to—just to, you know, iterate and evolve on this proposal. You know, between meetings I would like to see this proposal, you know, like -- I feel like this is kind of a big problem space, so I want to, you know—I feel like there’s a lot of ground to cover, so I wanted to maybe get a list of delegates who are maybe interested in at least say staying in the loop on these things, if we could, you know, you know, set up some additional time to discuss this, you know, also relates to decimal and some of the other numeric related proposals. So I just heard PHE’s possibly interested. The four people who actually made comments today, I assume are going to be included in that. I heard JHD make a peep, so I would like to have JHD involved with this, because, you know, I think that YSV have a lot of, you know, well informed opinions on numerics and how numeric types are handled. So, yeah, it looks like JMN is raising his hand in the queue. If there’s anyone else that I’m—I haven’t listed, you know, I would definitely like to sort of make a numerics -- numeric representation working group, if—well, not really, but maybe we can call it that. I’m not exactly sure what to call it. But I would definitely like to see this continue to make a lot of progress over the next couple of quarters. + +CDA: JMN, count me in. Did you want to speak? No. Okay. That’s it for the queue. And so that’s it for this topic. Thank you, BAN. Yes, thanks for the reminder, ACE. BAN, you can please dictate a summary for the notes. Key points. + +### Speaker's Summary of Key Points + +Plenary has decided to approve the measure proposal for Stage 1. We’ve also began to put together a numeric representation working group, which I think is a fine name, for making further motion on Measure. + +## import defer updates regarding evaluation triggers + +Presenter: Nicolò Ribaudo (NRO) + +- [proposal](https://github.com/tc39/proposal-defer-import-eval/) +- [slides](https://docs.google.com/presentation/d/1yFbqn6px5rIwAVjBbXgrYgql1L90tKPTWZq2A5D6f5Q/edit) + +NRO: Okay. Thank you. Okay, so this is our Stage 2.7 update for the deferred import evaluation. So we were running tests, and we’re in parallel working on experiment 8 implementation. But just to make sure that these steps are all correct. And spoiler, we found some bugs. So just to very quickly recap, I’m not going to go through how it works at this point, but, like, what’s important for this presentation is to remember when a deferred module is evaluated. It’s not deferred when you evaluate. That’s the whole point. It’s not evaluate when you reference the deferredness space. It’s only deferred when you rate it. Specifically, when you try to read a string property, it always triggers evaluation regardless of whether that string is an actual export of the module or not. And reason in modules when accessing unknown export names is that tools or platforms that can skip actually in the module list don’t need to pre-collect a list of exported names, and also so that the behavior evaluation—you can have a function that says with this property access to the relation and return. When you get a symbol, that never triggers evaluation, because modules cannot export symbol key properties, so we already know that the property will not exist anyway. This can be determined without PTM concept of modules, and this means, for example, you can get the symbol toStringTag, symbol is present on to object you have or not. For the other object operation, it depends on whether the internal can get or not. So by implementing this, we found that [[get]] is called less frequently than what is assumed when writing the spec text. + +NRO: So, for example, [[GetOwnProperty]] we first check if it’s an unexplored name and then call get. So it behaves differently. And this means that actually only evaluating get, we don’t get the benefits because we can just use another of the object methods and we’ll assume it to nod list of expert names, so the processed solution here is always check deferred name of exported names. This means that also if, for example, trying to reflect on keys with the relation, object keys are already evaluation because it calls get internally, but there are other ways to get list of keys from an object that do not call get internally, so it would be to get it covered. So that the rules becomes whenever you try to query properties about the object, if this properties depend on the contents of the module, the module would be evaluated. The end second problem we found was that import.defer, which is the dynamic form proposal, does not truly work as we were expecting it to work. So if you have this example here, when would you expect evaluation to happen? Well, it happens there, in the import of defer call. And not when you actually write something await space, and this is, like, completely against the proposal. There’s not deferred execution going on here. And why is this happening? Well, if we try to just, like think of how import defer works and in the code, import defer, we create a promise, we load the module, load the premises and then resolve the premise with the deferred namespace object. And how does resolve work? Well, resolves check if the given value has a property and if it is thenable, otherwise it actually resolves the promise with the values it’s given. And it’s this check that’s causing problems. We’re performing of get of this `then` property on the namespace object, and this always triggering evaluation of the module before the promise to import the deferred is resolved. + +NRO: What can we do about this? I’m going to list some solution here. And to be clear, I dislike all of them. But let’s go ahead and let’s see. So the first solution, like, that we could apply is that [[Get]] on the evaluation for known names, and these is probably good for some platforms, but it means that both tools and platforms that would like to skip actually a lot of the module will collect data, and so it makes implementation more difficult. And also there would be no way to trigger modules that have no exports. Maybe we could make import defer if a model is linking ever, but that will be—that was suggested in the past and was considered to be possibly surprising. + +NRO: Okay. Then another solution is that [[Get]] specifically `then` the property on the trigger evaluation if there’s an export, which is the same, like, unknown as both. There is just moment that you collect the module. But it could be okay because you’re just bound to one property and not to however many exports the model has. A third solution is that [[Get]] and `then` +never triggers evaluation. So getting `then` from a namespace, from a deferred namespace would return undefined, but then if it triggers evaluation through some other way, like through some other property, then dot dev property would show the value of the then export. + +NRO: Another solution considered, and that we was, like, add some sort of symbol dot thenable to the defer model namespace and the promise protocol would check the symbol of thenable before checking if there’s a then property. This was proposed in the past for non-deferred module namespaces. But it was rejected because it was not web compatible. And at the time, it was already shipping. + +NRO: And there are other solution if there’s an object that has a property that points to the deferred namespace, so that the—so that even if the promise proposal tries to go, nothing happens. Or maybe we just don’t do `import.defer()` at all. This syntax that we went for with import defer was designed to be—to be compatible across this different, like, module loading layers, so it would actually stick with that and make sure there’s a symbol that the developers have to learn what the syntax is, if you have syntax error, like, per phase, and lastly, actually, this animation was as a result of that. We could also just special case defer to model namespaces in the promised protocol and just before checking them, check the protocol namespace, and in that case, just resolve the promise to the top. + +NRO: So originally, well, until yesterday, my preferred solution was C. So accessing the `then` does not trigger evaluation, it goes up as defined and when the module is evaluated. And my second preference would be to not have `import.defer`. Keeping in mind that it’s not something we can then add later, like, as a way to have more time to solve this problem, because even if we remove `import.defer`, we have to decide how deferred namespaces work now. So this is—it’s not something we could then add back in the future. So as I mentioned, I like none of these. + +NRO: And yesterday when I was talking with ACE, ACE suggested some ideas I actually liked. So this is an idea that was presented in the previous slides, but here I just added this yesterday. I understand if you need more time to decide whether you like it or not. So I—I will be also happy with just coming back for consensus next time. So ACE said was to just completely hide `then` +exports from module namespaces, from deferred module namespaces, so trying to read them always triggers undefined regardless of whether the module has a then export or not, regardless of whether deferred namespace has a trigger or. No it does not meet the trigger evaluation because the result is not the same as when accessing single keys. The object of keys would also never contain `then`, because there is just, like, completely hidden from the object, object appearing on the property scripter of `then` would always return `undefined`. This would be like an actual object with no `then` property regardless of the module exports. And, well, what if people have a module that has a `then` export? Well, exporting `then` is from a module is, in general, probably not what you want, because it interacts directly with dynamic import. But if you’re really doing it to import one of those models and you need the `then` export, then you could just wrap a module and export as analysis. So this together with exports is now my preferred solution. That I will be very happy if anybody had other great ideas to solve this problem. To the queue. + +YSV: So just thinking about why you would want to use a deferred module. You’re trying to pay half of the cost of module ex-substantiation and evaluation up front. So you pay half and then you pay the second half later. It feels like using dynamic import for something like this, I’m wondering, like, I’m wondering how much you’re saving there, because already you’ve deferred the cost of loading and evaluation until a later point. You’re already in an asynchronous context, if I understand correctly, it’s still going to be async, so you’re not getting the synchronous behavior, which I know is something that’s been requested, back when we doing dynamic import to begin with, they were asking for a synchronous import. As far as I understand, we’re not getting that here either. So it feels like we’re not getting much of benefit from `import.defer` or maybe I’m missing the benefit here. + +NRO: The use case is indeed much smaller than the static import. The only use case I could, like—I so mentioned using `import.defer` is to have, like, a conditional import of the top levels of module, so you could doing like `if(node)` and import this and import that and after the module, you—while still then deferring the relation of the module two later when needed in some other synchronous path. You still get the—some benefit from being able to do a synchronous relation, but, yes, it’s true that the use case is much smaller here. + +YSV: Okay, so in this case, what makes me uncomfortable about, like, hiding `then` or shadowing it is then we are changing how users’ modules are going to potentially work. My preference in that case is actually to not do import defer. Unless there’s a really strong reason why we might want to have it. + +NRO: Okay, thank you. SYG. + +SYG: Yeah, I think I agree you’ll yeah. I would prefer removing the problematic API entirely rather than using one after huff censorship attempts, which if nothing else, I have trouble gaming out the effects of these one-off censorship idea versus disallowing dynamic import defer. + +NRO: Okay, thank you. ACE. + +ACE: So, yeah, we initially were looking at this. My go-to was let’s just remove this from proposal because the use cases aren’t so strong. My concern is while we don’t have loads of use case today, that doesn’t mean we won’t have use cases in the future. There’s lots of things we’re doing now that JavaScript developers weren’t doing in the past. And to a certain extent, we’re closing this off from doing it in the future, we can to it in the future, but we’re making it harder for us to do in the future, so it took me 24 hours to realize, that but I want to make sure that, like, maybe we do say no this, but we should be really—I don’t think we should just think no use cases today means no use case ever, if we’re closing the door for this. + +ACE: The door isn’t completely closed, so I can still be convinced. + +SYG: Say more about why it’s harder—why we close the door more than usual by not doing it right now? + +NRO: I can answer that. So the reason is that we need to decide now how deferred import objects work, even if we do not give an `import.defer` API. So let’s say we do not do `import.defer` and do no special, like, hiding of `then`. Then today if you get a deferred module next case from a static import and you pass it to promise.resolve, that would trigger evaluation, and in that case, it’s perfectly fine to trigger evaluation, and it’s also how modules work. If we call them. However, this means that if the future we decide to actually let’s introduce import.defer, then behavior of the object returned to `import.defer` is already defined, and we cannot change it and so `import.defer` would already trigger evaluation. And the only way around this is to use option F on screen. Which is, like, not a very nice looking option, but, like, that’s what it means if we not doing `import.defer` now doesn’t mean that we are deferring the problem, just means we’re saying we’re not going to have import.defer. + +YSV: okay, I have a response to this. Import—the dynamic import is already a deferred import, but it’s deferring all the work. So by—so imagine the use case where you—you want to save the up front work of loading the file, and then executing it. So you put—in this case, you would be branching in two ways. If you—okay, so the idea of static import is you’ve got a long running static defer on import, sorry, oh, you I got words. Static import with defer allows you to write long running application that has infrequently used libraries, loaded statically, but you pay the cost for those infrequently used libraries later on, and these infrequently used libraries are ubiquitous and difficult to refactor out of the code base. That’s the goal there. It’s a transparent hint about optimization to the compiler. But if you want to defer the loading of the file, then the right tool for that is dynamic import, because otherwise you are putting an if statement at the top level of your—of your file, and importing either A or B, but you are going to need to branch later in order to import that—in order to execute that code, because there’s going to be a fork—you’re going to have forking twice. So I don’t see why you wouldn’t in fact just prefer using the regular dynamic import in this case. + +NRO: Okay. Yeah. Thank you. ACE. + +ACE: Sorry. One thing we’re seeing at Bloomberg is you sometimes do have this halfway house. If we click on a button, we say for that case, we’ll go all the way completely to dynamic import, it’s okay, we can make that an async context it doesn’t have this whole refactoring version, that this static proposal is primarily motivated by the function calling, sometimes, though, it’s another important thing here is when someone clicks a button, how quickly do they get a response. There’s a lot of web metrics around that. If you delay too much to user interaction, you get fast app start-up, but now those user interactions can sometimes suffer. Because sometimes you want this halfway where it’s like, if you don’t want to do it immediately a app start-up statically, and you don’t want to completely defer it until the button click. So especially if that button click is going to be potentially going off to a network, doing something that could take a long time, so what we’ll do is we’ll just do the IO, load the files, do that the background, but still don’t do evaluation, so when a person clicks a button, we’ll do that last bit of work, which is deterministic and not IO. Again, this is in our primary use case, but I do think there can be use cases for this, which is why I’m nervous to completely close off the door. And I don’t think we are completely closing off the door. The solution would be in the future if do you a dynamic import defer, we have to introduce a third type of namespace, so we’d have three different types of namespace objects. So it’s in the completely closed off, it’s just that’s the kind of world we would head towards. + +YSV: Okay, so if I understand correctly, the button is being dynamically generated in the code which is why you don’t know if you’ll have the button behavior needed up front? why are you branching to load dynamically? What decision are made? + +ACE: I don’t follow. + +YSV: You have a button and you wanted to take a certain action, right? Is the action the actual import? + +ACE: So the—they click the button. When we click the button, that’s going to go and run some code to then decide how is that pop-up going to be displayed. So we have to fetch all the code for that, like, dialogue and all the things that dialogue is going to display, and the thing is going to display Monaco, so it’s a Monaco JavaScript and it’s going to go fetch Monaco. + +YSV: the component that you dynamically import, can you not use a defer on the static imports of that file to achieve the same thing? + +ACE: Yes, potentially. + +YSV: Okay. + +ACE: I’m saying, again, earlier, because I don’t have a completely concrete use case today, I think I can get close enough use case I can imagine these things coming up. + +YSV: Okay. + +ACE: In general, I don’t use dynamic imports a lot. But I’m saying I can imagine -- + +YSV: Sorry. + +ACE: And they come up in surprises cases, like sometimes the thing you’re switching is big because you have an AB test and you’re like, the string—it was static, but now next week, we want to AB test and load, so now the specifier is not static and we want to load—these things come up. So I think it could come up in the future. But I’m not going to try and claim this is a very strong case. + +YSV: The reason I was asking -- + +NRO: Sorry. Can I interrupt a second. Given that use case hypothetical, can I go through the queue and come back to this. + +JRL: We have eight minutes left and four more items. +[ WRITER SWITCHOVER ] + +WH: I’m trying to understand why people use thenable modules. Some do. Why? + +NRO: It was to support last. And that was in the conversations when we were discussing this problem for the dynamic part before that existed. But then it only worked with the dynamic import and not the static import. But, yes, it’s generally considered to be bad practice. + +WH: But some people do it. So if we hide `then` for them, that will break them? + +NRO: No. Because deferred cases have separate from the mixed cases and we can hide it only from the deferred namespace objects and not for the old plastic space object. Not break existing code. It means that those people would be—those modules not expecting. + +WH: If they did do dynamic import defer, they would get everything except the `then` method? + +NRO: Yes. + +WH: Okay, thank you. + +MM: KKL cannot unmute. I will read for him. Folks at Agoric have a mild preference for lazy exports name space only evaluating for existing exports access. And the module exports then, it is consumed with import defer keeping both pieces. + +NRO: Okay, thank you. + +JRL: That was the end of the message. Next up is SFC. + +SFC: Yeah, I will just note again that, you know, I think especially when we get to—when we talk about WebAssembly and, you know, getting to the point of being able to load WebAssembly modules, like, this having a dynamic defer seems like it would be potentially useful in that use case because like especially if the substantiation of the WebAssembly module is expensive might want to consider the loading of the source and then the substantiation to be two separate operations without having full dynamic import that makes the whole thing async. I think there is the use case there but I understand what this is proposing. It would be nice, I think, to be able to support this if possible if not now but not to completely box ourselves out of it. + +NRO: Why Wasm is different from JavaScript in this case given that static import defer? + +JRL: YSV has a response. + +YSV: I understood that Wasm already splits the two phases and you do it manually but maybe I’m misunderstanding. + +NRO: Two different ways with Wasm and let’s say always the final view that’s not necessary what is implemented today and either import source of the web module and then you can like substantiated or once the full static import of WebAssembly exists, so you can have import from the file, then at this point, you can use import for the module. I think. It’s like what SFC was saying here is—maybe you want to trail Wasm binary dynamically and substantiated later to import defer. + +JRL: And the queue got a little screwed up here. I’m imagining SYG is trying to respond to the current Wasm discussion and not GB’s comment. You’re up. + +SYG: I think evaluation phase of Wasm via Wasm imports with the machinery is still aways away. I wouldn’t consider that to be a very compelling use case right now. Source phase Wasm imports are coming but not evaluation phase. + +GB: Thanks JRL. Yeah, I mean, if I can speak briefly to that. We definitely did consider integration with defer in the WebAssembly as integration to respond to the comment on the comment that actually get to my main point which is firstly I agree with what others have said in that dynamic preloading is useful with `import.defer` dynamically. My preference actually looking at it today, today it’s looking like C to me, actually, which is that the behavior for a deferred name space is that the then function is like in an undefined state, while the module is it in its deferred state, so that accessing the function performs execution seems wrong. And maybe this aligns with KKL comment as well. Maybe KKL can comment if this aligns with his thinking for C. So ideally we would want it that like the then access doesn’t trigger while inside the promise machinery and dot defer resolve then you have the machinery working again but C seems the best compromise and not trigger exec for the then name but still make it available and still be able to give it a value later on. So I like the way you presented that NRO. Interesting to hear if there’s any problems with that as well. I also like ACE’s idea for what is it is worth. + +JRL: So I have a slightly different perspective on this than what GB just said. In the promise machinery the reason that when you `promise.resolve` with the thenable the reason that we have to do access to then and invoke the then is because we have to adopt the state of whatever it is. If we just skip that step of calling `promise.resolve` and fulfill the promise object directly without going into the nested adoption phase, we can avoid censoring then in module, you can export whatever you want. We won’t do the dynamic promised behavior for the namespaces. Something like that seems better than censoring and preventing a then export from even existing. + +NRO: So that would be—I guess `.then` to the promise, it changes to trigger – + +JRL: Somewhere in the module resolution there is a thing that wraps it in a promise and either calling promise resolve or choosing the promise capability and calling the resolve function on that. All that internally does the same stuff and tries to adopt the state of the internal thing. We have special promise handling of that thenables instead of trying to adopt the state of the value you’re resolving with just fulfill directly, turn the promise to the fulfilled state. Don’t try to adopt. We can skip this evaluation entirely. + +JHD: I just wanted to add that I may have not thought that through JRL’s suggestion clearly. Shouldn’t be possible to have the resolve promise or fulfilled promise if you have a value with a `.then` on it. We shouldn’t change that. + +NRO: I believe if you have a then that’s undefined, you’ll get an object with a then that’s undefined. + +JHD: Okay. I guess what I meant is—like intended to mean I don’t think it should be possible to have something in a promise that if you return it can change into something else. But you’re right, I had forgotten about the horror of Proxies and they can already sort of do that. + +JRL: We have three topics left on the queue and we’re now a minute over our time box. Okay to go on? + +NRO: JRL was not suggesting that. JRL was suggesting special casing the first promise creation from import from that number import, the then property. Just the first axis. And, yes, so I see the rest in the queue. Deferred use case. Complete use case from SFC, but I believe that this case is perfectly solved because if you have a Wasm instance, you don’t need `import.defer` and import the source and then later go with the send substantiate. So I heard Agoric opinion for actually doing ACE’s. But given the general discussion, I would prefer to ask if anybody has any like call it trunk position to just dropping it for defer. + +JWK: I support A but it does increase the difficulty for some implementations that needed to cross-file information to transform a file. The next choice is to drop `import.defer` and if we really have a need for that, we can add that in the future. + +NRO: Thank you. I’m happy to—ACE is okay with dropping it. So let’s go ahead to option G. Let me just go back to—YSV said she would like to discuss offline. Clarify import defer or prefer waiting to say – + +YSV: So I was pressing about the use case. I really want to understand what the worry is there. So maybe we can just sit down with a couple of examples and see if a regular dynamic import plus—a regular dynamic import importing a deferred module would fulfill the goals of import.defer. If not that, might actually help us understand what the use case is. + +NRO: Okay. Sounds good. I can expect the same question again next time. Also go back to this. Is there consensus for also making the change there that is instead of just sometimes with the export names do it consistently? I heard that Agoric prefers not doing this. I see this as very useful for different implementations. So can we proceed with this before checking list of export names? + +JRL: So can we get consensus on making this change? Is that what you’re asking for? + +NRO: Yes. + +JRL: Does anyone explicitly support this? Does anyone explicitly object to this? + +SYG: I don’t think I fully understand what is being asked for. + +JRL: This change on the slide. + +NRO: So right now SYG, the evaluation happens either with perform [[get]] which means there are still ways to actually get proper defer modules without emulating it. It is just not to the standard path. Like getting property or reflect the keys. So the question is, the consensus for in the modules tried to query the modules from the object which means changing the export trigger from either calling get or the list of expert names? + +SYG: Make sure that I understand, the change is what you have as steps 1 and 2 is the current thing. And the proposal is to not do Step 1 so that you always evaluate on +[[get]]; is that right? + +NRO: No. The proposal is to add before Step 1 evaluate the module because we want to define and don’t want to create in the script or it is not there. + +JRL: As a first step before the current steps right now, as zero step, perform evaluate deferred module? So on the slide insert a step zero and evaluates the deferred module? + +SYG: Okay. + +JRL: So is there any explicit objection to this change? + +GB: Is there an effect here on observability if you’re trying to observe the deferred object that you inadvertently executed by mistake? + +NRO: What do you mean? + +GB: So like if you console log it, would that cause execution? + +NRO: It depends on the implementation. It would not in browsers, it would in node because node performs on the properties. So it’s – + +GB: Is there some kind of that we could have for log in tools or something like that? + +NRO: If they are maintained just switch it to internals. So given the confusion in the queue, I will come back with this next time and talk to people offline. Because we are passed the time box. + +JRL: Okay. So there’s no consensus on this today. Would you like to dictate a final comment? What do we call this conclusion? Can’t think words. + +CDA: Did we get to KKL’s comment on the queue? Is that why you have the microphone. + +MM: KKL says still not clear on the change. And on chat, I will just report that when JRL said we don’t have consensus, KKL gave a thumbs up that we don’t yet have consensus. + +### Speaker's Summary of Key Points + +NRO: So there have been two problems found with the proposal. One is that get behaves differently from the various other object methods and the other one is that the dynamic `import.defer` from does not work. We focused mostly on the second problem. There have been presented different use cases for going with different. It seems like the one with the most sport was `import.defer` and we’re removing `import.defer` and we need to follow up. We also need to follow up offline with solution to the first problem, because—well, we need to follow up offline with some people. + +## Continuation: TG4: Source Map Specification, 2024 edition approval + +Presenter: Jon Kuperman (JKP) + +- [slides](https://docs.google.com/presentation/d/147UbvCUAoW07vkQjYKJrf8SqqmCnmicvVBwEu__lOu4/edit?usp=sharing) + +JKP: So I had presented on the morning of the first day for TG4 source maps looking for approval on the branch of the 2024 work. I think I got a few approvals and then I also got some folks that wanted to talk more about it, especially MM. I’m pretty sure I met with everybody that wanted to talk about it. We had some really productive conversations. MM had time to look through the spec. + +JKP: Just for context for those that haven’t, we have the magic source mapping URL comment that leads toilings that DevTools will source the map and we wanted to make the source say in order to get the comment you had to use the parser and parse the JavaScript and find an actual comment that linked to that and therefore you knew safely what you were fetching. We got feedback from a lot of IDEs that they need something faster than that and prefer somehow to use a regular expression or something quick to get the comment out. So MM had talked to us about potential security concerns especially with parsing and finding Temporal literal and thinking potentially it was a comment and fetching that source map. So we had talked JRL and MM and I and JRL come up with an idea for a solution which would say that we not only need to start at the bottom and look up for lines that start with the magic comment, the slash slash pound but we also if we ever encounter a back tick, we need to bail out. Therefore, we would not pick up anything inside of Temporal literal because on the final line of it we would just bail out. I would like to open up to discussion but I understand that JRL and MM are working together on this and feeling good about this discussion. + +MM: Just to elaborate on the nature of the threat so people understand what is motivating this is—so the way the spec describes it is parser-based recognition and non parser-based recognition. If one party submits a program, then an intermediate party wanting to know that the program fed to DevTtools will not cause a URL to be dereferenced on to the network for any number of reasons that might want to be filtering and suppressing that. If it’s doing a parser based recognition and does not see such a URL directive, you want it to be the case that the resulting thing is therefore necessarily safe with regard to this criteria if it then goes into the DeTtool that does the non parser based recognition. The rule just outlined succeeds at nonrecognizing a source URL via a non parser-based recognition if the parse-based recognition does not see the URL. + +JKP Thank you so much. I think that’s great. So I suppose what I would like to do is I would like to ask for consensus approval on the spec. We can go to the notes. What I’m currently asking for is conditional on updating on non-parser based rules to bail out when they detect a [???] + +CDA: On the queue, kind of missing context myself but OMT says like that more than REGX and end of message. + +JKP: Specifically bailing out early or the parser part? Sorry. + +OMT: As opposed to parsing it or REGX I like to have the specifically designed thing at least I think that’s much more helpful. + +JKP: Great, yeah, we did already have a little bit of this with the bottom-up approach but I think this hardens it and is a great vector we can prevent, so – + +NRO: I don’t see from the parser perspective. So we should be clear on what the goal of this discussion of the spec. Because like when we—the original did not define exactly how to get this comments. So when it is—the first approach is say go to the list of topics and find what corresponds to the single line comment. And however, that did not find a consensus with the TG4 and the reason for that is that tools were just not—we link some tools and not willing to move to the parser-based approach. What we did is to say, well, let’s just include both behaviors that have been used in real world here so that if you, for example, want to check with what is happening like let’s say—let’s say for example it’s a security problem you want to check, the comments behind the spec and it means that the spec reflects permutations you can just—that was the goal of that heading spec. The goal is not prescribe changes. We did prescribe the changes if possible. Here we’re talking about asking changes to change and we kept that based approach. Process-wise I don’t think we can change that in this meeting. Like, we either this is as good as it is and we can iterate after publishing or we need to take a break here. We need to have the spec ready by the end of the day or by tomorrow. And we just cannot do this change without going back to TG4 and consulting all involved parties. I also have topic on the merits of the proposed solution. + +JKP: Can we talk about the first thing. I guess I’m a little bit confused there because I don’t think this is a section of the specification where we’re discussing requirements. This is a section where we are providing potential implementers with methods for achieving these goals. Like, you know, if it were like in the section of the spec where we say, you know, what a source mapping URL comment is or isn’t, I would sort of understand that more. This is like saying basically here is a helpful program that would help you bind the source mapping URLs and making the helpful program more correct doesn’t seem to be changing anything about – making the program more correct wouldn’t invalidate what it is doing and it is a method that we are offering to extract these, right? + +MM: I’m sorry. As I understand the point of the normative spec is that if what VS code is currently doing is vulnerable to this attack that the—that we would be defining the normative spec so that what VS code is doing is outside the normative spec. IE, not conforming. Okay. + +JKP: So maybe this is just me not understanding the process here. NRO, if I understand what you’re saying is that like—are you saying that any change that would change the current following or not following, like, compliance level of any customer should not be made this late? Is that what you’re saying? + +NRO: No. I’m saying that any change that’s—right, I’m saying any change that is not needs to be brought up to TG4. + +NRO: Okay. So next topic. I don’t think the solution works. It make it weak because the language says only support multiline and support multiline string literals and put at the end of the line and check for quotes and you also need to check for closed multi-line comments because you have a multi-line comment that have a single line so it’s not pure comment. + +MM: So I didn’t understand the thing about quote. Back quote allows multi-line strings. + +JKP: Also that is—double quote back slash new line. + +MM: Okay. Right. You’re right. All the quotes—I thought about the multi-line commenting and let me tell you why I thought it was not an issue. Maybe I’m missing something. + +NRO: The case about multi-line comment is when you have a comment containing a source followed by a multi-line comment that inside contains something that looks like another comment that contain the source map URL. + +MM: So if you—I was thinking that since it’s in a comment either way, if you accept the rule that even in a parse-based recognition, if you have within a multi-line comment a line that by itself looks like a single line comment, you can consider that a valid directive. In which case, you don’t have to distinguish whether the single line comment is also within or not a multi-line comment. But I should have made that explicit. I really hadn’t thought that through very deeply. + +NRO: Yeah. Okay, no, that’s not the case. If you have a multi-line comment with some random words inside and then something that looks like a source map URL, that doesn’t count. The comment must also include the source map URL. + +MM: You’re right, the multi-line comment is certainly irregular and the rule that I was kind of vaguely thinking about makes things—makes the parsing-based recognition complex and irregular. You’re right about all of the costs. Nevertheless, I think the danger here is real. And I think that adding up all of the closed delimiters of multi-line things that you identified, I think it’s still a reasonable set of prohibitions. + +NRO: Can we go back a second to the nature of the security concern. + +MM: Yeah. So a DevTool fed this thing is going to defense the URL and go out to the network. I think JHD in the discussion mentioned telemetry and someone wanting to know who is using their stuff. So obviously piece of code when executed can go out on to the network if it has the privilege to do so but several answers to that and I did see the thing about the dereferencing of URL uses the privileges that the code would have if it were trying to dereference a URL in the same context which is a nice alignment. I appreciated that. But tooling will often dereference those source mapping URLs in context where they’re not about to execute the program, and also in a hardened JavaScript context, for example, the code when executed might not have access to the network, so the threat is that somebody builds a parser-based recognizer as an intermediate filter that says, okay, is this thing a safe thing to feed to downstream tools such that downstream tools are not going to go out on to the network and reveal that we’re taking a look at this code or more narrowly maybe it only wants to allow a source map URL to a particular restrictive range of network and particular origin or whatever, so it’s filtering for that and what happens via these non-parser-based recognition rules is somebody sneaks in a URL that communicates to a different origin than the parser-based recognition tool is filtering for. So I mean, in general, this is one of the things about syntactic ambiguity and why the HTML-like comment in the JavaScript specification is such a disaster from a security point of view is because tools will look at things under assumptions about how to parse and then make inferences about what is and isn’t safe and pass it on assuming that the decisions they’ve made are compatible, are at least conservatively safe predictors of what the downstream things might do. I’ve been bitten by this repeatedly. I’ve been bitten by this on HTML-like comments and JavaScript specifically. This is just—you know, this triggers all of those same alarms with two different parsing rules and you don’t know which one some downstream rule is going to use. It’s just not realistic to expect every such tool to parse both ways to form a conservative estimate. + +NRO: Thank you. That makes it clear. So I need a second to collect my thoughts. + +JKP: I’m just wondering if it would help for other people if we presented something and we can talk more concretely. Let me have the mic back. + +NRO: Slash slash at the beginning. Yes. So the concrete problem here is that if we do the next parsing, then—so for more context there are two ways of extracting comments from the spec. This is because some tools like Chrome DevTools parse it and look for tokens and the tools in this case would find source map. There are other tools such as VS Code or JS that instead take a different approach where they start from the bottom of the file and run on each line looking for things that look like comments and they bail out as soon as they find something that is not a comment. So in this case here, those tools would see line 6 is a comment and proceed to line 5 and see line 5 as a comment and do this source map and search the source map by notifying the server that this 5 has been debugged. + +MM: To make a point I’m sure you would appreciate is you can write a REGEX that detects all of the closed bracket issues, you know, the back quote and the quote and the double quote, the closed multi-line comment that detects and rejects all of those. So it’s the requirement that this recognition happened by REGX does not prevent us from adopting the safer rule. + +NRO: Yes, right. So my concern here is not with the solution itself, it’s with the process, just because I think if we decide to make this change before publishing the spec, then well it just means that we missed that and we have to meet again in six months, which maybe is acceptable, but I would like to offer—I would like to offer another solution here. And let’s see how you feel about it. So where I think we cannot normative changes today if we want to get this published, because we need to get that for, editorial changes are still with the editor discretion and hundred percent acceptable to that. So I would like to propose to do the following: We track this problem and we move the warning in the published spec say you need to check both ways because they might give different results and we are working towards that like linking to the issue. + +MM: I missed the critical word. You’re working towards what? + +NRO: Towards solving the problem. + +MM: Solving the problem. + +NRO: And then the draft—the published spec points to the living standard. Like, same as we do for this sentence saying limb stations should not implement what is written here but they should go out instead and check what is currently in the living standard and implement that. And that then will be published next year as a snapshot. And we made normative change in the living draft and not the published spec. In the published spec we have the warning saying we know that these map has this problem and then implementation would actually go to implement the living draft. + +WH: You made the claim that a regexp can detect comments. How would you do that? + +MM: Well, we’re talking about a regexp one line at a time. We’re talking about first of all rejecting the line that does not begin with a slash slash. IE, does not already look like a single-line comment. And then the only thing we’re trying to recognize, not trying to recognize comments on the line, we’re trying to recognize anything that might cause that line not to be a single line comment, IE, if it has in a program that parses, right? We’re not trying to—we don’t care about programs that don’t parse. So, you know, a program that parses something that scanning from the bottom to the top looks like a single line comment can only if I understand the grammar correctly, not be a single line comment if it has one of these cases, one of these four cases of a closing bracket. And in the case of a multi-line comment, the closing bracket would be star slash. What I’m saying seems—I mean, am I missing some subtlety about recognizing the close of a multi-line comment? + +WH: The example on the screen is not a multi-line comment. + +MM: No, the example on the screen is a backtick. I’m saying the thing that you would be recognizing is, you know, it’s basically saying that the line is acceptable if it begins with slash slash and then nowhere on the rest of that line does there appear a backtick, a quote, a double quote, or a star slash. + +WH: I don’t understand your point. Like, the example on the screen is not a comment. The approach that you described would claim that the example on the screen is a comment. But it is not. + +MM: No. The approach that I’m talking about would reject line 6 as not reliably being a comment. And if it’s not reliably a comment, then the process of looking for a source mapping URL stops there. + +JRL: Line 6. + +MM: We’re parsing backwards each line and process line 6. Line 6 we detect there is a quote and backtick and double quote that means it could possibly be a string and we stop execution at line 6. We’ll never get to line 5 to detect the source mapping URL. + +WH: Okay. So you’re going backwards and if somebody happens to have a true source map comment followed by a template, the source map is not recognized? + +MM: That’s correct. + +MM: NRO, did you still want to clarify? + +NRO: I think it was somewhat clarified. I was going to say that the doc here is not to detect comments in general, but just to make sure two different algorithms if they throw the result throw the same result. + +MM: So NRO, let me restate your proposal back to you in other words to see if I understand the proposal, that this first version of the spec that will have normative status for now would be unsafe with regard to this issue but it would have a non-normative note in it stating both that it is unsafe in this regard and the stricter parsing rule that it recommends happened instead and it’s just state that other parsing rule non-normatively so that existing tools would at this moment continue to be normatively conformant and then further more would be a non-normative statement which is essentially a future-proofing warning that we expect to tighten the normative spec in a later version to solve this vulnerability, so warning the tools in question VS Code, et cetera, please upgrade to the narrower recognition rules so that in anticipation of a stricter future spec? Am I understanding correctly what you’re suggesting? + +NRO: Yes, that is correct with the additional note that we will not have to wait for the next living +– the spec because the spec is referring to conditions to is the draft for Test262. + +??: I am fine with that. I accept that. In terms of, you know, the co-existence with the ecosystem and bringing things to a safe state in some ways it’s better because it brings the existing audience along without slapping the wrists—you know, without making them bad guys but brings them along in a cooperative manner but still gets enforce had is and is not normative. I’m okay with that. + +NRO: Okay. There is a topic from RGN. + +RGN: Yeah, following up on the conversation about regular expression-based matching and rejection, I wanted to point out to everyone that valid contents of a URL include single quote, asterisk, and slash, so you can have URLs that themselves contain some of the very characters you’re looking for as delimiters, which would severely complicate any such effort—for example, this is a valid URL: https://evil.example.com/'.A*/ + +NRO: Yeah, so that’s a very good point. It means to consistently bound these three closing limiters from source map URLs. In practice, this is not a problem because it can just parse the code dot special characters. And remember that here we’re talking about the tools and we’re talking about source code that developers have access to and not unreasonable to ask people to parse URL if they include these characters. But, yes, that’s a very good point which means we also need to add this requirement for the regex approach and non-regex approach and have this result so it would reject that example? + +JKP: I guess let me try to summarize. We presented the TG4 spec for approval. We heard back from MM about security concerns especially with the non parser-based approach and originally pitched to make a normative change today that NRO pointed out would not be acceptable without bringing it to TG4 and we came up with a compromised solution where we will today add I guess two warnings. One warning saying that we recommend the parser-based approach and why because there’s a security vulnerability and that the parser-based approach is better and another note saying we are actively working on fixing this problem in the non parser-based approach by coming one a normative change that would not have the security problem. And adding those two one note and one warning today would be helpful especially because the published spec would point to the living draft so as we fix it, implementers could see the new approach as they go. And one additional thing that RGN pointed out is that we need to be careful with making sure that we recommend to tooling authors that since we are checking for these characters, the URLs need to, you know, conform to this standard of not having those characters in it. Give it back to MM first. + +MM: I think that missed one very important element, which is that part of what we’re saying non-normatively is alerting people, warning people of an expected future version that normatively requires the safe rule so that everybody has a chance to move to the safe rule before it becomes normative. + +JKP: Yeah, that’s great. So I guess on that, we would be seeking conditional approval, the condition being the just outlined editorial changes and the, you know, promise of fixing this normatively in the draft. + +MM: So I approve of that. + +CDA: All right. I don’t think that changes the calculus from the group that we got earlier. + +JKP: I would like to give everyone, though, if it does change anything. + +CDA: Of course. Thumbs up from Justin, thumbs up from CM. Any concerns? Blocking? No, all right. Thank you very much. + +JKP: Just a note, thank you all for working through this. This is very cool to see and I really appreciate all the different pieces of feedback and context. + +CDA: Yes, absolutely. + +NRO: Yes, thank you everybody. + +### Conclusion + +- Resolved MM’s issue by adding an editorial note to the specification +- Agreed to send the 2024 TG4 specification to Ecma GA + +## Continuation: JSSugar/JS0 + +Presenter: Shu-yu Guo (SYG) + +- [slides](https://docs.google.com/presentation/d/1ylROTu3N6MyHzNzWJXQAc7Bo1O0FHO3lNKfQMfPOA4o/edit) + +SYG: I will read the queue. MAH’s queue item is dynamic JS requires shipping call compiler or ahead of time compilation of value added?. What does that mean? + +MM: I believe I know what he means. So if the normative language is the language that’s input to the JSsugar phase, then the eval in JavaScript is an eval that parses, you know, that also accepts the string in the full normative language. But part of the goal of what you’re doing is to move the JSsugar to JS compilation out to built tools, how is that compatible with having an eval that’s running within the engine? + +SYG: It isn’t. The eval would only support JS0 in this proposal, which is the same point, I guess, that it’s the smaller surface and if a particular application actually required dynamic evaluation of JSsugar features as well, it is up to that application to ship its own Transpiler. + +MM: So that’s very interesting. That’s sort of the first break where the split of JS zero versus JS sugar is something that a normal programmer would need to be aware of. + +SYG: Is it still true that the—like, I had assumed we did a pretty good job of dissuading people from using new function and using eval outside of, I don’t know, power use cases. So I wonder what the normal use case here is. + +CDA: Reply from KM. + +KM: Is it working? Sorry. I had it muted on the browser. Isn’t that also true for DEV tools and tried to type anything into the DEV tools you have a similar type of program in addition to eval. + +SYG: It’s different for DEV tools because DEV tools already is a—at least in Chrome is JavaScript application and when you load it, it loads all this stuff. If DEV tools found it’s important for its UX to developers to support JSsugar there’s nothing really architectural stopping it from shipping its own transpiler as a product choice for DEV tools. I think for an engine, it is a little different than there is more observable—it’s important that there’s interop for eval and I don’t want engines to decide one by one whether they ship a transpiler or not. For DEV tools there isn’t the same interop expectation. I think DEV tools can do something similar. We have—let’s see. There’s very similar to the problem today for DEV tools and not eval some folks might want to copy paste TypeScript. You can’t do this obviously in the engine. DevTools could do that if they wanted to. They have their own trade off and whether they want to ship something that does typed script or tiers nothing prevents from that today. We have as a committee explicitly rejected in the past the attempts to try to spec REPL behavior and such that and recommended basically REPLs actively deviate from the spec as it makes sense for their UX. + +KM: Sorry, I was trying to point it was a similar situation. I wasn’t saying there weren’t work arounds. I agree with everything that you said. + +YSV: Just to say Firefox we also have transpiling on console input. + +PFC: So transpilation is associated with JavaScript and I think it’s a browser centric concept. You don’t get people authoring their code in Python 3.12 and then downlevelling it to Python 3.10 and shipping that, right? I mean, this split into two JavaScripts it seems like it is beneficial for browsers but for people who write code that runs outside of the browser, it seems like we’re giving up all hope of ever going to a tool-less workflow. Or if you had one, you now have to give it up. For example, I’m involved with maintaining the JavaScript bindings for the GNOME platform that runs an embedded SpiderMonkey engine to do things like 'show your desktop', if you run a Linux distribution with GNOME. We are just sort of very awkwardly integrating a work flow with build tools in for people who want to author plugins in TypeScript. It’s a different environment than what you do in the browser. For example, SpiderMonkey provides an embedder API to get the current executing stack trace in a threadsafe way. So when the desktop crashes, which is a C program, we also grab the JavaScript trace in case it’s useful for the crash report. There’s no way that in the SIGSEGV handler we’re going to like try to load a source map library and provide a stack trace for the JSSugar code. That’s an illustration of the things that you run into when you try to apply a workflow with downleveling tools in an environment that is not really suited to it. So I guess my question is here like would there be any appetite for—like, would it be possible in the post-split world for non-browser runtimes to just implement the whole of JSsugar and JS0 as one language? + +SYG: Technically possible. Or asking if it’s realistic? + +PFC: Asking if it’s something that we as TC39, if we went down this road, would we consider that compliant or would it be a requirement to have the split into build-tool and non-build-tool JavaScript? + +SYG: I have thoughts here but I see a reply from CM. I want to go on the queue before I say my Peace. + +CM: Yeah, I’m having trouble understanding PFC’s model of the world and the idea that transpiling is browser-centric. Having a language defined as a core minimal language that is the target of higher level constructs being compiled to is a common language implementation pattern. And that’s a tool chain thing that people using those languages do this all the time. It’s definitely a shift in the operating pattern of some JavaScript developers, but I don’t see there’s anything fundamentally there that gets in the way of what people are doing when they’re targeting Node or some other non-browser environment. In fact, some environments like XS specifically need a tool chain in order to get there at all. So I’m just a little puzzled. + +SYG: Yeah, my piece here is that the evidence that I see is that the tools independently of this idea, right, this idea was inspired by what I saw in the tooling space in the server runtime space that they’re already integrating tools for the DX of their developers DENO and run and runtypeScript and node merging a thing of TypeScript to take TypeScript input directly and that is a thing that their users had demand for. And I certainly agree with CM that most of the world of programming that I’m familiar with outside of JavaScript are for compiled languages where this is a problem that definitely then needs to be solved. Tool chain problems are very real and require real engineering to address the concrete example that you gave of stack traces in the C program crashes and you get a dump with the stack trace you have to symbolic the memory addresses back to some sensible stack trace format with file names and line numbers and stuff like that. That is a thing that tool chains would need to do. Probably don’t want to do that in the segfault handler and dump thing and symbolic it after the fact. There’s a lot prior here that there are many, many—in fact I would say that most language in the world are compiled ahead of time and there is a lot of priority how to solve these, you know, production Issues. + +JGT: Yeah, I think one thing that would be helpful is I think I put an issue in the tracker too is just to define the various use cases, right? Because PFC is talking about one. I think the in-browser app is the obvious one that this sort of big reason for doing this I think node and REPLs are another one and DEV tools is another one and I think it would be helpful to enumerate and clarify what are those particular use cases are there for? Really only two with browser and nonbrowser? Sort of walk those through and I think to PFC’s point and others that integrating the transpiler, you know, essentially allowing node or DEV tools or whatever to JSsugar was a great idea. I find it hard to imagine this could succeed without doing that. That’s it. + +RPR: So the presentation was split between describing the problem and then pitching a solution. One of the comments on the solution was that JSsugar features can be less stable. This was cited as a benefit. But I didn’t see anything in the problem domain saying there’s a problem to be solved in this. So I just wanted to confirm is it—are we saying this just as imagining the JSsugar features represent somewhere between Stage 3 and Stage 4 and, you know, much like today, when we have something in Stage 3, it has not yet shipped in any browser, there is leeway for fixing up edge cases? Is this comment on stability reflecting that or inspiring to have more flexibility than we ever had before? + +SYG: What I had in mind is not really between Stage 3 and 4 thing but more like giving ourselves more flexibility to fix mistakes we wish we could fix post-Stage 4. Like, things we just didn’t anticipate ahead of time. + +RPR: In which case, then, the JSSugar features are not on the same linear proposal? + +SYG: They are. They’re still Stage 4. + +RPR: You’re saying once the JSSugar feature hits Stage 4, the stability of the Stage 4 JSsugar one will be different to the stability of – + +SYG: As a matter of fact but—de facto but not de jure and I think today there are proposals +– not proposals per se but maybe discussion topics that come to the committee where like that would be really nice to fix this thing or whatever and we have agreement here and then it comes to the browsers to say do you have cycles to investigate, do you want to even try to see if this is compatible, and I think for JSsugar feature when those kind of questions arise, it will be easier to try it for JSsugar features because if nothing else, tools are versioned. If we change it, the old tools compile to the old semantics, programs or apps that don’t update the tool chain remain with the old semantics and we change the new semantics, that’s a surprise to some set of app developers perhaps, but it’s a smaller subset of—it seems to be more like scoped problem that we can convince developers to update their tools to fix the thing we try to consider a mistake than the 2 billion aggregate users of browsers to update and break the sites. That’s the stability that I think will be less stability in the way that is beneficial to us, not less stability in the sense of we want to change the design of this thing and we should do because it’s in JSsugar. That’s my intention. + +RPR: Okay, that makes sense. And then, so we’re not talking about allowing things that Stage 4 JSSugar to have designed remit gated + +SYG: Like beings the working norm of the committee + +RPR: Yes. Thank you. + +YSV: Can yeah. I think SYG did this really well, but from our side, I think we would be very cautious about any kind of idea about versioning because part of the benefit of standardizing something is that developers can expect stability from those APIs and they won’t change out from underneath them. We need to respect that. We still have a very careful review process. And it would be really for addressing mistakes that we have made. Rather than versioning. + +CDA: KM? + +KM: Sorry. Yeah. I guess one can view it a little bit like aggregate—I agree Yulia, just like C++ has different versions and comes out every few years, and it’s rare to have things break backwards but have some flexibility to do things they consider mistakes. Yes, I think the—update being pitched or whatever is that, you know, there’s the real cost to going back on anything you said you were going to do, or we said we are going to do. But it is at least like having a viable shot rather than being completely non-viable. + +CDA: NRO? + +NRO Yeah. Just I wanted to mention that I think getting useful results of home compatible changes in tools can be hard from browsers. The reason being that usually for tools, the person that—like, let’s say—let’s say what we do today is the browser ship something in the better version, and then somebody says like there are users that better user verses of browser to help browsers with finding bugs, and so just navigate the web .… and the browser says, so, we have found this upset so we can do that. When it comes to tools, the equivalent of that is either to also do reins for tools. Some tools do that or to introduce bugs to non-experiment in the future. And the problem is that people that like the developers that would use the better version of a built tool are that aware of that plug to try to experiment or are usually developers that are, like, following closely what is happening, how the language is changing. And just going to run the tool, their code, which follows things, it’s likely the code would probably not be affected by the edge case. They are not going around randomly with people. So less likely to find incompatabilities. Although it has much lower chances of widely breaking existing websites. + +DE: Historically in TC39, we sometimes overestimated what could be done by engines. We have said, “this optimization should be able to make the semantics free or very cheap” but it doesn’t always work out that way in actual JavaScript engines. We shouldn’t make the same mistakes with tools. Tools also face limitations on what they can do in practice. In particular, although there are some optimizations that sometimes happen in tools, the basics of transpiling is really about very local transforms that don’t cross lots of boundaries and don’t do a lot of analysis. So I think overall, JSSugar doesn’t allow us to do different kinds of things in JavaScript that aren’t possible to implement interpreters because transpilers and interpreters are both limited, in practice, to the analysis they could do in a very limited local online way. + +NRO: I fully agree with what DE said. And I want to bring up some counterexamples thought about tools that did try to do like to adjust this based on very powerful code analysis. One of them was just called tray pack, developed by FaceBook, five to ten years ago, I don’t know exactly when doing analysis to see how various could line—what values would variables have with types, like be sure about the types, but about then—let’s hope developers get the approach. And it didn’t go, not get to go something concrete. Because although they worked in a lot of, like, types of analysis, they just kept the steps and it was difficult to cover. There is another very powerful one that was again developed by FaceBook and released very frequently. That is the—I don’t remember the name. The react compiler, which actually is a proper compiler, that is hair own, does their loan optimization passes and everything. But the only way that they can do that is by not actually following the semantics. They have react its own subset of JavaScript saying you can only call functions at this time and you can only call these types of functions. And so that compiler has to assume that developers have been very good at following those rules, just working with generic semantics. + +KM: Yeah. I think—I think there’s—my understanding and obviously SYG can correct me if I am wrong, the idea is not so much tools would remove these overheads. It’s that by existing in tools for incubating longer, that developers will start to see the potential, issues and performance from the APIs at a stage where we could actually revisit it. Rather than, like, after it’s too late. And so I don’t know. That’s just my two thoughts—two cents. + +RPR: So yeah. On this topic of limiting analysis to the file level or smaller verses, called program analysis, I think in terms of categories of tools, it’s interesting because there are whole program tools out there at the moment. Bundlers, webpack and so on that look at everything. But if we imagine this was something that might go into a runtime, the way that runtimes have integrated tooling so far is usually referred to as loaders. Where it intercepts the file being loaded, processes it, before putting it into the runtime. And as far as I am aware, all the real life examples that have today are all at the file run and there’s no such thing as, like, Deno or bun or whatever, slurping up the entire file program compiling it before running any kind of code. + +SYG: A quick response, do you think there’s no—unlikely that the whole program optimization is spaced? Unlikely to bundle? Not a bundle to happen. Like, bundlers do that because they’re, you know, they want to bundle, they are in a space to do that. The runtimes don’t do that because they have the same loading performance pressures that engines do. So of course the thing that comes out of that is, local transform per function step translation fast compilation because you want to get it running as quickly as possible. But the—and part of my thinking on why that is, is it’s obviously very biased. It feels like that the entire ecosystem for a long time incorrectly assumed that VMs are these magical go faster boxes. And part of this proposal would be to kind of shine a light on that. It would—like, it is—I mean, I agree with what Dan said, if you do it—if you look at local function transforms, then the limits on analysis between what the VMs do and what these local transforms do are very similar. But if that is the case, that is a cost that the user out to see explicitly in—association the developers out to see explicitly instead of saying, looking at—having it be hidden and hoping eventually, it will magically get faster because the VM gets getting better and better. Some of it is a psychological shift. If we make it more explicit, does that move the needle? + +RPR: I am not saying it’s impossible to convert the runtimes to be a full program JSSugar. I am just saying that’s like a new category. It’s a leap. And – + +SYG: I am not asking for that either. I think it’s probably the right choice for startup reasons that a server runtime does that—does like the perfect file transform without sophisticated optimization because presumably you are doing that, running a pipe file directly during development. When you minifier to prod on the web, you do that head of time compiler, whole program analysis when you are running the server workloads for reels. Why do not that work – + +DE: If I could address that. I wasn't talking about incentives, but what is realistic to implement. And for various different detailed reasons for different things, whether that’s transpilers, build tools or server runtimes for developers, everybody has trouble reliably implementing complex analysis that are always going to get you. It’s something in general you should avoid expecting this optimization language-wide because everyone faces this kind of structural constraint. Your presentation Shu is very good because it did shine a light on this issue. And we should be thinking more about tools and shining a light on there. We actually come into this kind of fortunate situation where it turns out that the constraints are very similar. So we’re actually designing the same kind of thing. So there isn’t actually a reason for the language to diverge based on the implementation needs of tools and engines. + +JMN: I realize this is all very much work in progress, but I just wanted to make sure that we don’t drop peer-review standards. I find this to be something that is very important for getting stuff out of TC39. We have a lot of people here and it’s a shame that tools took up the burden of evaluating whether these things were correct and whatnot. Maybe that’s not in scope. Spec it’s not my intention to move JSSugar outside of TC39. + +SYG: It is also a fact today that we don’t have as good representation of tool maintainers and authors as we do browsers and I would like that to change. + +DE: All right. I agree, we all like to have more tool representation and that’s why, you know, we go and I over the years have been recruiting more tools authors to come. The topic here is keeping JavaScript open for future .… I agree with JMN point. We need a strong review of proposals. Particular, browsers give strong review for TC39 proposals ensuring the rigor of everything we do. Thinking through a lot of these corner cases, I really, really hope that that continues. And I hope that we don’t start to see some TC39 proposals as kind of out of scope for browsers so they can let it go by. That’s a very important part of peer-review. For reunification, even for a while we accept a world where, you know, browsers are not implementing certain features that tools implement, if about change sometimes, then we have a language that is suitable for late implementation by browsers. We have been through different periods of development and disinvestment in the past, and we should continue to design something that if there is interest, then everything that we add to JavaScript could later be implemented in browsers. Even if it’s initially in some JS substance sublanguage. + +JGT: One thing that occurred to me, it does seem like a lot of the proposals that go through the committee might seem to have some pieces that would need to be in JS0. And then some pieces that could be in JSSugar. I want to ask if you thought about examples that have and kind of how that would work. And maybe an example that came to mind is something like BigInt. Where there’s a BigInt implementation, API, that might make sense to put in JS 5 for performance and then there could be the BigInt primitive type, which is syntax that could be JSSugar. I have no idea if that’s a good idea. Probably not. That was the thing I was thinking of and is that something you see happening? + +SYG: Yeah. I think that wouldn’t actually happen. Having a split—if you have—proposing something that is hard to do today, it needs a new syntax proponent and a new runtime proponent, it makes sense to have to be in JS0 and have that a minimum core that the engine support while the syntax part is in the JSSugar. Interesting thing that came up talking with Chip earlier, I hadn't considered before, it’s also, I suppose possible for there to be a small syntax core that is put into JS0 to support a complex syntax proposal we don’t otherwise want like extractors. With the feedback from yesterday, that destructuring is very—it’s difficult for tools to transform extractors inside the structuring without—without doing everything. I suppose one way forward could be to use a small syntax core that does something with destructuring to enable structuring to be purely implemented to transform better. I need to do more on that. Like I suppose that’s possible. I don’t want to be so absolute as, like, we will never add new syntax again. But the more manageable the problem is, the more we think about, like,—more confident, the engines can be more confident in scoping out the complexity, whether that is runtime or syntax. + +JGT: Make sense. There’s one sort of—related to this is one thing I found hard reading through the slides was to think about how—how—like, what would this look like for an actual proposal? And I almost feel like it would be much easier to talk about concrete examples and the tradeoffs inherent in those than it might be to sort of think of it theoretically. So that would be sort of my I guess request for the next time this comes up is, to try to sort of layout maybe some proposals, volunteer to see how that could work. But that would be very interesting. I think, and very helpful. That’s it + +CDA: I am a little bit nervous about this. Especially like moving to a new world overnight, we are in this new working mode. What is the feasibility of something like taking a low stakes proposal and one we like to see in the language itself? We want it in JS0, but could we try and do it in this way, as a proof of concept? + +SYG: What is ‘this way’? + +CDA: JSSugar + +SYG: Like a syntax thing we agree to be in JS0, but to try—I don’t know what it means to trial in JS. That is the state of the world today. When something goes to Stage 3, the first thing ways—the initial ways that JavaScript develops is via transpilers, not direct execution. That is the world today. I want to know what the delta is what you are thinking of trialing. + +SYG: Like Stage 3 decorators, I suspect a very long time until there’s direct execution in a way across the major browsers. TypeScript got that out quickly. And I imagine it’s getting used + +CDA: The trial is polyfilling? + +SYG: No. Transpiling. It is the world today. That people consume Staged 3 features via transpilers that have been implemented. + +ACE: I almost completely agree we have been effectively trialing this for like a decade. Apart from the stability aspect. There are libraries using Stage 3 decorators. But I suspect they are groups of people not using decorators because many things in Stage 3, they have been in Stage 3 for a while and that adds a nervousness to things. The thing we could trial, I am not saying we should do this, but to add to—we could trial a one-time change to our process where we say that something is stable, even though it’s not in browsers. It’s one thing, we will try and see, does saying this thing is stable, does that change the way it’s treated and do more tools adopt it? So potentially there is a delta from where we are today to something else. + +OMT: Yeah. I will say this is an example, but they have six different versions of the decorators proposals in the plug in. Because they say TC39 plenary where it’s discussed. Stability should be posh, even with transpilers. + +CDA: DE, you had on the queue, How do tradeoffs for JavaScript compare to CSS and WebAssembly + +DE: Yeah. So SYG’s proposal for JS0 and JSSugar singles out features and aren’t libraries as special sources of features that are especially unmotivated to be featured in browsers. Sometimes this happens in other web platform features, like CSS nesting. I am not sure if that has runtime applications or adds to more syntax sugar. Is there some difference between CSS and WebAssembly in terms of costing less to change or being seen as more useful? + +SYG: I don’t want to speak too much about CSS. Because I am not familiar—I am not familiar with the working code of the CSS working group. How features get proposed, I don’t want to really just like out of ignorance, I don’t want to do comparison, me. If somebody else know .… but between JS and Wasm, I think JS is a much, much more mature language. Relatively speaking, it is pretty expressive already. There are a few missing capabilities in my opinion and I am working on exposing them, but outside of things—there aren’t that many things you can’t do for the purposes of general programming today. So I think from that point of view, a lot of JS features are about the quality of life improvements, dev, problems that a mature language has. Wasm, is in growth mode. It needs to attract more developers. But I think Wasm already works kind of closely, somewhat close to the JSSugar, they have tool chains in their process. People aren’t writing Wasm by hand. So big tool chain, Emscripten does this in the standardization process, and as I understand there is discussion when they proposes a feature ask what to go into Wasm core and handled by the tools and this is often a in-depth discussion and harder than the JSSugar problem because Wasm is a general purpose target, I propose JSSugar to be a JS target. Wasm is a target for C++, Java, Kotlin, Swift, whatever else that people want to target Wasm for. And you can imagine that with many different programming paradigms, you can get into a pretty big design scope of what ought to be in core Wasm and what is solved by the tool chains. So I think Wasm already works in this way basically. They—and different people on the WasmCG have different opinions of course on how small the core should be. But I think they do generally have a constraint that they want to keep the core both language agnostic and small. We’ve butted heads with them as a JS community before. We wanted the strings to be first class in the core spec—and the response was, no. Because that seems to favor JS, they said no. Because of not favoring any particular source language, out of minimalism and things like that. So I guess, I think the tradeoffs for Wasm definitely exist. But there are also—I don’t know. I am rambling at this appointment. Is there anything .… any more – + +DE: Thanks for the explanation. Two parts to that. One part: JS is mature. That means we should be cautious about things to add, but a lot of value is created by taking the right things to JavaScript. The other thing is, you mentioned earlier, in your slides a bunch of CVEs and some of these were for syntax things but some of them were not for—anyway, do you see greater security risks to add features to JavaScript to WebAssembly? Or cost to users in the various dimensions you mentioned. + +SYG: As a VM, I would have—I don’t think adding things to JS is any worse for security than adding things to WebAssembly. I feel they are the same. If you look at CVEs, which are, as I said in the slides, an undercount for V8, like probably in recent years, there’s more Wasm CVEs than JS. If it goes into a browser VM, it’s super targeted and attacked. And I think that at—it re-enforces the point that complexity = security. There’s more features are added to WebAssembly, big Wasm GC and JSPI and that is complexity and that = security bugs. So whatever gets more features and complexity tends to have more security bugs. + +DE: Okay. Thanks. + +KM: Yeah. I mean a lot of the WebAssembly—going—there is a lot of parts to get to on the WebAssembly side. In our side, WebAssembly, I have hard numbers, I didn’t look this up in advance, WebAssembly has dramatically less CVEs than JavaScript. And so—but Wasm GC is likely, you know, the next wonderful frontier of CVEs for us. But the—I think a lot of WebAssembly features have gone out, most of them were just smaller performance improvements or an—general performance improvements. So the use case to the users is pretty compelling, I guess, rather than features that we don’t think people will use and since we know people will use them, if they will use Wasm at all. + +RPR: Yes. The original assertion we are going to do tools only would—mostly tools. I think the set of people that use tools overlap strongly with the member who use third party libraries and the supply chain of JavaScript. At the moment, NPM is the world’s largest place of JavaScript. And nearly all of that is directly executable. Even with, you know, the most popular compiled form of JavaScript at the moment, TypeScript, the vast majority of that gets compiled to JavaScript first. So we are distributing executable code. With the JSSugar proposal, what mode do we expect or what mode would we encourage? Still encourage people to convert to executable JavaScript or raise that? + +SYG: I mean, I am not an expert here. I am not really—I don’t know very much, but the folks that I have talked to, in the meta framework sense, like Next and Nuxt, it raises questions from folks from FaceBook, it feels like it’s JS0 today. Keep publishing executable code. And maybe we need more innovation or support on how to recover the source. I imagine the problem exists today, maybe .d.ts files are good enough that you don’t need to recover the full source or something. But like when you publish something to NPM today as an executable source, don’t you also have to choose charge level, the verse—as ES5 or whatever? How is that dealt with today? I guess that is the question. I want to see what the status quo is and figure out how to get guidance. + +DRO: Yeah. I was going to say that I feel like a lot of NPM packages I have seen, especially recently have both JavaScript and script type bundled into them already. The answer to that question is that, you know, if you want to to be something that is directly executable you can have it be that way, and also something that if the package author wants it to just only be JSSugar, they can make that decision. And ultimately, it’s up to them. There’s the minimum bar, if you want to be executable, you can do JS0. Or do both. Like, I feel there’s options for either. + +RPR: Yes. I appreciate that it’s a free world. People can choose. I think we should go into this with a plan, as opposed to letting stuff happen. + +SYG: I want to push back on that a little bit of stuff happened regardless. Where I am taking stock of the stuff that has happened. And trying to kind of meet that as going with the flow instead of saying we don’t like it and forcing something else to happen. Why should TC39 say something about NPM? We can give guidance, but how do we make them do stuff? + +RPR: I am not saying we have to make anyone do something. But checking what we are doing is setting things up for success. + +JWK: We also shouldn’t publish JSSugar on NPM directly. Today some library publishes as TypeScript directly and it brings many problems to their users because a classical transpiler configuration is to exclude everything in node_modules. If libraries transpile the code to JS0, then everything works well. + +JGT: Yeah. Just a note on that, as I have spent a bunch of time in space to try to understand when TS code is included in published packages. It’s there for debugging purposes especially VSCode and it likes debugging in files a lot more than in source maps and you can do things like setting breakpoints is easier. A bunch of things are easier to debug when the fills are there. One of the challenges is that frustrated people run it, it won’t work, transpile, it won’t work because the T S config is different. This is a—this is a similar problem. Right? Where people are going to debug it, have a way to do it and the tools have to buy off on this is the way you debug all code, not just TypeScript. So you are essentially TypeScript-ification of the entire ecosystem inheriting some of the challenges + +SYG: When I was talking with the meta frameworks another example they gave, I will parrot at face value, there’s some cases where people want the original source because they want everything to be in one bundle to go through one tool chain at the end for everything. If you publish JS0, that is difficult for that kind of work flow and maybe is another reason why you want this dual kind of original source and directly executable kind of code. + +DE: There’s been a movement over the past several years of doing less transpilation before putting things into NPM. Actually, Chrome has been important in promoting this practice of doing `