- 
                Notifications
    You must be signed in to change notification settings 
- Fork 122
Added LocalTime.Formats.ISO_BASIC (ISO 8601 basic format) #518
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
base: master
Are you sure you want to change the base?
Conversation
| ISO 8601 specifies that the format for time-of-day includes a leading  | 
| You are right, thanks for spotting this! I believe the same should also apply to LocalTime.Formats.ISO. Should I attempt to change it as well? | 
| 
 The extended ISO format is allowed to omit the  In fact, regarding usefulness: is  | 
| I find it very useful, for example if one need to have a timestamp stored in a file name, in my case  | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Got it! I don't see any problems with adding a LocalTime.Formats.ISO_BASIC if it is indeed useful, but by that logic, it also makes sense to add LocalDateTime.Formats.ISO_BASIC (which is the 20250415T065500 part in your message).
| // these are constants so that the formats are not recreated every time they are used | ||
| internal val ISO_TIME_BASIC by lazy { | ||
| LocalTimeFormat.build { | ||
| char('T') | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please see internal val ISO_DATETIME: our format is case-insensitive.
| * | ||
| * @sample kotlinx.datetime.test.samples.LocalTimeSamples.Formats.isoBasic | ||
| */ | ||
| public val ISO_BASIC: DateTimeFormat<LocalTime> | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please also add tests. Examples: LocalDateTimeFormatTest#testIso, LocalDateFormatTest#testBasicIso, etc.
        
          
                core/common/src/LocalTime.kt
              
                Outdated
          
        
      | /** | ||
| * ISO 8601 basic format. | ||
| * | ||
| * Examples: `1234`, `123456`, `123456.789`, `123456.1234`. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
These examples are not valid for this format.
| Should be complete now! | 
        
          
                core/common/src/LocalDateTime.kt
              
                Outdated
          
        
      | * When formatting, seconds are included, only if they are non-zero. | ||
| * Fractional parts of the second are included if non-zero. | ||
| * | ||
| * See ISO-8601-1:2019, 5.4.2.1b), the version without the offset, together with | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
| * See ISO-8601-1:2019, 5.4.2.1b), the version without the offset, together with | |
| * See ISO-8601-1:2019, 5.4.2.1a), the version without the offset, together with | 
b) is the extended format.
| * | ||
| * When formatting, seconds are always included, even if they are zero. | ||
| * Fractional parts of the second are included if non-zero. | ||
| * | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think LocalDateTime.ISO_BASIC is also worth mentioning (given that you personally were interested in LocalDateTime.Formats.ISO_BASIC, for example).
| optional { | ||
| second() | ||
| } | ||
| optional { | ||
| char('.') | ||
| secondFraction(1, 9) | ||
| } | ||
| } | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
| optional { | |
| second() | |
| } | |
| optional { | |
| char('.') | |
| secondFraction(1, 9) | |
| } | |
| } | |
| optional { | |
| second() | |
| optional { | |
| char('.') | |
| secondFraction(1, 9) | |
| } | |
| } | |
| } | 
As the tests show, otherwise, 12:30:00.123 gets formatted as T1230.123, which is not valid: fractions of a second are only emitted together with the second value itself, even if it is zero.
… itself, even if it is zero.
| Thanks for your careful review! | 
| @Test | ||
| fun testBasicIso() { | ||
| val dateTimes = buildMap<LocalDateTime, Pair<String, Set<String>>> { | ||
| put(LocalDateTime(2008, 7, 5, 0, 0, 0, 0), ("20080705T0000" to setOf("20080705T0000"))) | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here, the same parsable representation is listed twice.
| optional { | ||
| second() | ||
| optional { | ||
| char('.') | ||
| secondFraction(1, 9) | ||
| } | ||
| } | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Now that I look at it, it's inconsistent to optionally omit seconds here on formatting. ISO allows parsing missing seconds, but does not omit them on its own.
| optional { | |
| second() | |
| optional { | |
| char('.') | |
| secondFraction(1, 9) | |
| } | |
| } | |
| second() | |
| optional { | |
| char('.') | |
| secondFraction(1, 9) | |
| } | 
| Sorry for the long delay. Everything should now be good to go, hopefully! | 
| * - `00000830T184300.5` | ||
| * - `-00010830T184300.123456789` | ||
| * | ||
| * When formatting, seconds always included. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
| * When formatting, seconds always included. | |
| * When formatting, seconds are always included. | 
| * See ISO-8601-1:2019, 5.4.2.1a), the version without the offset, together with | ||
| * [LocalDate.Formats.ISO_BASIC] and [LocalTime.Formats.ISO_BASIC]. | ||
| * | ||
| * @sample kotlinx.datetime.test.samples.LocalDateTimeSamples.Formats.basicIso | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LocalDateTimeSamples needs to include the actual sample. Also, please use consistent naming: for LocalDate and LocalTime, this sample is called isoBasic.
| } | ||
|  | ||
| @Test | ||
| fun testBasicIso() { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please also add the corresponding test for LocalTime.Formats.ISO_BASIC.
| @Test | ||
| fun testBasicIso() { | ||
| val dateTimes = buildMap<LocalDateTime, Pair<String, Set<String>>> { | ||
| put(LocalDateTime(2008, 7, 5, 0, 0, 0, 0), ("20080705T000000" to setOf("20080705t000000"))) | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please also test the scenarios of parsing a string which does not include the seconds.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Another thing that I noticed just now: please also add the new constants to this list:
kotlinx-datetime/core/common/src/format/DateTimeFormat.kt
Lines 154 to 171 in 90be731
| private val allFormatConstants: List<Pair<String, CachedFormatStructure<*>>> by lazy { | |
| fun unwrap(format: DateTimeFormat<*>): CachedFormatStructure<*> = (format as AbstractDateTimeFormat<*, *>).actualFormat | |
| // the formats are ordered vaguely by decreasing length, as the topmost among suitable ones is chosen. | |
| listOf( | |
| "${DateTimeFormatBuilder.WithDateTimeComponents::dateTimeComponents.name}(DateTimeComponents.Formats.RFC_1123)" to | |
| unwrap(DateTimeComponents.Formats.RFC_1123), | |
| "${DateTimeFormatBuilder.WithDateTimeComponents::dateTimeComponents.name}(DateTimeComponents.Formats.ISO_DATE_TIME_OFFSET)" to | |
| unwrap(DateTimeComponents.Formats.ISO_DATE_TIME_OFFSET), | |
| "${DateTimeFormatBuilder.WithDateTime::date.name}(LocalDateTime.Formats.ISO)" to unwrap(LocalDateTime.Formats.ISO), | |
| "${DateTimeFormatBuilder.WithDate::date.name}(LocalDate.Formats.ISO)" to unwrap(LocalDate.Formats.ISO), | |
| "${DateTimeFormatBuilder.WithDate::date.name}(LocalDate.Formats.ISO_BASIC)" to unwrap(LocalDate.Formats.ISO_BASIC), | |
| "${DateTimeFormatBuilder.WithTime::time.name}(LocalTime.Formats.ISO)" to unwrap(LocalTime.Formats.ISO), | |
| "${DateTimeFormatBuilder.WithUtcOffset::offset.name}(UtcOffset.Formats.ISO)" to unwrap(UtcOffset.Formats.ISO), | |
| "${DateTimeFormatBuilder.WithUtcOffset::offset.name}(UtcOffset.Formats.ISO_BASIC)" to unwrap(UtcOffset.Formats.ISO_BASIC), | |
| "${DateTimeFormatBuilder.WithUtcOffset::offset.name}(UtcOffset.Formats.FOUR_DIGITS)" to unwrap(UtcOffset.Formats.FOUR_DIGITS), | |
| "${DateTimeFormatBuilder.WithYearMonth::yearMonth.name}(YearMonth.Formats.ISO)" to unwrap(YearMonth.Formats.ISO), | |
| ) | |
| } | 
No description provided.