Skip to content

Commit

Permalink
LibJS: Fix the Now.plainDateTime in case they go over a year boundary
Browse files Browse the repository at this point in the history
Since years don't have a constant amount of seconds because they can be
leap years no constant will work in all cases. We now test a timezone in
both the positive and negative direction and check that at least one
worked. Assuming years are at least 2 days long this will always pass
at least one test.
  • Loading branch information
davidot authored and linusg committed Dec 31, 2021
1 parent c296df6 commit c0a3b14
Show file tree
Hide file tree
Showing 2 changed files with 93 additions and 22 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -10,32 +10,68 @@ describe("correct behavior", () => {
expect(plainDateTime.calendar).toBe(calendar);
});

test("custom time zone", () => {
const plainDateTimeToEpochSeconds = plainDateTime =>
(plainDateTime.year - 1970) * 31_556_952 +
plainDateTime.dayOfYear * 86_400 +
plainDateTime.hour * 3_600 +
plainDateTime.minute * 60 +
plainDateTime.second +
plainDateTime.millisecond / 1_000 +
plainDateTime.microsecond / 1_000_000 +
plainDateTime.nanosecond / 1_000_000_000;

let timeZoneTested = false;

// Note: We test both positive and negative timezones because one might cross a year boundary.
// Since a year does not have a fixed amount of seconds because it can be a leap year,
// we cannot have a correct constant for seconds per year which is always correct.
// However, by assuming years are at least 2 days long we can simply try the positive
// and negative timezones and skip one if we jump the year. To ensure at least one is
// tested we have the timeZoneTested which is only set to true if one of the tests passed.

test("custom time zone positive", () => {
const calendar = new Temporal.Calendar("iso8601");
const timeZone = {
getOffsetNanosecondsFor() {
return 86400000000000;
},
};

const plainDateTimeToEpochSeconds = plainDateTime =>
(plainDateTime.year - 1970) * 31_556_952 +
plainDateTime.dayOfYear * 86_400 +
plainDateTime.hour * 3_600 +
plainDateTime.minute * 60 +
plainDateTime.second +
plainDateTime.millisecond / 1_000 +
plainDateTime.microsecond / 1_000_000 +
plainDateTime.nanosecond / 1_000_000_000;

const plainDateTime = Temporal.Now.plainDateTime(calendar);
const plainDateTimeWithOffset = Temporal.Now.plainDateTime(calendar, timeZone);

if (plainDateTime.year !== plainDateTimeWithOffset.year) return;

// Let's hope the duration between the above two lines is less than a second :^)
const differenceSeconds =
plainDateTimeToEpochSeconds(plainDateTimeWithOffset) -
plainDateTimeToEpochSeconds(plainDateTime);
expect(Math.floor(differenceSeconds)).toBe(86400);
timeZoneTested = true;
});

test("custom time zone negative", () => {
const calendar = new Temporal.Calendar("iso8601");
const timeZone = {
getOffsetNanosecondsFor() {
return -86400000000000;
},
};

const plainDateTime = Temporal.Now.plainDateTime(calendar);
const plainDateTimeWithOffset = Temporal.Now.plainDateTime(calendar, timeZone);

if (plainDateTime.year !== plainDateTimeWithOffset.year) return;

// Let's hope the duration between the above two lines is less than a second :^)
const differenceSeconds =
plainDateTimeToEpochSeconds(plainDateTimeWithOffset) -
plainDateTimeToEpochSeconds(plainDateTime);
expect(Math.floor(differenceSeconds)).toBe(-86400);
timeZoneTested = true;
});

expect(timeZoneTested).toBeTrue();
});

describe("errors", () => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,31 +9,66 @@ describe("correct behavior", () => {
expect(plainDateTime.calendar.id).toBe("iso8601");
});

test("custom time zone", () => {
const plainDateTimeToEpochSeconds = plainDateTime =>
(plainDateTime.year - 1970) * 31_556_952 +
plainDateTime.dayOfYear * 86_400 +
plainDateTime.hour * 3_600 +
plainDateTime.minute * 60 +
plainDateTime.second +
plainDateTime.millisecond / 1_000 +
plainDateTime.microsecond / 1_000_000 +
plainDateTime.nanosecond / 1_000_000_000;

let timeZoneTested = false;

// Note: We test both positive and negative timezones because one might cross a year boundary.
// Since a year does not have a fixed amount of seconds because it can be a leap year,
// we cannot have a correct constant for seconds per year which is always correct.
// However, by assuming years are at least 2 days long we can simply try the positive
// and negative timezones and skip one if we jump the year. To ensure at least one is
// tested we have the timeZoneTested which is only set to true if one of the tests passed.

test("custom time zone positive", () => {
const timeZone = {
getOffsetNanosecondsFor() {
return 86400000000000;
},
};

const plainDateTimeToEpochSeconds = plainDateTime =>
(plainDateTime.year - 1970) * 31_556_952 +
plainDateTime.dayOfYear * 86_400 +
plainDateTime.hour * 3_600 +
plainDateTime.minute * 60 +
plainDateTime.second +
plainDateTime.millisecond / 1_000 +
plainDateTime.microsecond / 1_000_000 +
plainDateTime.nanosecond / 1_000_000_000;

const plainDateTime = Temporal.Now.plainDateTimeISO();
const plainDateTimeWithOffset = Temporal.Now.plainDateTimeISO(timeZone);

if (plainDateTime.year !== plainDateTimeWithOffset.year) return;

// Let's hope the duration between the above two lines is less than a second :^)
const differenceSeconds =
plainDateTimeToEpochSeconds(plainDateTimeWithOffset) -
plainDateTimeToEpochSeconds(plainDateTime);
expect(Math.floor(differenceSeconds)).toBe(86400);
timeZoneTested = true;
});

test("custom time zone negative", () => {
const timeZone = {
getOffsetNanosecondsFor() {
return -86400000000000;
},
};

const plainDateTime = Temporal.Now.plainDateTimeISO();
const plainDateTimeWithOffset = Temporal.Now.plainDateTimeISO(timeZone);

if (plainDateTime.year !== plainDateTimeWithOffset.year) return;

// Let's hope the duration between the above two lines is less than a second :^)
const differenceSeconds =
plainDateTimeToEpochSeconds(plainDateTimeWithOffset) -
plainDateTimeToEpochSeconds(plainDateTime);
expect(Math.floor(differenceSeconds)).toBe(-86400);
timeZoneTested = true;
});

expect(timeZoneTested).toBeTrue();
});

describe("errors", () => {
Expand Down

0 comments on commit c0a3b14

Please sign in to comment.