-
Notifications
You must be signed in to change notification settings - Fork 158
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Proposed semantics for calendar system support in Temporal #268
Comments
What kind of object would be returned from Temporal.Date.prototype.as? Temporal.Date.prototype is currently Gregorian and specifically uses ISO 8601 conventions for |
A plain object, or possibly a Record. This object would only be returned from the getter and would have no other fancy operations on it.
We'd have to do more research on
The string "reiwa" in my example would be an identifier that we define. That identifier could be passed to Intl.DisplayNames, for example, to get the localized string. new Intl.DisplayNames("ja", { type: "era", calendar: "japanese" }).of("reiwa");
// "令和" |
Bikeshed of other APIs for the getters: // Alt 1a
date.inCalendar("hebrew") // returns object
// Alt 1b
date.calendar("hebrew") // returns object
// Alt 1c, if 1a and 1b are too long
date.cal("hebrew") // returns object
// Alt 2
date.getMonth("hebrew") // returns number
date.getEra("japanese") // returns string
// Alt 3
date.getIsoMonth()
date.getHebrewMonth()
date.getJapaneseEra()
// Alt 4
date.getIsoObject()
/*
{
calendar: "iso",
month: 11,
day: 19,
year: 2019
}
*/ The fundamental concept is that the internal representation of Temporal.Date is always the same, but you have to specify the calendar upon input/output operations that are calendar-dependent. The exact API is flexible. |
playing devil's advocate -- instead of universal-epoch, maybe lookup-tables between iso-year-ranges 1000-3000 (or even 1500-2500) are a more cost-effective approach? i don't think anyone realistically expects utc-accuracy (or consistency with external database-operations) outside of those date-ranges. i'm also not certain utc-based time or the javascript-language (much less this proposal) would still be relevant 500-1000 years from now. |
By "cost-effective" do you mean better for performance? I would assume that yes, anyone implementing this proposal would probably use some kind of lookup table to convert between calendars. That's an implementation detail that doesn't need to go into the spec IMO. An interesting question would be, if we were to move forward with my proposal in this thread, would we change the spec to make Temporal.Date store the "# of days since epoch" as a BigInt instead of the year/month/day tuple? Again, that's an implementation detail, and I claim my proposal can be implemented regardless of the internal representation of Temporal.Date. However, we may want to at least document that Temporal.Date is defined based on "# of days since epoch" to make the logic for calendar math more clear to users. |
I've experimented with supporting different calendars. Specifically I investigated the Hebrew, and Ethiopian calendar system. I do not believe that this suggestion works consistently especially once one adds time information. We did discuss this (and I was a vocal advocate for enabling support for multiple calendar systems) about a year ago and concluded that we would keep that aspect out of scope for this proposal. I would therefore ask if you can agree that this can be part of a follow-on, and not cram it into this proposal. |
For calendars where the day does not start at midnight, I proposed an approximation based on long-standing ICU behavior where we treat days starting at sunset as starting at midnight approximately 6 hours later.
Although full calendar support could be a follow-on, I do think we should consider carefully the assumptions that would be hard to take back later. I claim that the bulk of Temporal actually works fine even if we're not in the Gregorian calendar. And, the best way to think about how to prevent bad assumptions is to design calendar support in v1. |
This comment has been minimized.
This comment has been minimized.
It could be seen as incorrect in ar-SA for date arithmetic to operate in the Gregorian calendar, when dates are displayed to the user in their local calendar. In fact, the best default for the calendar may be the locale's default calendar, but that might introduce SES problems. Making the calendar explicit solves any ambiguity. If you want your operations to be in the user's preferred calendar, you can do, let calendar = new Intl.DateTimeFormat().resolvedOptions().calendar;
// use that calendar for Temporal operations And if you really want your operations to be in Gregorian, then you just let
I'm specifically suggesting that Temporal.Date does not have calendar as part of its data model. There would be no calendar-specific types in my proposal. Rather, conceptualize Temporal.Date as a calendar-agnostic number of days since an epoch, which can be mapped without loss of precision into any calendar system by use of getters. If a Temporal.Date is passed into an Intl.DateTimeFormat, that's perfectly fine, because the calendar used for display is the one coming from the Intl.DateTimeFormat. |
This comment has been minimized.
This comment has been minimized.
To be clear, are you suggesting that the default calendar remain Gregorian when the programmer performs arithmetic? Or would the default calendar come from the locale? Here is the use case: you are building a web site in which you have a month/day picker. You use |
This comment has been minimized.
This comment has been minimized.
Yes, I totally agree with @sffc and also encourage NOT to defer to the future designing calendar support. It's going to be just harder compared to doing it now in v1 where the API freedom is bigger. |
It's important to emphasize that "No support for non-Gregorian calendars" is one of the six major problems stated in the motivation section for Temporal. Therefore, such support goes in line with this proposal's mission. We all probably don't want to take the risk of needing a Temporal2 in the future 😄 |
@sffc proposal looks good to me in terms of i18n. It would be great if the proposal experts can help improve it. Thank you |
Way back when (like 2 years ago) a younger, naiver and much less cynical me entered Temporal-land with the idea that we should have types to reflect all calendar systems in an effort to a as multicultural and inclusive as we could possibly be. I did about 3 months of research to get to know different calendars, by the end of which I came to the conclusion that supporting a subset of the ISO-Proleptic-Gregorian calendar is more than enough of a challenge and more than enough to take on for starters. I did for example investigate the Hebrew calendar. Depending on which rabbi you speak to you get differing rules:
And that's just the rough outline of the Hebrew calendar; we haven't even gotten started on the astronomical database and required geo-location required to actually implement this. In short I am unwilling to start the Hebrew calendar wars of 5780! And that's before getting started of the Ethiopian calendar/time and the slight differences to the Eritreans. There's a whole other sets of conflicts in that. In short: while I am very sympathetic to supporting multiple different calendar/time systems, I would definitely want to declare this out of scope for this proposal. I encourage anyone reading this to contribute to a dedicated issue to discuss how to support different calendars and bring that to both a future Temporal follow-on proposal as well as ECMA402 |
About the Hebrew calendar, I've stated that my opinion is that we should follow what the industry has been already doing for 20+ years, which is what you refer to as the "progressive" option. This is not a question we can avoid, because we have to decide one way or another in order to implement #262, toLocaleString().
We would not want to get in depth on every corner of every calendar system conflict. We only need first-class support for the most common calendars, which I propose to be the ones that are preferred in at least one region according to CLDR data. We could even narrow our scope further and only support calendars that are the first choice in at least one region, which a shorter list of only 4 calendars (gregorian, islamic-umalqura, persian, and buddhist).
The problem with this, as I've stated before, is that if we don't think about this now, we are likely to build in assumptions that would make adding calendar support more challenging in the future. |
I like sffc@ proposal, because it sidesteps some of the problems with current Temporal approach by using days since epoch, and keeps door open for more i18n friendly APIs. If we marry Temporal to gregorian (because it solves 90% of use cases for example) then that door is going to be much harder to adjust later on. Similar thing happened with original JS spec - there was lots of pain to push ES 402 through because of baked in assumptions, and English first approach (which did solve 90% of problems at the time). |
Thanks for this background on a way forward for the start of days, @sffc . Are there any other technical concerns people have with the proposal? |
I think it can work, but I'm not 100% certain that I understand the entirety of how currently-planned interfaces would change. Ignoring new methods, what if anything is missing from this list:
|
@gibson042 Thanks for the reply :)
Correct.
These would become accessible via the
Not correct. Temporal.{Date,DateTime} would not have a calendar field because they would be defined in a calendar-agnostic way.
These properties could also go on the
Temporal.{YearMonth,MonthDay}, if we keep them, would need to have a calendar field, as discussed in #264.
Irrelevant for Temporal.{Date,DateTime}. Would need to think about what behavior to adopt for those methods on Temporal.{YearMonth,MonthDay}, if we were to keep those types.
Irrelevant for Temporal.{Date,DateTime}. For the other types, yes, we'd probably have to fail if
Correct, although Temporal.{Date,DateTime} have no calendar field. |
I don't think so, because the interpretation is calendar-dependent (even in Gregorian systems, USA and perhaps all of North America conventions start weeks on Sunday but ISO 8601 starts them on Monday).
That might have bad ergonomics for arithmetic since Temporal.Duration does have a Still, I don't see any fatal flaws with the approach. |
Note that the first day of the week is region-dependent (not calendar-dependent) (CLDR ref) (UTS#35 ref). For example, it's "sun" for US but "mon" for DE. That being said, I believe that's irrelevant for Temporal, which is expected to always return the same ISO-8601 numeric index for PS: |
Right, this is my understanding for dayOfWeek.
My proposed arithmetic semantics are: let date = Temporal.now.date(); // a Temporal.Date
// d1 = date minus 1 gregorian month
// From November to October, this equates to 31 days
let d1 = date.minus(new Temporal.Duration({
calendar: "iso",
months: 1
}));
// d2 = date minus 1 hebrew month
// From Cheshvan to Tishrei, this equates to 30 days
let d2 = date.minus(new Temporal.Duration({
calendar: "hebrew",
months: 1
}));
date.toLocaleString("en"); // 11/22/2019
date.toLocaleString("en-u-ca-hebrew"); // 2/24/5780
// `d1` has the same day as `date` in Gregorian; the Hebrew day changes
d1.toLocaleString("en"); // 10/22/2019
d1.toLocaleString("en-u-ca-hebrew"); // 1/23/5780
// `d2` has the same day as `date` in Hebrew; the Gregorian day changes
d2.toLocaleString("en"); // 10/23/2019
d2.toLocaleString("en-u-ca-hebrew"); // 1/24/5780 |
Ok so let me review:
Of those we'd have to drop a whole bunch because they are intrinsically calendar specific leaving:
Except that a lot of those are actually calendar specific as well:
Which leaves us with an object wit the properties:
with which I cannot actually do anything useful beyond what I could do with a plain JS object. Beside that, it takes away a lot of the comfort and guaranteed correctness away. Temporal currently guarantees that a Date object is actually fully correct when it is created. It guarantees that comparison and arithmetic will be correct for any two Date objects. It guarantees that combination with a Time object will result in a valid DateTime. It guarantees that it will complain (throw) at the earliest moment possible when something would result in an incorrect object. Beyond that this proposal becomes very complicated in actual use. So that while it's indubitably correct, it's also something that would be so hard to understand and work with as to be entirely irrelevant. Now as to why we originally decided to make differing calendars out of scope and what we have done to make adding them later easy. The So to implement a Hewbrew calendar (or any other) one would have to specify 2-3 objects (HewbewDateTime, HewbrewDate & possibly HebrewTime if one were orthodox). These objects would encompass all the properties, limits and algorithms that define the Hewbrew calendar. That's it, and it becomes easy and quite clear that there is a difference between As for In short: we had a lot of in person conversations where I was trying to push for multi-calendar support and calendar agnosticism. It took several months of conversations and lots of research to convince me that while possible, and possible to do correctly, the result would be entirely impractical. So again: I urge those present here to take their time and think through a lot of the edge-cases. And hopefully come to realize that while possible, this is not something that can be easily done. At the same time, I believe Temporal as it already stands can function for the 99% case quite well and does not (because I've taken great care to ensure that) make it impossible or even hard to add support for other calendars in a follow on proposal. Follow-On proposals: Please don't misunderstand me. When I say follow-on proposal, I don't envision this to be a thing of additional years, but rather something that could be started immediately after Temporal. I'm very much with Shu who is thinking of proposing a more iterative modus operandi for built-in Libraries. I see Temporal as building a house. What I propose is to defer this to a next step; akin to waiting to put on the roof until the walls are standing. Apologies if this has been long and possibly harsh. I admit to being a bit exhausted right now and didn't really want to write this comment until some-time later; However I felt this was getting off to now good place and beginning to be detrimental. Be patient with me and take this in the spirit in which it is meant: one of trying to reproduce the content & arguments of discussions had and resolved long ago. |
I want to be clear: in the OP, I made a concrete proposal with semantics on how to handle calendar systems in an uninvasive way. Your reply above lists out many points about generally why calendar systems are hard to support, but it does not address the substance of my specific proposal. I claim that my proposal addresses most of the issues you raised.
|
This comment has been minimized.
This comment has been minimized.
I claim that your proposal does not address the issues I mentioned.
In short I did mean to interact with your proposal directly. I accept that the explanations I gave were more in terms of calendars in general. I am however definitely claiming that your proposal solves none (or very few) of those issues while at the same time significantly impacting ergonomics. |
Ok. This is our fundamental disagreement.
Could you share an example where conversion between an ISO 8601 Date and some other modern-use calendar system is not possible?
This is objectively only true if months and years are in play, which is often not the case. Richard's cookbook suggests that only a small portion of Temporal use cases are calendar-sensitive. My proposal has zero impact on the ergonomics of the majority of use cases. |
In the meantime it’s probably worth it to have a bit of a think about time |
I agree with @pipobscure . Though most (all?) calendars have very similar concepts like year, month, day, most programmers would think We already see japanese year is not a number but number plus era, it's very unclear how Japanese programmers expect So it's very unclear to me that how the surface consistency like |
Hi, hax, If I read your post correctly, it sounds like you believe true JapaneseDate and ChineseDate types present better ergonomics than a generic record as I had proposed returning by This is a good point to discuss. In my proposal, I had preferred generic records rather than fully powered types, because:
However, I can see the other side of the argument: perhaps the idea of a calendar-agnostic "epoch days" type is too abstract for the average programmer. My counters to that argument would be:
Thanks for the discussion. P.S. I've already been in touch with the Temporal champions, but I'm planning a deep-dive session on this topic this Monday before TC39. If you are interested in joining, please DM me (e.g., send an email to sffc at google.com). |
@sffc please feel free to correct me if I'm wrong, but do programmers using That said, even if average programmers need to be made aware of
would be a big deal, since it's not a very different idea than unix time and tbqh that's also a similar abstract idea that is expected to be understood by most if not all programmers anyway, so... |
for most programmers, Temporal source-of-truth is what goes on inside databases rather than inside javascript/c#/java/etc. even if the database is wrong, you often still go with it, because you don't want customers to see different answers (and perceive bugs) who use the same database on their end. speccing Temporal's internal behavior is beneficial not because i care to understand it (i don't), but because i care to know which operations are inconsisent with their equivalents in sqlite3/sqlserver/mysql etc. (so i can avoid them). |
You're correct. Most programmers don't need to know this. In fact, the internal representation of Temporal.Date could remain as it is today as a proleptic Gregorian date. I'm using "epoch days" in this thread to make more apparent the 1-to-1 mapping between Temporal.Date and arbitrary calendar systems.
Yep, pending full API design. I expect to embark on the naming of methods, parameter order, etc., if/when we reach consensus on this being a good idea. |
Here is a photo of the whiteboard from today's deep dive. We hashed out some of our differences in understanding. Some key insights:
Discussions will continue this week at TC39. |
To be more precise: I'd personally prefer ISO to be spec'd in 262, CLDR/ICU-supported calendars in 402 and everything else in userland. It's important to mention the latter and think about third-party calendars during the design process.
I see the sense in both the arguments. While we shouldn't be too biased against non ISO-compatible calendars in that we make them un-implementable (there's well-documented use-cases, IIUC, like financial calendars, or hotel calendars (days begin at noon)), I understand that the percentage of users/developers who will stick to an ISO-compatible calendar would dwarf the rest. What about some sort of middle-ground? What about giving a preference to ISO-compatible calendars? Since incompatible calendars aren't sanctioned by CLDR (going by @sffc's assumption here), they'd likely always be implemented in userland. It wouldn't therefore be too outrageous if people implementing these special calendars were expected to jump through an extra set of hoops IMO... Just spitballing here, but here's an idea: What if a |
Sorry to chip in with my 2 cents here after some absence, but some thoughts:
|
It's hard for me to make sense of this comment. This is a bit of a tangent, but: We haven't been normatively referencing CLDR/ICU in the specifications, and I don't think we should change this: implementations may tailor their data. As far as "userland", are you picturing a "data-based API" for calendars, or a polyfill of Temporal? |
One other small thought. I think that when we say "ISO Compatible" what we really mean is "Does not require time to make calendar computations". The differentiation between "Date Math" - math that involves units of days and greater - and "Time Math" - fuzzy because it includes things like DST, or possible sunrise/sunset times - is a distinction I often make in conference talks and the like. |
Default CalendarsI think there are two sensible default calendars:
I suspect that (2) is actually what most users building global-ready web sites should prefer as the default. Therefore, I feel that it is bad i18n design to have an implicit ISO calendar default. However, in my proposal, I discuss "calendarless" operations, for which the user would not have to specify a calendar, because those operations are calendar-agnostic. More specifically, they are calendar-agnostic insofar as the target calendar is ISO-compatible, a property shared by all modern-use calendars I have researched. The programmer, such as a bootcamp student, would need to explicitly specify the calendar only when performing a calendared operation, which is the minority of use cases according to @gibson042's research. So, it's a choice between (1) accepting the idea of making ISO-compatible calendars more equal than non-ISO-compatible calendars, (2) accepting that all call sites would require some boilerplate to specify the calendar, and (3) accepting behavior that is not i18n-optimal. 262 vs 402There are a few reasons why putting non-ISO calendar definitions in 402 makes sense:
|
I had the good fortune of meeting @amireh, who worked for many years in Jordan as a software developer. My understading from this conversation (please correct me, @amireh!): We're doing a very good thing for developers working to produce work for these locales by providing support for the Hijri calendar. It'll be really appreciated by people who have to work with these calendars. (Probably there would be a lot of benefit just by telling more people about In many places across the Middle East and North Africa, the Hijri and Gregorian calendars are used side by side; different people have preferences for different calendars (e.g., a trend might be younger people preferring Gregorian and older people preferring Hijri, in certain places). Sometimes, multiple forms of the same date are presented (e.g., one on top of the other). Sometimes, an application's settings would let someone toggle their preferred calendar. In Jordan, if I understood correctly, the official calendar is Gregorian, but many people really prefer Hijri--it makes more intuitive sense for them. Software developers just have to think about the two different calendars all the time. When we discussed the default calendar, Ahmed was initially really excited about automatically getting the right default calendar based on the locale. However, as we chatted about how this works out in practice (no forwarding of OS preferences, just based on the region, and of course no application-specific preferences taken into account), it started to seem like the important thing is to have built-in Hijri calendar support, that an ISO default is understandable and not harmful, and that a region-based Hijri default wouldn't really be enough to solve the problems that come up in practice. CLDR includes some really interesting calendar preference data. In CLDR 36, the four regions which have a first-ranked non-gregorian calendar are:
If we're considering a non-gregorian default locale to software developers working for these regions, I want to suggest that we see if we can get in touch with some of those developers; they may have some interesting feedback for us. Sidenote: 262 vs 402I think we're getting at a bit of an editorial issue, not one with normative/observable semantics. Right now, 402 patches 262, e.g., with toLocaleString() implementations. You don't need ICU to implement 262, just 402; you can use the dummy toLocaleString implementations that ECMA-262 has. I think we could do the same here for Temporal, and just use the ISO calendar in 262. But it's really really hard for me to think of any observable consequences of this spec layering. Has anyone argued for some other spec layering or observable behavior? |
Awesome! I don’t think anyone has suggested a different layering. |
i feel iso-gregorian should be the default for interoperability with databases. this proposal has lots of nice-to-have features, but lets be honest -- most temporal business-logic will not be done in javascript -- in-practice we rely on external databases (or client-side sqlite3) for most temporal heavy-lifting. @littledan @amireh, imagine the pain of having to explicitly set calendar=iso everytime you have to interface temporals with sqlserver/mysql/postgres/sqlite3. the cost in bugs if you forget that chore is not worth it. |
This issue is pre-December TC39. A fair bit has changed since then, which addresses some of what is posted above. I'm going to close this issue, since I triaged these discussions into their own issues. Default calendar: #292 Set of calendars to support: tc39/ecma402#395 @kaizhu256, the "pain" of explicitly setting calendar = "iso" is at the forefront of the default calendar discussion, and we have a couple of options on the table to address that. Read more in #292 and the linked section in calendar-draft.md. |
Discordian ftw. |
We had a discussion today about this at the Google i18n design meetup. I wanted to post an idea for a way to make Temporal support different calendar systems without making major changes to the API.
Lemma: We can think of a Temporal.Date as some discrete number of days from a universal epoch (say January 1, 1970). For example, today, November 19, 2019, is 18219 days since January 1, 1970.
Given this mindset, all calendar-sensitive operations can operate as though that discrete date were first mapped into the respective calendar. The fundamental calendar-sensitive operations are only:
Note that adding 1 day is not calendar-sensitive in my proposed framework.
What could this API look like? Here are some examples for the getters:
For the arithmetic (Temporal.Duration):
If a calendar is not specified, should we default to the ISO calendar? From an i18n perspective, no, we should not operate using a default calendar. Requiring users to specify the calendar makes them go through the thought process of making an informed decision.
@rxaviers @littledan @younies
The text was updated successfully, but these errors were encountered: