2021-07-16 15:25:55 +02:00
|
|
|
// Copyright (C) 2021 Igalia, S.L. All rights reserved.
|
|
|
|
// This code is governed by the BSD license found in the LICENSE file.
|
|
|
|
/*---
|
|
|
|
description: |
|
|
|
|
This defines helper objects and functions for testing Temporal.
|
|
|
|
defines: [TemporalHelpers]
|
|
|
|
features: [Symbol.species, Symbol.iterator, Temporal]
|
|
|
|
---*/
|
|
|
|
|
2023-06-07 10:59:16 +02:00
|
|
|
const ASCII_IDENTIFIER = /^[$_a-zA-Z][$_a-zA-Z0-9]*$/u;
|
2023-04-22 01:59:09 +02:00
|
|
|
|
2022-09-20 21:53:57 +02:00
|
|
|
function formatPropertyName(propertyKey, objectName = "") {
|
|
|
|
switch (typeof propertyKey) {
|
|
|
|
case "symbol":
|
|
|
|
if (Symbol.keyFor(propertyKey) !== undefined) {
|
|
|
|
return `${objectName}[Symbol.for('${Symbol.keyFor(propertyKey)}')]`;
|
|
|
|
} else if (propertyKey.description.startsWith('Symbol.')) {
|
|
|
|
return `${objectName}[${propertyKey.description}]`;
|
|
|
|
} else {
|
|
|
|
return `${objectName}[Symbol('${propertyKey.description}')]`
|
|
|
|
}
|
2023-04-22 01:59:09 +02:00
|
|
|
case "string":
|
|
|
|
if (propertyKey !== String(Number(propertyKey))) {
|
2023-06-07 10:59:16 +02:00
|
|
|
if (ASCII_IDENTIFIER.test(propertyKey)) {
|
2023-04-22 01:59:09 +02:00
|
|
|
return objectName ? `${objectName}.${propertyKey}` : propertyKey;
|
|
|
|
}
|
|
|
|
return `${objectName}['${propertyKey.replace(/'/g, "\\'")}']`
|
|
|
|
}
|
|
|
|
// fall through
|
2022-09-20 21:53:57 +02:00
|
|
|
default:
|
2023-04-22 01:59:09 +02:00
|
|
|
// integer or string integer-index
|
|
|
|
return `${objectName}[${propertyKey}]`;
|
2022-09-20 21:53:57 +02:00
|
|
|
}
|
|
|
|
}
|
2023-04-22 01:59:09 +02:00
|
|
|
|
2022-11-26 04:20:11 +01:00
|
|
|
const SKIP_SYMBOL = Symbol("Skip");
|
2022-09-20 21:53:57 +02:00
|
|
|
|
2021-07-16 15:25:55 +02:00
|
|
|
var TemporalHelpers = {
|
2023-08-21 23:20:34 +02:00
|
|
|
/*
|
|
|
|
* Codes and maximum lengths of months in the ISO 8601 calendar.
|
|
|
|
*/
|
|
|
|
ISOMonths: [
|
|
|
|
{ month: 1, monthCode: "M01", daysInMonth: 31 },
|
|
|
|
{ month: 2, monthCode: "M02", daysInMonth: 29 },
|
|
|
|
{ month: 3, monthCode: "M03", daysInMonth: 31 },
|
|
|
|
{ month: 4, monthCode: "M04", daysInMonth: 30 },
|
|
|
|
{ month: 5, monthCode: "M05", daysInMonth: 31 },
|
|
|
|
{ month: 6, monthCode: "M06", daysInMonth: 30 },
|
|
|
|
{ month: 7, monthCode: "M07", daysInMonth: 31 },
|
|
|
|
{ month: 8, monthCode: "M08", daysInMonth: 31 },
|
|
|
|
{ month: 9, monthCode: "M09", daysInMonth: 30 },
|
|
|
|
{ month: 10, monthCode: "M10", daysInMonth: 31 },
|
|
|
|
{ month: 11, monthCode: "M11", daysInMonth: 30 },
|
|
|
|
{ month: 12, monthCode: "M12", daysInMonth: 31 }
|
|
|
|
],
|
|
|
|
|
2021-07-20 21:26:29 +02:00
|
|
|
/*
|
|
|
|
* assertDuration(duration, years, ..., nanoseconds[, description]):
|
|
|
|
*
|
|
|
|
* Shorthand for asserting that each field of a Temporal.Duration is equal to
|
|
|
|
* an expected value.
|
|
|
|
*/
|
|
|
|
assertDuration(duration, years, months, weeks, days, hours, minutes, seconds, milliseconds, microseconds, nanoseconds, description = "") {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(duration instanceof Temporal.Duration, `${prefix}instanceof`);
|
|
|
|
assert.sameValue(duration.years, years, `${prefix}years result:`);
|
|
|
|
assert.sameValue(duration.months, months, `${prefix}months result:`);
|
|
|
|
assert.sameValue(duration.weeks, weeks, `${prefix}weeks result:`);
|
|
|
|
assert.sameValue(duration.days, days, `${prefix}days result:`);
|
|
|
|
assert.sameValue(duration.hours, hours, `${prefix}hours result:`);
|
|
|
|
assert.sameValue(duration.minutes, minutes, `${prefix}minutes result:`);
|
|
|
|
assert.sameValue(duration.seconds, seconds, `${prefix}seconds result:`);
|
|
|
|
assert.sameValue(duration.milliseconds, milliseconds, `${prefix}milliseconds result:`);
|
|
|
|
assert.sameValue(duration.microseconds, microseconds, `${prefix}microseconds result:`);
|
|
|
|
assert.sameValue(duration.nanoseconds, nanoseconds, `${prefix}nanoseconds result`);
|
2021-07-20 21:26:29 +02:00
|
|
|
},
|
|
|
|
|
2023-07-01 23:08:06 +02:00
|
|
|
/*
|
|
|
|
* assertDateDuration(duration, years, months, weeks, days, [, description]):
|
|
|
|
*
|
|
|
|
* Shorthand for asserting that each date field of a Temporal.Duration is
|
|
|
|
* equal to an expected value.
|
|
|
|
*/
|
|
|
|
assertDateDuration(duration, years, months, weeks, days, description = "") {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(duration instanceof Temporal.Duration, `${prefix}instanceof`);
|
|
|
|
assert.sameValue(duration.years, years, `${prefix}years result:`);
|
|
|
|
assert.sameValue(duration.months, months, `${prefix}months result:`);
|
|
|
|
assert.sameValue(duration.weeks, weeks, `${prefix}weeks result:`);
|
|
|
|
assert.sameValue(duration.days, days, `${prefix}days result:`);
|
|
|
|
assert.sameValue(duration.hours, 0, `${prefix}hours result should be zero:`);
|
|
|
|
assert.sameValue(duration.minutes, 0, `${prefix}minutes result should be zero:`);
|
|
|
|
assert.sameValue(duration.seconds, 0, `${prefix}seconds result should be zero:`);
|
|
|
|
assert.sameValue(duration.milliseconds, 0, `${prefix}milliseconds result should be zero:`);
|
|
|
|
assert.sameValue(duration.microseconds, 0, `${prefix}microseconds result should be zero:`);
|
|
|
|
assert.sameValue(duration.nanoseconds, 0, `${prefix}nanoseconds result should be zero:`);
|
2023-07-01 23:08:06 +02:00
|
|
|
},
|
|
|
|
|
2021-11-04 22:06:23 +01:00
|
|
|
/*
|
|
|
|
* assertDurationsEqual(actual, expected[, description]):
|
|
|
|
*
|
|
|
|
* Shorthand for asserting that each field of a Temporal.Duration is equal to
|
|
|
|
* the corresponding field in another Temporal.Duration.
|
|
|
|
*/
|
|
|
|
assertDurationsEqual(actual, expected, description = "") {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(expected instanceof Temporal.Duration, `${prefix}expected value should be a Temporal.Duration`);
|
2021-11-04 22:06:23 +01:00
|
|
|
TemporalHelpers.assertDuration(actual, expected.years, expected.months, expected.weeks, expected.days, expected.hours, expected.minutes, expected.seconds, expected.milliseconds, expected.microseconds, expected.nanoseconds, description);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* assertInstantsEqual(actual, expected[, description]):
|
|
|
|
*
|
|
|
|
* Shorthand for asserting that two Temporal.Instants are of the correct type
|
|
|
|
* and equal according to their equals() methods.
|
|
|
|
*/
|
|
|
|
assertInstantsEqual(actual, expected, description = "") {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(expected instanceof Temporal.Instant, `${prefix}expected value should be a Temporal.Instant`);
|
|
|
|
assert(actual instanceof Temporal.Instant, `${prefix}instanceof`);
|
|
|
|
assert(actual.equals(expected), `${prefix}equals method`);
|
2021-11-04 22:06:23 +01:00
|
|
|
},
|
|
|
|
|
2021-07-20 21:26:29 +02:00
|
|
|
/*
|
|
|
|
* assertPlainDate(date, year, ..., nanosecond[, description[, era, eraYear]]):
|
|
|
|
*
|
|
|
|
* Shorthand for asserting that each field of a Temporal.PlainDate is equal to
|
|
|
|
* an expected value. (Except the `calendar` property, since callers may want
|
|
|
|
* to assert either object equality with an object they put in there, or the
|
2023-02-17 21:28:55 +01:00
|
|
|
* value of date.calendarId.)
|
2021-07-20 21:26:29 +02:00
|
|
|
*/
|
|
|
|
assertPlainDate(date, year, month, monthCode, day, description = "", era = undefined, eraYear = undefined) {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(date instanceof Temporal.PlainDate, `${prefix}instanceof`);
|
|
|
|
assert.sameValue(date.era, era, `${prefix}era result:`);
|
|
|
|
assert.sameValue(date.eraYear, eraYear, `${prefix}eraYear result:`);
|
|
|
|
assert.sameValue(date.year, year, `${prefix}year result:`);
|
|
|
|
assert.sameValue(date.month, month, `${prefix}month result:`);
|
|
|
|
assert.sameValue(date.monthCode, monthCode, `${prefix}monthCode result:`);
|
|
|
|
assert.sameValue(date.day, day, `${prefix}day result:`);
|
2021-07-20 21:26:29 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* assertPlainDateTime(datetime, year, ..., nanosecond[, description[, era, eraYear]]):
|
|
|
|
*
|
|
|
|
* Shorthand for asserting that each field of a Temporal.PlainDateTime is
|
|
|
|
* equal to an expected value. (Except the `calendar` property, since callers
|
|
|
|
* may want to assert either object equality with an object they put in there,
|
2023-02-17 21:28:55 +01:00
|
|
|
* or the value of datetime.calendarId.)
|
2021-07-20 21:26:29 +02:00
|
|
|
*/
|
|
|
|
assertPlainDateTime(datetime, year, month, monthCode, day, hour, minute, second, millisecond, microsecond, nanosecond, description = "", era = undefined, eraYear = undefined) {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(datetime instanceof Temporal.PlainDateTime, `${prefix}instanceof`);
|
|
|
|
assert.sameValue(datetime.era, era, `${prefix}era result:`);
|
|
|
|
assert.sameValue(datetime.eraYear, eraYear, `${prefix}eraYear result:`);
|
|
|
|
assert.sameValue(datetime.year, year, `${prefix}year result:`);
|
|
|
|
assert.sameValue(datetime.month, month, `${prefix}month result:`);
|
|
|
|
assert.sameValue(datetime.monthCode, monthCode, `${prefix}monthCode result:`);
|
|
|
|
assert.sameValue(datetime.day, day, `${prefix}day result:`);
|
|
|
|
assert.sameValue(datetime.hour, hour, `${prefix}hour result:`);
|
|
|
|
assert.sameValue(datetime.minute, minute, `${prefix}minute result:`);
|
|
|
|
assert.sameValue(datetime.second, second, `${prefix}second result:`);
|
|
|
|
assert.sameValue(datetime.millisecond, millisecond, `${prefix}millisecond result:`);
|
|
|
|
assert.sameValue(datetime.microsecond, microsecond, `${prefix}microsecond result:`);
|
|
|
|
assert.sameValue(datetime.nanosecond, nanosecond, `${prefix}nanosecond result:`);
|
2021-07-20 21:26:29 +02:00
|
|
|
},
|
|
|
|
|
2021-11-04 22:06:23 +01:00
|
|
|
/*
|
|
|
|
* assertPlainDateTimesEqual(actual, expected[, description]):
|
|
|
|
*
|
|
|
|
* Shorthand for asserting that two Temporal.PlainDateTimes are of the correct
|
|
|
|
* type, equal according to their equals() methods, and additionally that
|
2023-02-17 21:28:55 +01:00
|
|
|
* their calendar internal slots are the same value.
|
2021-11-04 22:06:23 +01:00
|
|
|
*/
|
|
|
|
assertPlainDateTimesEqual(actual, expected, description = "") {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(expected instanceof Temporal.PlainDateTime, `${prefix}expected value should be a Temporal.PlainDateTime`);
|
|
|
|
assert(actual instanceof Temporal.PlainDateTime, `${prefix}instanceof`);
|
|
|
|
assert(actual.equals(expected), `${prefix}equals method`);
|
2023-02-17 21:28:55 +01:00
|
|
|
assert.sameValue(
|
2024-06-04 17:35:22 +02:00
|
|
|
actual.calendarId,
|
|
|
|
expected.calendarId,
|
2024-01-18 00:26:55 +01:00
|
|
|
`${prefix}calendar same value:`
|
2023-02-17 21:28:55 +01:00
|
|
|
);
|
2021-11-04 22:06:23 +01:00
|
|
|
},
|
|
|
|
|
2021-07-20 21:26:29 +02:00
|
|
|
/*
|
2021-09-29 21:48:22 +02:00
|
|
|
* assertPlainMonthDay(monthDay, monthCode, day[, description [, referenceISOYear]]):
|
2021-07-20 21:26:29 +02:00
|
|
|
*
|
|
|
|
* Shorthand for asserting that each field of a Temporal.PlainMonthDay is
|
|
|
|
* equal to an expected value. (Except the `calendar` property, since callers
|
|
|
|
* may want to assert either object equality with an object they put in there,
|
2023-02-17 21:28:55 +01:00
|
|
|
* or the value of monthDay.calendarId().)
|
2021-07-20 21:26:29 +02:00
|
|
|
*/
|
2021-09-29 21:48:22 +02:00
|
|
|
assertPlainMonthDay(monthDay, monthCode, day, description = "", referenceISOYear = 1972) {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(monthDay instanceof Temporal.PlainMonthDay, `${prefix}instanceof`);
|
|
|
|
assert.sameValue(monthDay.monthCode, monthCode, `${prefix}monthCode result:`);
|
|
|
|
assert.sameValue(monthDay.day, day, `${prefix}day result:`);
|
2024-06-04 17:35:22 +02:00
|
|
|
const isoYear = Number(monthDay.toString({ calendarName: "always" }).split("-")[0]);
|
|
|
|
assert.sameValue(isoYear, referenceISOYear, `${prefix}referenceISOYear result:`);
|
2021-07-20 21:26:29 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* assertPlainTime(time, hour, ..., nanosecond[, description]):
|
|
|
|
*
|
|
|
|
* Shorthand for asserting that each field of a Temporal.PlainTime is equal to
|
|
|
|
* an expected value.
|
|
|
|
*/
|
|
|
|
assertPlainTime(time, hour, minute, second, millisecond, microsecond, nanosecond, description = "") {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(time instanceof Temporal.PlainTime, `${prefix}instanceof`);
|
|
|
|
assert.sameValue(time.hour, hour, `${prefix}hour result:`);
|
|
|
|
assert.sameValue(time.minute, minute, `${prefix}minute result:`);
|
|
|
|
assert.sameValue(time.second, second, `${prefix}second result:`);
|
|
|
|
assert.sameValue(time.millisecond, millisecond, `${prefix}millisecond result:`);
|
|
|
|
assert.sameValue(time.microsecond, microsecond, `${prefix}microsecond result:`);
|
|
|
|
assert.sameValue(time.nanosecond, nanosecond, `${prefix}nanosecond result:`);
|
2021-07-20 21:26:29 +02:00
|
|
|
},
|
|
|
|
|
2021-11-04 22:06:23 +01:00
|
|
|
/*
|
|
|
|
* assertPlainTimesEqual(actual, expected[, description]):
|
|
|
|
*
|
|
|
|
* Shorthand for asserting that two Temporal.PlainTimes are of the correct
|
|
|
|
* type and equal according to their equals() methods.
|
|
|
|
*/
|
|
|
|
assertPlainTimesEqual(actual, expected, description = "") {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(expected instanceof Temporal.PlainTime, `${prefix}expected value should be a Temporal.PlainTime`);
|
|
|
|
assert(actual instanceof Temporal.PlainTime, `${prefix}instanceof`);
|
|
|
|
assert(actual.equals(expected), `${prefix}equals method`);
|
2021-11-04 22:06:23 +01:00
|
|
|
},
|
|
|
|
|
2021-07-20 21:26:29 +02:00
|
|
|
/*
|
2022-04-20 00:43:18 +02:00
|
|
|
* assertPlainYearMonth(yearMonth, year, month, monthCode[, description[, era, eraYear, referenceISODay]]):
|
2021-07-20 21:26:29 +02:00
|
|
|
*
|
|
|
|
* Shorthand for asserting that each field of a Temporal.PlainYearMonth is
|
|
|
|
* equal to an expected value. (Except the `calendar` property, since callers
|
|
|
|
* may want to assert either object equality with an object they put in there,
|
2023-02-17 21:28:55 +01:00
|
|
|
* or the value of yearMonth.calendarId.)
|
2021-07-20 21:26:29 +02:00
|
|
|
*/
|
2022-04-20 00:43:18 +02:00
|
|
|
assertPlainYearMonth(yearMonth, year, month, monthCode, description = "", era = undefined, eraYear = undefined, referenceISODay = 1) {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(yearMonth instanceof Temporal.PlainYearMonth, `${prefix}instanceof`);
|
|
|
|
assert.sameValue(yearMonth.era, era, `${prefix}era result:`);
|
|
|
|
assert.sameValue(yearMonth.eraYear, eraYear, `${prefix}eraYear result:`);
|
|
|
|
assert.sameValue(yearMonth.year, year, `${prefix}year result:`);
|
|
|
|
assert.sameValue(yearMonth.month, month, `${prefix}month result:`);
|
|
|
|
assert.sameValue(yearMonth.monthCode, monthCode, `${prefix}monthCode result:`);
|
2024-06-04 17:35:22 +02:00
|
|
|
const isoDay = Number(yearMonth.toString({ calendarName: "always" }).slice(1).split('-')[2].slice(0, 2));
|
|
|
|
assert.sameValue(isoDay, referenceISODay, `${prefix}referenceISODay result:`);
|
2021-07-20 21:26:29 +02:00
|
|
|
},
|
2021-09-29 21:48:22 +02:00
|
|
|
|
2021-11-04 22:06:23 +01:00
|
|
|
/*
|
|
|
|
* assertZonedDateTimesEqual(actual, expected[, description]):
|
|
|
|
*
|
|
|
|
* Shorthand for asserting that two Temporal.ZonedDateTimes are of the correct
|
|
|
|
* type, equal according to their equals() methods, and additionally that
|
2023-02-17 21:28:55 +01:00
|
|
|
* their time zones and calendar internal slots are the same value.
|
2021-11-04 22:06:23 +01:00
|
|
|
*/
|
|
|
|
assertZonedDateTimesEqual(actual, expected, description = "") {
|
2024-01-18 00:26:55 +01:00
|
|
|
const prefix = description ? `${description}: ` : "";
|
|
|
|
assert(expected instanceof Temporal.ZonedDateTime, `${prefix}expected value should be a Temporal.ZonedDateTime`);
|
|
|
|
assert(actual instanceof Temporal.ZonedDateTime, `${prefix}instanceof`);
|
|
|
|
assert(actual.equals(expected), `${prefix}equals method`);
|
|
|
|
assert.sameValue(actual.timeZone, expected.timeZone, `${prefix}time zone same value:`);
|
2023-02-17 21:28:55 +01:00
|
|
|
assert.sameValue(
|
2024-06-04 17:35:22 +02:00
|
|
|
actual.calendarId,
|
|
|
|
expected.calendarId,
|
2024-01-18 00:26:55 +01:00
|
|
|
`${prefix}calendar same value:`
|
2023-02-17 21:28:55 +01:00
|
|
|
);
|
2021-11-04 22:06:23 +01:00
|
|
|
},
|
|
|
|
|
2021-09-29 21:48:22 +02:00
|
|
|
/*
|
|
|
|
* assertUnreachable(description):
|
|
|
|
*
|
2024-06-04 12:56:40 +02:00
|
|
|
* Helper for asserting that code is not executed.
|
2021-09-29 21:48:22 +02:00
|
|
|
*/
|
|
|
|
assertUnreachable(description) {
|
|
|
|
let message = "This code should not be executed";
|
|
|
|
if (description) {
|
|
|
|
message = `${message}: ${description}`;
|
|
|
|
}
|
|
|
|
throw new Test262Error(message);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* checkPlainDateTimeConversionFastPath(func):
|
|
|
|
*
|
|
|
|
* ToTemporalDate and ToTemporalTime should both, if given a
|
|
|
|
* Temporal.PlainDateTime instance, convert to the desired type by reading the
|
|
|
|
* PlainDateTime's internal slots, rather than calling any getters.
|
|
|
|
*
|
2024-06-04 12:49:30 +02:00
|
|
|
* func(datetime) is the actual operation to test, that must
|
2021-09-29 21:48:22 +02:00
|
|
|
* internally call the abstract operation ToTemporalDate or ToTemporalTime.
|
2024-06-04 12:49:30 +02:00
|
|
|
* It is passed a Temporal.PlainDateTime instance.
|
2021-09-29 21:48:22 +02:00
|
|
|
*/
|
2022-06-08 18:02:01 +02:00
|
|
|
checkPlainDateTimeConversionFastPath(func, message = "checkPlainDateTimeConversionFastPath") {
|
2021-09-29 21:48:22 +02:00
|
|
|
const actual = [];
|
|
|
|
const expected = [];
|
|
|
|
|
2024-06-04 11:25:38 +02:00
|
|
|
const calendar = "iso8601";
|
2021-09-29 21:48:22 +02:00
|
|
|
const datetime = new Temporal.PlainDateTime(2000, 5, 2, 12, 34, 56, 987, 654, 321, calendar);
|
|
|
|
const prototypeDescrs = Object.getOwnPropertyDescriptors(Temporal.PlainDateTime.prototype);
|
|
|
|
["year", "month", "monthCode", "day", "hour", "minute", "second", "millisecond", "microsecond", "nanosecond"].forEach((property) => {
|
|
|
|
Object.defineProperty(datetime, property, {
|
|
|
|
get() {
|
2022-09-20 21:53:57 +02:00
|
|
|
actual.push(`get ${formatPropertyName(property)}`);
|
2021-09-29 21:48:22 +02:00
|
|
|
const value = prototypeDescrs[property].get.call(this);
|
|
|
|
return {
|
|
|
|
toString() {
|
2022-09-20 21:53:57 +02:00
|
|
|
actual.push(`toString ${formatPropertyName(property)}`);
|
2021-09-29 21:48:22 +02:00
|
|
|
return value.toString();
|
|
|
|
},
|
|
|
|
valueOf() {
|
2022-09-20 21:53:57 +02:00
|
|
|
actual.push(`valueOf ${formatPropertyName(property)}`);
|
2021-09-29 21:48:22 +02:00
|
|
|
return value;
|
|
|
|
},
|
|
|
|
};
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
Object.defineProperty(datetime, "calendar", {
|
|
|
|
get() {
|
|
|
|
actual.push("get calendar");
|
|
|
|
return calendar;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2024-06-04 12:49:30 +02:00
|
|
|
func(datetime);
|
2022-06-08 18:02:01 +02:00
|
|
|
assert.compareArray(actual, expected, `${message}: property getters not called`);
|
2021-09-29 21:48:22 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that an options bag that accepts units written in the singular form,
|
|
|
|
* also accepts the same units written in the plural form.
|
|
|
|
* func(unit) should call the method with the appropriate options bag
|
|
|
|
* containing unit as a value. This will be called twice for each element of
|
|
|
|
* validSingularUnits, once with singular and once with plural, and the
|
|
|
|
* results of each pair should be the same (whether a Temporal object or a
|
|
|
|
* primitive value.)
|
|
|
|
*/
|
|
|
|
checkPluralUnitsAccepted(func, validSingularUnits) {
|
|
|
|
const plurals = {
|
|
|
|
year: 'years',
|
|
|
|
month: 'months',
|
|
|
|
week: 'weeks',
|
|
|
|
day: 'days',
|
|
|
|
hour: 'hours',
|
|
|
|
minute: 'minutes',
|
|
|
|
second: 'seconds',
|
|
|
|
millisecond: 'milliseconds',
|
|
|
|
microsecond: 'microseconds',
|
|
|
|
nanosecond: 'nanoseconds',
|
|
|
|
};
|
|
|
|
|
|
|
|
validSingularUnits.forEach((unit) => {
|
|
|
|
const singularValue = func(unit);
|
|
|
|
const pluralValue = func(plurals[unit]);
|
2021-11-04 22:06:23 +01:00
|
|
|
const desc = `Plural ${plurals[unit]} produces the same result as singular ${unit}`;
|
2021-09-29 21:48:22 +02:00
|
|
|
if (singularValue instanceof Temporal.Duration) {
|
2021-11-04 22:06:23 +01:00
|
|
|
TemporalHelpers.assertDurationsEqual(pluralValue, singularValue, desc);
|
|
|
|
} else if (singularValue instanceof Temporal.Instant) {
|
|
|
|
TemporalHelpers.assertInstantsEqual(pluralValue, singularValue, desc);
|
|
|
|
} else if (singularValue instanceof Temporal.PlainDateTime) {
|
|
|
|
TemporalHelpers.assertPlainDateTimesEqual(pluralValue, singularValue, desc);
|
|
|
|
} else if (singularValue instanceof Temporal.PlainTime) {
|
|
|
|
TemporalHelpers.assertPlainTimesEqual(pluralValue, singularValue, desc);
|
|
|
|
} else if (singularValue instanceof Temporal.ZonedDateTime) {
|
|
|
|
TemporalHelpers.assertZonedDateTimesEqual(pluralValue, singularValue, desc);
|
2021-09-29 21:48:22 +02:00
|
|
|
} else {
|
|
|
|
assert.sameValue(pluralValue, singularValue);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* checkRoundingIncrementOptionWrongType(checkFunc, assertTrueResultFunc, assertObjectResultFunc):
|
|
|
|
*
|
|
|
|
* Checks the type handling of the roundingIncrement option.
|
|
|
|
* checkFunc(roundingIncrement) is a function which takes the value of
|
|
|
|
* roundingIncrement to test, and calls the method under test with it,
|
|
|
|
* returning the result. assertTrueResultFunc(result, description) should
|
|
|
|
* assert that result is the expected result with roundingIncrement: true, and
|
|
|
|
* assertObjectResultFunc(result, description) should assert that result is
|
|
|
|
* the expected result with roundingIncrement being an object with a valueOf()
|
|
|
|
* method.
|
|
|
|
*/
|
|
|
|
checkRoundingIncrementOptionWrongType(checkFunc, assertTrueResultFunc, assertObjectResultFunc) {
|
|
|
|
// null converts to 0, which is out of range
|
|
|
|
assert.throws(RangeError, () => checkFunc(null), "null");
|
|
|
|
// Booleans convert to either 0 or 1, and 1 is allowed
|
|
|
|
const trueResult = checkFunc(true);
|
|
|
|
assertTrueResultFunc(trueResult, "true");
|
|
|
|
assert.throws(RangeError, () => checkFunc(false), "false");
|
|
|
|
// Symbols and BigInts cannot convert to numbers
|
|
|
|
assert.throws(TypeError, () => checkFunc(Symbol()), "symbol");
|
|
|
|
assert.throws(TypeError, () => checkFunc(2n), "bigint");
|
|
|
|
|
|
|
|
// Objects prefer their valueOf() methods when converting to a number
|
|
|
|
assert.throws(RangeError, () => checkFunc({}), "plain object");
|
|
|
|
|
|
|
|
const expected = [
|
|
|
|
"get roundingIncrement.valueOf",
|
|
|
|
"call roundingIncrement.valueOf",
|
|
|
|
];
|
|
|
|
const actual = [];
|
|
|
|
const observer = TemporalHelpers.toPrimitiveObserver(actual, 2, "roundingIncrement");
|
|
|
|
const objectResult = checkFunc(observer);
|
|
|
|
assertObjectResultFunc(objectResult, "object with valueOf");
|
|
|
|
assert.compareArray(actual, expected, "order of operations");
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* checkStringOptionWrongType(propertyName, value, checkFunc, assertFunc):
|
|
|
|
*
|
|
|
|
* Checks the type handling of a string option, of which there are several in
|
|
|
|
* Temporal.
|
|
|
|
* propertyName is the name of the option, and value is the value that
|
|
|
|
* assertFunc should expect it to have.
|
|
|
|
* checkFunc(value) is a function which takes the value of the option to test,
|
|
|
|
* and calls the method under test with it, returning the result.
|
|
|
|
* assertFunc(result, description) should assert that result is the expected
|
|
|
|
* result with the option value being an object with a toString() method
|
|
|
|
* which returns the given value.
|
|
|
|
*/
|
|
|
|
checkStringOptionWrongType(propertyName, value, checkFunc, assertFunc) {
|
|
|
|
// null converts to the string "null", which is an invalid string value
|
|
|
|
assert.throws(RangeError, () => checkFunc(null), "null");
|
|
|
|
// Booleans convert to the strings "true" or "false", which are invalid
|
|
|
|
assert.throws(RangeError, () => checkFunc(true), "true");
|
|
|
|
assert.throws(RangeError, () => checkFunc(false), "false");
|
|
|
|
// Symbols cannot convert to strings
|
|
|
|
assert.throws(TypeError, () => checkFunc(Symbol()), "symbol");
|
|
|
|
// Numbers convert to strings which are invalid
|
|
|
|
assert.throws(RangeError, () => checkFunc(2), "number");
|
|
|
|
// BigInts convert to strings which are invalid
|
|
|
|
assert.throws(RangeError, () => checkFunc(2n), "bigint");
|
|
|
|
|
|
|
|
// Objects prefer their toString() methods when converting to a string
|
|
|
|
assert.throws(RangeError, () => checkFunc({}), "plain object");
|
|
|
|
|
|
|
|
const expected = [
|
|
|
|
`get ${propertyName}.toString`,
|
|
|
|
`call ${propertyName}.toString`,
|
|
|
|
];
|
|
|
|
const actual = [];
|
|
|
|
const observer = TemporalHelpers.toPrimitiveObserver(actual, value, propertyName);
|
|
|
|
const result = checkFunc(observer);
|
|
|
|
assertFunc(result, "object with toString");
|
|
|
|
assert.compareArray(actual, expected, "order of operations");
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* checkSubclassingIgnored(construct, constructArgs, method, methodArgs,
|
|
|
|
* resultAssertions):
|
|
|
|
*
|
|
|
|
* Methods of Temporal classes that return a new instance of the same class,
|
|
|
|
* must not take the constructor of a subclass into account, nor the @@species
|
|
|
|
* property. This helper runs tests to ensure this.
|
|
|
|
*
|
|
|
|
* construct(...constructArgs) must yield a valid instance of the Temporal
|
|
|
|
* class. instance[method](...methodArgs) is the method call under test, which
|
|
|
|
* must also yield a valid instance of the same Temporal class, not a
|
|
|
|
* subclass. See below for the individual tests that this runs.
|
|
|
|
* resultAssertions() is a function that performs additional assertions on the
|
|
|
|
* instance returned by the method under test.
|
|
|
|
*/
|
|
|
|
checkSubclassingIgnored(...args) {
|
|
|
|
this.checkSubclassConstructorNotObject(...args);
|
|
|
|
this.checkSubclassConstructorUndefined(...args);
|
|
|
|
this.checkSubclassConstructorThrows(...args);
|
|
|
|
this.checkSubclassConstructorNotCalled(...args);
|
|
|
|
this.checkSubclassSpeciesInvalidResult(...args);
|
|
|
|
this.checkSubclassSpeciesNotAConstructor(...args);
|
|
|
|
this.checkSubclassSpeciesNull(...args);
|
|
|
|
this.checkSubclassSpeciesUndefined(...args);
|
|
|
|
this.checkSubclassSpeciesThrows(...args);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Checks that replacing the 'constructor' property of the instance with
|
|
|
|
* various primitive values does not affect the returned new instance.
|
|
|
|
*/
|
|
|
|
checkSubclassConstructorNotObject(construct, constructArgs, method, methodArgs, resultAssertions) {
|
|
|
|
function check(value, description) {
|
|
|
|
const instance = new construct(...constructArgs);
|
|
|
|
instance.constructor = value;
|
|
|
|
const result = instance[method](...methodArgs);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype, description);
|
|
|
|
resultAssertions(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
check(null, "null");
|
|
|
|
check(true, "true");
|
|
|
|
check("test", "string");
|
|
|
|
check(Symbol(), "Symbol");
|
|
|
|
check(7, "number");
|
|
|
|
check(7n, "bigint");
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Checks that replacing the 'constructor' property of the subclass with
|
|
|
|
* undefined does not affect the returned new instance.
|
|
|
|
*/
|
|
|
|
checkSubclassConstructorUndefined(construct, constructArgs, method, methodArgs, resultAssertions) {
|
|
|
|
let called = 0;
|
|
|
|
|
|
|
|
class MySubclass extends construct {
|
|
|
|
constructor() {
|
|
|
|
++called;
|
|
|
|
super(...constructArgs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const instance = new MySubclass();
|
|
|
|
assert.sameValue(called, 1);
|
|
|
|
|
|
|
|
MySubclass.prototype.constructor = undefined;
|
|
|
|
|
|
|
|
const result = instance[method](...methodArgs);
|
|
|
|
assert.sameValue(called, 1);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype);
|
|
|
|
resultAssertions(result);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Checks that making the 'constructor' property of the instance throw when
|
|
|
|
* called does not affect the returned new instance.
|
|
|
|
*/
|
|
|
|
checkSubclassConstructorThrows(construct, constructArgs, method, methodArgs, resultAssertions) {
|
|
|
|
function CustomError() {}
|
|
|
|
const instance = new construct(...constructArgs);
|
|
|
|
Object.defineProperty(instance, "constructor", {
|
|
|
|
get() {
|
|
|
|
throw new CustomError();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
const result = instance[method](...methodArgs);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype);
|
|
|
|
resultAssertions(result);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Checks that when subclassing, the subclass constructor is not called by
|
|
|
|
* the method under test.
|
|
|
|
*/
|
|
|
|
checkSubclassConstructorNotCalled(construct, constructArgs, method, methodArgs, resultAssertions) {
|
|
|
|
let called = 0;
|
|
|
|
|
|
|
|
class MySubclass extends construct {
|
|
|
|
constructor() {
|
|
|
|
++called;
|
|
|
|
super(...constructArgs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const instance = new MySubclass();
|
|
|
|
assert.sameValue(called, 1);
|
|
|
|
|
|
|
|
const result = instance[method](...methodArgs);
|
|
|
|
assert.sameValue(called, 1);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype);
|
|
|
|
resultAssertions(result);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that the constructor's @@species property is ignored when it's a
|
|
|
|
* constructor that returns a non-object value.
|
|
|
|
*/
|
|
|
|
checkSubclassSpeciesInvalidResult(construct, constructArgs, method, methodArgs, resultAssertions) {
|
|
|
|
function check(value, description) {
|
|
|
|
const instance = new construct(...constructArgs);
|
|
|
|
instance.constructor = {
|
|
|
|
[Symbol.species]: function() {
|
|
|
|
return value;
|
|
|
|
},
|
|
|
|
};
|
|
|
|
const result = instance[method](...methodArgs);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype, description);
|
|
|
|
resultAssertions(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
check(undefined, "undefined");
|
|
|
|
check(null, "null");
|
|
|
|
check(true, "true");
|
|
|
|
check("test", "string");
|
|
|
|
check(Symbol(), "Symbol");
|
|
|
|
check(7, "number");
|
|
|
|
check(7n, "bigint");
|
|
|
|
check({}, "plain object");
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that the constructor's @@species property is ignored when it's not a
|
|
|
|
* constructor.
|
|
|
|
*/
|
|
|
|
checkSubclassSpeciesNotAConstructor(construct, constructArgs, method, methodArgs, resultAssertions) {
|
|
|
|
function check(value, description) {
|
|
|
|
const instance = new construct(...constructArgs);
|
|
|
|
instance.constructor = {
|
|
|
|
[Symbol.species]: value,
|
|
|
|
};
|
|
|
|
const result = instance[method](...methodArgs);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype, description);
|
|
|
|
resultAssertions(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
check(true, "true");
|
|
|
|
check("test", "string");
|
|
|
|
check(Symbol(), "Symbol");
|
|
|
|
check(7, "number");
|
|
|
|
check(7n, "bigint");
|
|
|
|
check({}, "plain object");
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that the constructor's @@species property is ignored when it's null.
|
|
|
|
*/
|
|
|
|
checkSubclassSpeciesNull(construct, constructArgs, method, methodArgs, resultAssertions) {
|
|
|
|
let called = 0;
|
|
|
|
|
|
|
|
class MySubclass extends construct {
|
|
|
|
constructor() {
|
|
|
|
++called;
|
|
|
|
super(...constructArgs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const instance = new MySubclass();
|
|
|
|
assert.sameValue(called, 1);
|
|
|
|
|
|
|
|
MySubclass.prototype.constructor = {
|
|
|
|
[Symbol.species]: null,
|
|
|
|
};
|
|
|
|
|
|
|
|
const result = instance[method](...methodArgs);
|
|
|
|
assert.sameValue(called, 1);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype);
|
|
|
|
resultAssertions(result);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that the constructor's @@species property is ignored when it's
|
|
|
|
* undefined.
|
|
|
|
*/
|
|
|
|
checkSubclassSpeciesUndefined(construct, constructArgs, method, methodArgs, resultAssertions) {
|
|
|
|
let called = 0;
|
|
|
|
|
|
|
|
class MySubclass extends construct {
|
|
|
|
constructor() {
|
|
|
|
++called;
|
|
|
|
super(...constructArgs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const instance = new MySubclass();
|
|
|
|
assert.sameValue(called, 1);
|
|
|
|
|
|
|
|
MySubclass.prototype.constructor = {
|
|
|
|
[Symbol.species]: undefined,
|
|
|
|
};
|
|
|
|
|
|
|
|
const result = instance[method](...methodArgs);
|
|
|
|
assert.sameValue(called, 1);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype);
|
|
|
|
resultAssertions(result);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that the constructor's @@species property is ignored when it throws,
|
|
|
|
* i.e. it is not called at all.
|
|
|
|
*/
|
|
|
|
checkSubclassSpeciesThrows(construct, constructArgs, method, methodArgs, resultAssertions) {
|
|
|
|
function CustomError() {}
|
|
|
|
|
|
|
|
const instance = new construct(...constructArgs);
|
|
|
|
instance.constructor = {
|
|
|
|
get [Symbol.species]() {
|
|
|
|
throw new CustomError();
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const result = instance[method](...methodArgs);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* checkSubclassingIgnoredStatic(construct, method, methodArgs, resultAssertions):
|
|
|
|
*
|
|
|
|
* Static methods of Temporal classes that return a new instance of the class,
|
|
|
|
* must not use the this-value as a constructor. This helper runs tests to
|
|
|
|
* ensure this.
|
|
|
|
*
|
|
|
|
* construct[method](...methodArgs) is the static method call under test, and
|
|
|
|
* must yield a valid instance of the Temporal class, not a subclass. See
|
|
|
|
* below for the individual tests that this runs.
|
|
|
|
* resultAssertions() is a function that performs additional assertions on the
|
|
|
|
* instance returned by the method under test.
|
|
|
|
*/
|
|
|
|
checkSubclassingIgnoredStatic(...args) {
|
|
|
|
this.checkStaticInvalidReceiver(...args);
|
|
|
|
this.checkStaticReceiverNotCalled(...args);
|
|
|
|
this.checkThisValueNotCalled(...args);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that calling the static method with a receiver that's not callable,
|
|
|
|
* still calls the intrinsic constructor.
|
|
|
|
*/
|
|
|
|
checkStaticInvalidReceiver(construct, method, methodArgs, resultAssertions) {
|
|
|
|
function check(value, description) {
|
|
|
|
const result = construct[method].apply(value, methodArgs);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype);
|
|
|
|
resultAssertions(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
check(undefined, "undefined");
|
|
|
|
check(null, "null");
|
|
|
|
check(true, "true");
|
|
|
|
check("test", "string");
|
|
|
|
check(Symbol(), "symbol");
|
|
|
|
check(7, "number");
|
|
|
|
check(7n, "bigint");
|
|
|
|
check({}, "Non-callable object");
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that calling the static method with a receiver that returns a value
|
|
|
|
* that's not callable, still calls the intrinsic constructor.
|
|
|
|
*/
|
|
|
|
checkStaticReceiverNotCalled(construct, method, methodArgs, resultAssertions) {
|
|
|
|
function check(value, description) {
|
|
|
|
const receiver = function () {
|
|
|
|
return value;
|
|
|
|
};
|
|
|
|
const result = construct[method].apply(receiver, methodArgs);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype);
|
|
|
|
resultAssertions(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
check(undefined, "undefined");
|
|
|
|
check(null, "null");
|
|
|
|
check(true, "true");
|
|
|
|
check("test", "string");
|
|
|
|
check(Symbol(), "symbol");
|
|
|
|
check(7, "number");
|
|
|
|
check(7n, "bigint");
|
|
|
|
check({}, "Non-callable object");
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that the receiver isn't called.
|
|
|
|
*/
|
|
|
|
checkThisValueNotCalled(construct, method, methodArgs, resultAssertions) {
|
|
|
|
let called = false;
|
|
|
|
|
|
|
|
class MySubclass extends construct {
|
|
|
|
constructor(...args) {
|
|
|
|
called = true;
|
|
|
|
super(...args);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const result = MySubclass[method](...methodArgs);
|
|
|
|
assert.sameValue(called, false);
|
|
|
|
assert.sameValue(Object.getPrototypeOf(result), construct.prototype);
|
|
|
|
resultAssertions(result);
|
|
|
|
},
|
|
|
|
|
2021-07-16 15:25:55 +02:00
|
|
|
/*
|
|
|
|
* Check that any calendar-carrying Temporal object has its [[Calendar]]
|
|
|
|
* internal slot read by ToTemporalCalendar, and does not fetch the calendar
|
|
|
|
* by calling getters.
|
|
|
|
*/
|
|
|
|
checkToTemporalCalendarFastPath(func) {
|
2024-06-04 12:49:30 +02:00
|
|
|
const plainDate = new Temporal.PlainDate(2000, 5, 2, "iso8601");
|
|
|
|
const plainDateTime = new Temporal.PlainDateTime(2000, 5, 2, 12, 34, 56, 987, 654, 321, "iso8601");
|
|
|
|
const plainMonthDay = new Temporal.PlainMonthDay(5, 2, "iso8601");
|
|
|
|
const plainYearMonth = new Temporal.PlainYearMonth(2000, 5, "iso8601");
|
|
|
|
const zonedDateTime = new Temporal.ZonedDateTime(1_000_000_000_000_000_000n, "UTC", "iso8601");
|
2021-07-16 15:25:55 +02:00
|
|
|
|
|
|
|
[plainDate, plainDateTime, plainMonthDay, plainYearMonth, zonedDateTime].forEach((temporalObject) => {
|
|
|
|
const actual = [];
|
|
|
|
const expected = [];
|
|
|
|
|
|
|
|
Object.defineProperty(temporalObject, "calendar", {
|
|
|
|
get() {
|
|
|
|
actual.push("get calendar");
|
|
|
|
return calendar;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2024-06-04 12:49:30 +02:00
|
|
|
func(temporalObject);
|
2021-07-16 15:25:55 +02:00
|
|
|
assert.compareArray(actual, expected, "calendar getter not called");
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2021-09-29 21:48:22 +02:00
|
|
|
checkToTemporalInstantFastPath(func) {
|
|
|
|
const actual = [];
|
|
|
|
const expected = [];
|
|
|
|
|
|
|
|
const datetime = new Temporal.ZonedDateTime(1_000_000_000_987_654_321n, "UTC");
|
|
|
|
Object.defineProperty(datetime, 'toString', {
|
|
|
|
get() {
|
|
|
|
actual.push("get toString");
|
|
|
|
return function (options) {
|
|
|
|
actual.push("call toString");
|
|
|
|
return Temporal.ZonedDateTime.prototype.toString.call(this, options);
|
|
|
|
};
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
func(datetime);
|
|
|
|
assert.compareArray(actual, expected, "toString not called");
|
|
|
|
},
|
|
|
|
|
|
|
|
checkToTemporalPlainDateTimeFastPath(func) {
|
|
|
|
const actual = [];
|
|
|
|
const expected = [];
|
|
|
|
|
2024-06-04 11:25:38 +02:00
|
|
|
const date = new Temporal.PlainDate(2000, 5, 2, "iso8601");
|
2021-09-29 21:48:22 +02:00
|
|
|
const prototypeDescrs = Object.getOwnPropertyDescriptors(Temporal.PlainDate.prototype);
|
|
|
|
["year", "month", "monthCode", "day"].forEach((property) => {
|
|
|
|
Object.defineProperty(date, property, {
|
|
|
|
get() {
|
2022-09-20 21:53:57 +02:00
|
|
|
actual.push(`get ${formatPropertyName(property)}`);
|
2021-09-29 21:48:22 +02:00
|
|
|
const value = prototypeDescrs[property].get.call(this);
|
|
|
|
return TemporalHelpers.toPrimitiveObserver(actual, value, property);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
["hour", "minute", "second", "millisecond", "microsecond", "nanosecond"].forEach((property) => {
|
|
|
|
Object.defineProperty(date, property, {
|
|
|
|
get() {
|
2022-09-20 21:53:57 +02:00
|
|
|
actual.push(`get ${formatPropertyName(property)}`);
|
2021-09-29 21:48:22 +02:00
|
|
|
return undefined;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
Object.defineProperty(date, "calendar", {
|
|
|
|
get() {
|
|
|
|
actual.push("get calendar");
|
2024-06-04 12:49:30 +02:00
|
|
|
return "iso8601";
|
2021-09-29 21:48:22 +02:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2024-06-04 12:49:30 +02:00
|
|
|
func(date);
|
2021-09-29 21:48:22 +02:00
|
|
|
assert.compareArray(actual, expected, "property getters not called");
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* observeProperty(calls, object, propertyName, value):
|
|
|
|
*
|
|
|
|
* Defines an own property @object.@propertyName with value @value, that
|
|
|
|
* will log any calls to its accessors to the array @calls.
|
|
|
|
*/
|
2022-09-20 21:53:57 +02:00
|
|
|
observeProperty(calls, object, propertyName, value, objectName = "") {
|
2021-09-29 21:48:22 +02:00
|
|
|
Object.defineProperty(object, propertyName, {
|
|
|
|
get() {
|
2022-09-20 21:53:57 +02:00
|
|
|
calls.push(`get ${formatPropertyName(propertyName, objectName)}`);
|
2021-09-29 21:48:22 +02:00
|
|
|
return value;
|
|
|
|
},
|
|
|
|
set(v) {
|
2022-09-20 21:53:57 +02:00
|
|
|
calls.push(`set ${formatPropertyName(propertyName, objectName)}`);
|
2021-09-29 21:48:22 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2022-12-14 03:46:20 +01:00
|
|
|
/*
|
|
|
|
* observeMethod(calls, object, propertyName, value):
|
|
|
|
*
|
|
|
|
* Defines an own property @object.@propertyName with value @value, that
|
|
|
|
* will log any calls of @value to the array @calls.
|
|
|
|
*/
|
|
|
|
observeMethod(calls, object, propertyName, objectName = "") {
|
|
|
|
const method = object[propertyName];
|
|
|
|
object[propertyName] = function () {
|
|
|
|
calls.push(`call ${formatPropertyName(propertyName, objectName)}`);
|
|
|
|
return method.apply(object, arguments);
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2022-11-26 04:20:11 +01:00
|
|
|
/*
|
|
|
|
* Used for substituteMethod to indicate default behavior instead of a
|
|
|
|
* substituted value
|
|
|
|
*/
|
|
|
|
SUBSTITUTE_SKIP: SKIP_SYMBOL,
|
|
|
|
|
|
|
|
/*
|
|
|
|
* substituteMethod(object, propertyName, values):
|
|
|
|
*
|
|
|
|
* Defines an own property @object.@propertyName that will, for each
|
|
|
|
* subsequent call to the method previously defined as
|
|
|
|
* @object.@propertyName:
|
|
|
|
* - Call the method, if no more values remain
|
|
|
|
* - Call the method, if the value in @values for the corresponding call
|
|
|
|
* is SUBSTITUTE_SKIP
|
|
|
|
* - Otherwise, return the corresponding value in @value
|
|
|
|
*/
|
|
|
|
substituteMethod(object, propertyName, values) {
|
|
|
|
let calls = 0;
|
|
|
|
const method = object[propertyName];
|
|
|
|
object[propertyName] = function () {
|
|
|
|
if (calls >= values.length) {
|
|
|
|
return method.apply(object, arguments);
|
|
|
|
} else if (values[calls] === SKIP_SYMBOL) {
|
|
|
|
calls++;
|
|
|
|
return method.apply(object, arguments);
|
|
|
|
} else {
|
|
|
|
return values[calls++];
|
|
|
|
}
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2022-09-20 20:22:00 +02:00
|
|
|
/*
|
|
|
|
* propertyBagObserver():
|
|
|
|
* Returns an object that behaves like the given propertyBag but tracks Get
|
|
|
|
* and Has operations on any of its properties, by appending messages to an
|
|
|
|
* array. If the value of a property in propertyBag is a primitive, the value
|
|
|
|
* of the returned object's property will additionally be a
|
|
|
|
* TemporalHelpers.toPrimitiveObserver that will track calls to its toString
|
|
|
|
* and valueOf methods in the same array. This is for the purpose of testing
|
|
|
|
* order of operations that are observable from user code. objectName is used
|
|
|
|
* in the log.
|
2024-06-04 13:15:07 +02:00
|
|
|
* If skipToPrimitive is given, it must be an array of property keys. Those
|
|
|
|
* properties will not have a TemporalHelpers.toPrimitiveObserver returned,
|
|
|
|
* and instead just be returned directly.
|
2022-09-20 20:22:00 +02:00
|
|
|
*/
|
2024-06-04 13:15:07 +02:00
|
|
|
propertyBagObserver(calls, propertyBag, objectName, skipToPrimitive) {
|
2022-09-20 20:22:00 +02:00
|
|
|
return new Proxy(propertyBag, {
|
|
|
|
ownKeys(target) {
|
|
|
|
calls.push(`ownKeys ${objectName}`);
|
|
|
|
return Reflect.ownKeys(target);
|
|
|
|
},
|
2022-10-18 02:37:39 +02:00
|
|
|
getOwnPropertyDescriptor(target, key) {
|
|
|
|
calls.push(`getOwnPropertyDescriptor ${formatPropertyName(key, objectName)}`);
|
|
|
|
return Reflect.getOwnPropertyDescriptor(target, key);
|
|
|
|
},
|
2022-09-20 20:22:00 +02:00
|
|
|
get(target, key, receiver) {
|
|
|
|
calls.push(`get ${formatPropertyName(key, objectName)}`);
|
|
|
|
const result = Reflect.get(target, key, receiver);
|
|
|
|
if (result === undefined) {
|
|
|
|
return undefined;
|
|
|
|
}
|
2023-02-24 03:21:41 +01:00
|
|
|
if ((result !== null && typeof result === "object") || typeof result === "function") {
|
2022-09-20 20:22:00 +02:00
|
|
|
return result;
|
|
|
|
}
|
2024-06-04 13:15:07 +02:00
|
|
|
if (skipToPrimitive && skipToPrimitive.indexOf(key) >= 0) {
|
|
|
|
return result;
|
2022-09-20 21:18:21 +02:00
|
|
|
}
|
2024-06-04 13:15:07 +02:00
|
|
|
return TemporalHelpers.toPrimitiveObserver(calls, result, `${formatPropertyName(key, objectName)}`);
|
2022-09-20 21:18:21 +02:00
|
|
|
},
|
|
|
|
has(target, key) {
|
2023-02-08 03:48:28 +01:00
|
|
|
calls.push(`has ${formatPropertyName(key, objectName)}`);
|
2022-09-20 21:18:21 +02:00
|
|
|
return Reflect.has(target, key);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2021-09-29 21:48:22 +02:00
|
|
|
/*
|
|
|
|
* Returns an object that will append logs of any Gets or Calls of its valueOf
|
|
|
|
* or toString properties to the array calls. Both valueOf and toString will
|
|
|
|
* return the actual primitiveValue. propertyName is used in the log.
|
|
|
|
*/
|
|
|
|
toPrimitiveObserver(calls, primitiveValue, propertyName) {
|
|
|
|
return {
|
|
|
|
get valueOf() {
|
|
|
|
calls.push(`get ${propertyName}.valueOf`);
|
|
|
|
return function () {
|
|
|
|
calls.push(`call ${propertyName}.valueOf`);
|
|
|
|
return primitiveValue;
|
|
|
|
};
|
|
|
|
},
|
|
|
|
get toString() {
|
|
|
|
calls.push(`get ${propertyName}.toString`);
|
|
|
|
return function () {
|
|
|
|
calls.push(`call ${propertyName}.toString`);
|
|
|
|
if (primitiveValue === undefined) return undefined;
|
|
|
|
return primitiveValue.toString();
|
|
|
|
};
|
|
|
|
},
|
|
|
|
};
|
|
|
|
},
|
2022-08-18 01:55:22 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* An object containing further methods that return arrays of ISO strings, for
|
|
|
|
* testing parsers.
|
|
|
|
*/
|
|
|
|
ISO: {
|
2022-10-21 23:53:42 +02:00
|
|
|
/*
|
|
|
|
* PlainMonthDay strings that are not valid.
|
|
|
|
*/
|
|
|
|
plainMonthDayStringsInvalid() {
|
|
|
|
return [
|
|
|
|
"11-18junk",
|
2023-09-19 20:04:09 +02:00
|
|
|
"11-18[u-ca=gregory]",
|
|
|
|
"11-18[u-ca=hebrew]",
|
2024-02-14 02:36:54 +01:00
|
|
|
"11-18[U-CA=iso8601]",
|
|
|
|
"11-18[u-CA=iso8601]",
|
|
|
|
"11-18[FOO=bar]",
|
2022-10-21 23:53:42 +02:00
|
|
|
];
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PlainMonthDay strings that are valid and that should produce October 1st.
|
|
|
|
*/
|
|
|
|
plainMonthDayStringsValid() {
|
|
|
|
return [
|
|
|
|
"10-01",
|
|
|
|
"1001",
|
|
|
|
"1965-10-01",
|
|
|
|
"1976-10-01T152330.1+00:00",
|
|
|
|
"19761001T15:23:30.1+00:00",
|
|
|
|
"1976-10-01T15:23:30.1+0000",
|
|
|
|
"1976-10-01T152330.1+0000",
|
|
|
|
"19761001T15:23:30.1+0000",
|
|
|
|
"19761001T152330.1+00:00",
|
|
|
|
"19761001T152330.1+0000",
|
|
|
|
"+001976-10-01T152330.1+00:00",
|
|
|
|
"+0019761001T15:23:30.1+00:00",
|
|
|
|
"+001976-10-01T15:23:30.1+0000",
|
|
|
|
"+001976-10-01T152330.1+0000",
|
|
|
|
"+0019761001T15:23:30.1+0000",
|
|
|
|
"+0019761001T152330.1+00:00",
|
|
|
|
"+0019761001T152330.1+0000",
|
|
|
|
"1976-10-01T15:23:00",
|
|
|
|
"1976-10-01T15:23",
|
|
|
|
"1976-10-01T15",
|
|
|
|
"1976-10-01",
|
|
|
|
"--10-01",
|
|
|
|
"--1001",
|
|
|
|
];
|
|
|
|
},
|
|
|
|
|
2022-08-18 01:55:22 +02:00
|
|
|
/*
|
|
|
|
* PlainTime strings that may be mistaken for PlainMonthDay or
|
|
|
|
* PlainYearMonth strings, and so require a time designator.
|
|
|
|
*/
|
|
|
|
plainTimeStringsAmbiguous() {
|
|
|
|
const ambiguousStrings = [
|
|
|
|
"2021-12", // ambiguity between YYYY-MM and HHMM-UU
|
2022-10-24 21:35:09 +02:00
|
|
|
"2021-12[-12:00]", // ditto, TZ does not disambiguate
|
2022-08-18 01:55:22 +02:00
|
|
|
"1214", // ambiguity between MMDD and HHMM
|
|
|
|
"0229", // ditto, including MMDD that doesn't occur every year
|
|
|
|
"1130", // ditto, including DD that doesn't occur in every month
|
|
|
|
"12-14", // ambiguity between MM-DD and HH-UU
|
2022-10-24 21:35:09 +02:00
|
|
|
"12-14[-14:00]", // ditto, TZ does not disambiguate
|
2022-08-18 01:55:22 +02:00
|
|
|
"202112", // ambiguity between YYYYMM and HHMMSS
|
2022-10-24 21:35:09 +02:00
|
|
|
"202112[UTC]", // ditto, TZ does not disambiguate
|
2022-08-18 01:55:22 +02:00
|
|
|
];
|
|
|
|
// Adding a calendar annotation to one of these strings must not cause
|
|
|
|
// disambiguation in favour of time.
|
|
|
|
const stringsWithCalendar = ambiguousStrings.map((s) => s + '[u-ca=iso8601]');
|
|
|
|
return ambiguousStrings.concat(stringsWithCalendar);
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PlainTime strings that are of similar form to PlainMonthDay and
|
|
|
|
* PlainYearMonth strings, but are not ambiguous due to components that
|
|
|
|
* aren't valid as months or days.
|
|
|
|
*/
|
|
|
|
plainTimeStringsUnambiguous() {
|
|
|
|
return [
|
|
|
|
"2021-13", // 13 is not a month
|
|
|
|
"202113", // ditto
|
|
|
|
"2021-13[-13:00]", // ditto
|
|
|
|
"202113[-13:00]", // ditto
|
|
|
|
"0000-00", // 0 is not a month
|
|
|
|
"000000", // ditto
|
|
|
|
"0000-00[UTC]", // ditto
|
|
|
|
"000000[UTC]", // ditto
|
|
|
|
"1314", // 13 is not a month
|
|
|
|
"13-14", // ditto
|
|
|
|
"1232", // 32 is not a day
|
|
|
|
"0230", // 30 is not a day in February
|
|
|
|
"0631", // 31 is not a day in June
|
|
|
|
"0000", // 0 is neither a month nor a day
|
|
|
|
"00-00", // ditto
|
|
|
|
];
|
2022-10-21 23:53:42 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PlainYearMonth-like strings that are not valid.
|
|
|
|
*/
|
|
|
|
plainYearMonthStringsInvalid() {
|
|
|
|
return [
|
|
|
|
"2020-13",
|
2024-02-14 02:36:54 +01:00
|
|
|
"1976-11[u-ca=gregory]",
|
|
|
|
"1976-11[u-ca=hebrew]",
|
|
|
|
"1976-11[U-CA=iso8601]",
|
|
|
|
"1976-11[u-CA=iso8601]",
|
|
|
|
"1976-11[FOO=bar]",
|
2022-10-21 23:53:42 +02:00
|
|
|
];
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PlainYearMonth-like strings that are valid and should produce November
|
|
|
|
* 1976 in the ISO 8601 calendar.
|
|
|
|
*/
|
|
|
|
plainYearMonthStringsValid() {
|
|
|
|
return [
|
|
|
|
"1976-11",
|
|
|
|
"1976-11-10",
|
|
|
|
"1976-11-01T09:00:00+00:00",
|
|
|
|
"1976-11-01T00:00:00+05:00",
|
|
|
|
"197611",
|
|
|
|
"+00197611",
|
2024-06-28 02:28:31 +02:00
|
|
|
"1976-11-18T15:23:30.1-02:00",
|
2022-10-21 23:53:42 +02:00
|
|
|
"1976-11-18T152330.1+00:00",
|
|
|
|
"19761118T15:23:30.1+00:00",
|
|
|
|
"1976-11-18T15:23:30.1+0000",
|
|
|
|
"1976-11-18T152330.1+0000",
|
|
|
|
"19761118T15:23:30.1+0000",
|
|
|
|
"19761118T152330.1+00:00",
|
|
|
|
"19761118T152330.1+0000",
|
|
|
|
"+001976-11-18T152330.1+00:00",
|
|
|
|
"+0019761118T15:23:30.1+00:00",
|
|
|
|
"+001976-11-18T15:23:30.1+0000",
|
|
|
|
"+001976-11-18T152330.1+0000",
|
|
|
|
"+0019761118T15:23:30.1+0000",
|
|
|
|
"+0019761118T152330.1+00:00",
|
|
|
|
"+0019761118T152330.1+0000",
|
|
|
|
"1976-11-18T15:23",
|
|
|
|
"1976-11-18T15",
|
|
|
|
"1976-11-18",
|
|
|
|
];
|
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PlainYearMonth-like strings that are valid and should produce November of
|
|
|
|
* the ISO year -9999.
|
|
|
|
*/
|
|
|
|
plainYearMonthStringsValidNegativeYear() {
|
|
|
|
return [
|
2024-06-28 02:28:31 +02:00
|
|
|
"-009999-11",
|
2022-10-21 23:53:42 +02:00
|
|
|
];
|
|
|
|
},
|
2022-08-18 01:55:22 +02:00
|
|
|
}
|
2021-07-16 15:25:55 +02:00
|
|
|
};
|