mirror of
https://github.com/tc39/test262.git
synced 2025-08-22 02:18:23 +02:00
Expands and updates (following the normative change in https://github.com/tc39/proposal-temporal/pull/3107, approved May 2025) the current coverage with cases where the number of seconds is incorrect, or rounded. Adds missing coverage for sub-minute offsets in strings passed to Temporal.ZonedDateTime.compare. Adds the unusual test case of Pacific/Niue on October 15, 1952, where the offset shifted by 20 seconds to a whole-minute boundary. Fixes a few minor errors in the existing tests such as a missing ] in a string, and the wrong sign for the offset in Africa/Monrovia in property bags.
234 lines
6.7 KiB
JavaScript
234 lines
6.7 KiB
JavaScript
// Copyright (C) 2021 Igalia, S.L. All rights reserved.
|
|
// This code is governed by the BSD license found in the LICENSE file.
|
|
|
|
/*---
|
|
esid: sec-temporal.zoneddatetime.from
|
|
description: Fuzzy matching behaviour with UTC offsets in ISO 8601 strings with named time zones and offset option
|
|
includes: [temporalHelpers.js]
|
|
features: [Temporal]
|
|
---*/
|
|
|
|
["use", "ignore", "prefer", "reject"].forEach((offset) => {
|
|
const result = Temporal.ZonedDateTime.from("1970-01-01T12:00-00:44:30[Africa/Monrovia]", { offset });
|
|
assert.sameValue(result.epochNanoseconds, 45870_000_000_000n, `accepts the exact offset string (offset: ${offset})`);
|
|
assert.sameValue(result.offset, "-00:44:30", "offset property is correct");
|
|
});
|
|
|
|
["use", "ignore", "prefer", "reject"].forEach((offset) => {
|
|
const result = Temporal.ZonedDateTime.from("1970-01-01T12:00-00:44:30.000000000[Africa/Monrovia]", { offset });
|
|
assert.sameValue(
|
|
result.epochNanoseconds,
|
|
45870_000_000_000n,
|
|
`accepts trailing zeroes after ISO string offset (offset: ${offset})`
|
|
);
|
|
assert.sameValue(result.offset, "-00:44:30", "offset property removes trailing zeroes from input");
|
|
});
|
|
|
|
assert.throws(
|
|
RangeError,
|
|
() => Temporal.ZonedDateTime.from("1970-01-01T00:00-00:44:30[-00:45]", { offset: "reject" }),
|
|
"minute rounding not supported for offset time zones"
|
|
);
|
|
|
|
const str = "1970-01-01T12:00-00:45[Africa/Monrovia]";
|
|
|
|
["ignore", "prefer", "reject"].forEach((offset) => {
|
|
const result = Temporal.ZonedDateTime.from(str, { offset });
|
|
assert.sameValue(
|
|
result.epochNanoseconds,
|
|
45870_000_000_000n,
|
|
`accepts the offset string rounded to minutes (offset=${offset})`
|
|
);
|
|
assert.sameValue(result.offset, "-00:44:30", "offset property is still the full precision");
|
|
TemporalHelpers.assertPlainDateTime(
|
|
result.toPlainDateTime(),
|
|
1970,
|
|
1,
|
|
"M01",
|
|
1,
|
|
12,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
"wall time is preserved"
|
|
);
|
|
});
|
|
|
|
const result = Temporal.ZonedDateTime.from(str, { offset: "use" });
|
|
assert.sameValue(
|
|
result.epochNanoseconds,
|
|
45900_000_000_000n,
|
|
"prioritizes the offset string with HH:MM precision when offset=use"
|
|
);
|
|
assert.sameValue(result.offset, "-00:44:30", "offset property is still the full precision");
|
|
TemporalHelpers.assertPlainDateTime(
|
|
result.toPlainDateTime(),
|
|
1970,
|
|
1,
|
|
"M01",
|
|
1,
|
|
12,
|
|
0,
|
|
30,
|
|
0,
|
|
0,
|
|
0,
|
|
"wall time is shifted by the difference between exact and rounded offset"
|
|
);
|
|
|
|
const wrongSeconds = "1970-01-01T12-00:44:40[Africa/Monrovia]";
|
|
const roundedSeconds = "1970-01-01T12-00:45:00[Africa/Monrovia]";
|
|
|
|
const useResultWrongSeconds = Temporal.ZonedDateTime.from(wrongSeconds, { offset: "use" });
|
|
assert.sameValue(
|
|
useResultWrongSeconds.epochNanoseconds,
|
|
45880_000_000_000n,
|
|
"uses the wrong offset with HH:MM:SS precision when offset=use"
|
|
);
|
|
assert.sameValue(useResultWrongSeconds.offset, "-00:44:30", "offset property is still the full precision");
|
|
TemporalHelpers.assertPlainDateTime(
|
|
useResultWrongSeconds.toPlainDateTime(),
|
|
1970,
|
|
1,
|
|
"M01",
|
|
1,
|
|
12,
|
|
0,
|
|
10,
|
|
0,
|
|
0,
|
|
0,
|
|
"wall time is shifted by the difference between exact and given offset"
|
|
);
|
|
|
|
|
|
const useResultRoundedSeconds = Temporal.ZonedDateTime.from(roundedSeconds, { offset: "use" });
|
|
assert.sameValue(
|
|
useResultRoundedSeconds.epochNanoseconds,
|
|
45900_000_000_000n,
|
|
"uses the rounded offset with HH:MM:SS precision when offset=use"
|
|
);
|
|
assert.sameValue(useResultRoundedSeconds.offset, "-00:44:30", "offset property is still the full precision");
|
|
TemporalHelpers.assertPlainDateTime(
|
|
useResultRoundedSeconds.toPlainDateTime(),
|
|
1970,
|
|
1,
|
|
"M01",
|
|
1,
|
|
12,
|
|
0,
|
|
30,
|
|
0,
|
|
0,
|
|
0,
|
|
"wall time is shifted by the difference between exact and given offset"
|
|
);
|
|
|
|
["ignore", "prefer"].forEach((offset) => {
|
|
const resultWrongSeconds = Temporal.ZonedDateTime.from(wrongSeconds, { offset });
|
|
assert.sameValue(
|
|
resultWrongSeconds.epochNanoseconds,
|
|
45870_000_000_000n,
|
|
`does not use the offset string with wrong :SS (offset=${offset})`
|
|
);
|
|
assert.sameValue(resultWrongSeconds.offset, "-00:44:30", "offset property is still the full precision");
|
|
TemporalHelpers.assertPlainDateTime(
|
|
resultWrongSeconds.toPlainDateTime(),
|
|
1970,
|
|
1,
|
|
"M01",
|
|
1,
|
|
12,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
"wall time is preserved"
|
|
);
|
|
|
|
const resultRoundedSeconds = Temporal.ZonedDateTime.from(roundedSeconds, { offset });
|
|
assert.sameValue(
|
|
resultRoundedSeconds.epochNanoseconds,
|
|
45870_000_000_000n,
|
|
`does not use the offset string with rounded HH:MM:SS (offset=${offset})`
|
|
);
|
|
assert.sameValue(resultRoundedSeconds.offset, "-00:44:30", "offset property is still the full precision");
|
|
TemporalHelpers.assertPlainDateTime(
|
|
resultRoundedSeconds.toPlainDateTime(),
|
|
1970,
|
|
1,
|
|
"M01",
|
|
1,
|
|
12,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
"wall time is preserved"
|
|
);
|
|
});
|
|
|
|
assert.throws(
|
|
RangeError,
|
|
() => Temporal.ZonedDateTime.from(wrongSeconds, { offset: "reject" }),
|
|
"wrong :SS not accepted in string offset (offset=reject)"
|
|
);
|
|
|
|
assert.throws(
|
|
RangeError,
|
|
() => Temporal.ZonedDateTime.from(roundedSeconds, { offset: "reject" }),
|
|
"rounded HH:MM:SS not accepted in string offset (offset=reject)"
|
|
);
|
|
|
|
const properties = { year: 1970, month: 1, day: 1, hour: 12, offset: "-00:45", timeZone: "Africa/Monrovia" };
|
|
|
|
["ignore", "prefer"].forEach((offset) => {
|
|
const result = Temporal.ZonedDateTime.from(properties, { offset });
|
|
assert.sameValue(
|
|
result.epochNanoseconds,
|
|
45870_000_000_000n,
|
|
`no fuzzy matching is done on offset in property bag (offset=${offset})`
|
|
);
|
|
});
|
|
|
|
const result2 = Temporal.ZonedDateTime.from(properties, { offset: "use" });
|
|
assert.sameValue(
|
|
result2.epochNanoseconds,
|
|
45900_000_000_000n,
|
|
"no fuzzy matching is done on offset in property bag (offset=use)"
|
|
);
|
|
|
|
assert.throws(
|
|
RangeError,
|
|
() => Temporal.ZonedDateTime.from(properties, { offset: "reject" }),
|
|
"no fuzzy matching is done on offset in property bag (offset=reject)"
|
|
);
|
|
|
|
// Pacific/Niue edge case
|
|
|
|
const reference = -543069621_000_000_000n;
|
|
|
|
assert.sameValue(
|
|
Temporal.ZonedDateTime.from("1952-10-15T23:59:59-11:19:40[Pacific/Niue]").epochNanoseconds,
|
|
reference,
|
|
"-11:19:40 is accepted as -11:19:40 in Pacific/Niue edge case"
|
|
);
|
|
assert.sameValue(
|
|
Temporal.ZonedDateTime.from("1952-10-15T23:59:59-11:20[Pacific/Niue]").epochNanoseconds,
|
|
reference,
|
|
"-11:20 matches the first candidate -11:19:40 in the Pacific/Niue edge case"
|
|
);
|
|
assert.sameValue(
|
|
Temporal.ZonedDateTime.from("1952-10-15T23:59:59-11:20:00[Pacific/Niue]").epochNanoseconds,
|
|
reference + 20_000_000_000n,
|
|
"-11:20:00 is accepted as -11:20:00 in the Pacific/Niue edge case"
|
|
);
|
|
assert.throws(
|
|
RangeError, () => Temporal.ZonedDateTime.from("1952-10-15T23:59:59-11:19:50[Pacific/Niue]"),
|
|
"wrong :SS not accepted in the Pacific/Niue edge case"
|
|
);
|