mirror of https://github.com/tc39/test262.git
440 lines
12 KiB
JavaScript
440 lines
12 KiB
JavaScript
//Date.library.js
|
|
// Copyright 2009 the Sputnik authors. All rights reserved.
|
|
// This code is governed by the BSD license found in the LICENSE file.
|
|
|
|
//15.9.1.2 Day Number and Time within Day
|
|
function Day(t) {
|
|
return Math.floor(t/msPerDay);
|
|
}
|
|
|
|
function TimeWithinDay(t) {
|
|
return t%msPerDay;
|
|
}
|
|
|
|
//15.9.1.3 Year Number
|
|
function DaysInYear(y){
|
|
if(y%4 != 0) return 365;
|
|
if(y%4 == 0 && y%100 != 0) return 366;
|
|
if(y%100 == 0 && y%400 != 0) return 365;
|
|
if(y%400 == 0) return 366;
|
|
}
|
|
|
|
function DayFromYear(y) {
|
|
return (365*(y-1970)
|
|
+ Math.floor((y-1969)/4)
|
|
- Math.floor((y-1901)/100)
|
|
+ Math.floor((y-1601)/400));
|
|
}
|
|
|
|
function TimeFromYear(y){
|
|
return msPerDay*DayFromYear(y);
|
|
}
|
|
|
|
function YearFromTime(t) {
|
|
t = Number(t);
|
|
var sign = ( t < 0 ) ? -1 : 1;
|
|
var year = ( sign < 0 ) ? 1969 : 1970;
|
|
|
|
for(var time = 0;;year += sign){
|
|
time = TimeFromYear(year);
|
|
|
|
if(sign > 0 && time > t){
|
|
year -= sign;
|
|
break;
|
|
}
|
|
else if(sign < 0 && time <= t){
|
|
break;
|
|
}
|
|
};
|
|
return year;
|
|
}
|
|
|
|
function InLeapYear(t){
|
|
if(DaysInYear(YearFromTime(t)) == 365)
|
|
return 0;
|
|
|
|
if(DaysInYear(YearFromTime(t)) == 366)
|
|
return 1;
|
|
}
|
|
|
|
function DayWithinYear(t) {
|
|
return Day(t)-DayFromYear(YearFromTime(t));
|
|
}
|
|
|
|
//15.9.1.4 Month Number
|
|
function MonthFromTime(t){
|
|
var day = DayWithinYear(t);
|
|
var leap = InLeapYear(t);
|
|
|
|
if((0 <= day) && (day < 31)) return 0;
|
|
if((31 <= day) && (day < (59+leap))) return 1;
|
|
if(((59+leap) <= day) && (day < (90+leap))) return 2;
|
|
if(((90+leap) <= day) && (day < (120+leap))) return 3;
|
|
if(((120+leap) <= day) && (day < (151+leap))) return 4;
|
|
if(((151+leap) <= day) && (day < (181+leap))) return 5;
|
|
if(((181+leap) <= day) && (day < (212+leap))) return 6;
|
|
if(((212+leap) <= day) && (day < (243+leap))) return 7;
|
|
if(((243+leap) <= day) && (day < (273+leap))) return 8;
|
|
if(((273+leap) <= day) && (day < (304+leap))) return 9;
|
|
if(((304+leap) <= day) && (day < (334+leap))) return 10;
|
|
if(((334+leap) <= day) && (day < (365+leap))) return 11;
|
|
}
|
|
|
|
//15.9.1.5 Date Number
|
|
function DateFromTime(t) {
|
|
var day = DayWithinYear(t);
|
|
var month = MonthFromTime(t);
|
|
var leap = InLeapYear(t);
|
|
|
|
if(month == 0) return day+1;
|
|
if(month == 1) return day-30;
|
|
if(month == 2) return day-58-leap;
|
|
if(month == 3) return day-89-leap;
|
|
if(month == 4) return day-119-leap;
|
|
if(month == 5) return day-150-leap;
|
|
if(month == 6) return day-180-leap;
|
|
if(month == 7) return day-211-leap;
|
|
if(month == 8) return day-242-leap;
|
|
if(month == 9) return day-272-leap;
|
|
if(month == 10) return day-303-leap;
|
|
if(month == 11) return day-333-leap;
|
|
}
|
|
|
|
//15.9.1.6 Week Day
|
|
function WeekDay(t) {
|
|
var weekday = (Day(t)+4)%7;
|
|
return (weekday < 0 ? 7+weekday : weekday);
|
|
}
|
|
|
|
//15.9.1.9 Daylight Saving Time Adjustment
|
|
$LocalTZ = (new Date()).getTimezoneOffset() / -60;
|
|
if (DaylightSavingTA((new Date()).valueOf()) !== 0) {
|
|
$LocalTZ -= 1;
|
|
}
|
|
var LocalTZA = $LocalTZ*msPerHour;
|
|
|
|
function DaysInMonth(m, leap) {
|
|
m = m%12;
|
|
|
|
//April, June, Sept, Nov
|
|
if(m == 3 || m == 5 || m == 8 || m == 10 ) {
|
|
return 30;
|
|
}
|
|
|
|
//Jan, March, May, July, Aug, Oct, Dec
|
|
if(m == 0 || m == 2 || m == 4 || m == 6 || m == 7 || m == 9 || m == 11){
|
|
return 31;
|
|
}
|
|
|
|
//Feb
|
|
return 28+leap;
|
|
}
|
|
|
|
function GetSundayInMonth(t, m, count){
|
|
var year = YearFromTime(t);
|
|
var tempDate;
|
|
|
|
if (count==='"first"') {
|
|
for (var d=1; d <= DaysInMonth(m, InLeapYear(t)); d++) {
|
|
tempDate = new Date(year, m, d);
|
|
if (tempDate.getDay()===0) {
|
|
return tempDate.valueOf();
|
|
}
|
|
}
|
|
} else if(count==='"last"') {
|
|
for (var d=DaysInMonth(m, InLeapYear(t)); d>0; d--) {
|
|
tempDate = new Date(year, m, d);
|
|
if (tempDate.getDay()===0) {
|
|
return tempDate.valueOf();
|
|
}
|
|
}
|
|
}
|
|
throw new Error("Unsupported 'count' arg:" + count);
|
|
}
|
|
/*
|
|
function GetSundayInMonth(t, m, count){
|
|
var year = YearFromTime(t);
|
|
var leap = InLeapYear(t);
|
|
var day = 0;
|
|
|
|
if(m >= 1) day += DaysInMonth(0, leap);
|
|
if(m >= 2) day += DaysInMonth(1, leap);
|
|
if(m >= 3) day += DaysInMonth(2, leap);
|
|
if(m >= 4) day += DaysInMonth(3, leap);
|
|
if(m >= 5) day += DaysInMonth(4, leap);
|
|
if(m >= 6) day += DaysInMonth(5, leap);
|
|
if(m >= 7) day += DaysInMonth(6, leap);
|
|
if(m >= 8) day += DaysInMonth(7, leap);
|
|
if(m >= 9) day += DaysInMonth(8, leap);
|
|
if(m >= 10) day += DaysInMonth(9, leap);
|
|
if(m >= 11) day += DaysInMonth(10, leap);
|
|
|
|
var month_start = TimeFromYear(year)+day*msPerDay;
|
|
var sunday = 0;
|
|
|
|
if(count === "last"){
|
|
for(var last_sunday = month_start+DaysInMonth(m, leap)*msPerDay;
|
|
WeekDay(last_sunday)>0;
|
|
last_sunday -= msPerDay
|
|
){};
|
|
sunday = last_sunday;
|
|
}
|
|
else {
|
|
for(var first_sunday = month_start;
|
|
WeekDay(first_sunday)>0;
|
|
first_sunday += msPerDay
|
|
){};
|
|
sunday = first_sunday+7*msPerDay*(count-1);
|
|
}
|
|
|
|
return sunday;
|
|
}*/
|
|
|
|
function DaylightSavingTA(t) {
|
|
// t = t-LocalTZA;
|
|
|
|
var DST_start = GetSundayInMonth(t, $DST_start_month, $DST_start_sunday) +
|
|
$DST_start_hour*msPerHour +
|
|
$DST_start_minutes*msPerMinute;
|
|
|
|
var k = new Date(DST_start);
|
|
|
|
var DST_end = GetSundayInMonth(t, $DST_end_month, $DST_end_sunday) +
|
|
$DST_end_hour*msPerHour +
|
|
$DST_end_minutes*msPerMinute;
|
|
|
|
if ( t >= DST_start && t < DST_end ) {
|
|
return msPerHour;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
//15.9.1.9 Local Time
|
|
function LocalTime(t){
|
|
return t+LocalTZA+DaylightSavingTA(t);
|
|
}
|
|
|
|
function UTC(t) {
|
|
return t-LocalTZA-DaylightSavingTA(t-LocalTZA);
|
|
}
|
|
|
|
//15.9.1.10 Hours, Minutes, Second, and Milliseconds
|
|
function HourFromTime(t){
|
|
return Math.floor(t/msPerHour)%HoursPerDay;
|
|
}
|
|
|
|
function MinFromTime(t){
|
|
return Math.floor(t/msPerMinute)%MinutesPerHour;
|
|
}
|
|
|
|
function SecFromTime(t){
|
|
return Math.floor(t/msPerSecond)%SecondsPerMinute;
|
|
}
|
|
|
|
function msFromTime(t){
|
|
return t%msPerSecond;
|
|
}
|
|
|
|
//15.9.1.11 MakeTime (hour, min, sec, ms)
|
|
function MakeTime(hour, min, sec, ms){
|
|
if ( !isFinite(hour) || !isFinite(min) || !isFinite(sec) || !isFinite(ms)) {
|
|
return Number.NaN;
|
|
}
|
|
|
|
hour = ToInteger(hour);
|
|
min = ToInteger(min);
|
|
sec = ToInteger(sec);
|
|
ms = ToInteger(ms);
|
|
|
|
return ((hour*msPerHour) + (min*msPerMinute) + (sec*msPerSecond) + ms);
|
|
}
|
|
|
|
//15.9.1.12 MakeDay (year, month, date)
|
|
function MakeDay(year, month, date) {
|
|
if ( !isFinite(year) || !isFinite(month) || !isFinite(date)) {
|
|
return Number.NaN;
|
|
}
|
|
|
|
year = ToInteger(year);
|
|
month = ToInteger(month);
|
|
date = ToInteger(date );
|
|
|
|
var result5 = year + Math.floor(month/12);
|
|
var result6 = month%12;
|
|
|
|
var sign = ( year < 1970 ) ? -1 : 1;
|
|
var t = ( year < 1970 ) ? 1 : 0;
|
|
var y = ( year < 1970 ) ? 1969 : 1970;
|
|
|
|
if( sign == -1 ){
|
|
for ( y = 1969; y >= year; y += sign ) {
|
|
t += sign * DaysInYear(y)*msPerDay;
|
|
}
|
|
} else {
|
|
for ( y = 1970 ; y < year; y += sign ) {
|
|
t += sign * DaysInYear(y)*msPerDay;
|
|
}
|
|
}
|
|
|
|
var leap = 0;
|
|
for ( var m = 0; m < month; m++ ) {
|
|
//if year is changed, than we need to recalculate leep
|
|
leap = InLeapYear(t);
|
|
t += DaysInMonth(m, leap)*msPerDay;
|
|
}
|
|
|
|
if ( YearFromTime(t) != result5 ) {
|
|
return Number.NaN;
|
|
}
|
|
if ( MonthFromTime(t) != result6 ) {
|
|
return Number.NaN;
|
|
}
|
|
if ( DateFromTime(t) != 1 ) {
|
|
return Number.NaN;
|
|
}
|
|
|
|
return Day(t)+date-1;
|
|
}
|
|
|
|
//15.9.1.13 MakeDate (day, time)
|
|
function MakeDate( day, time ) {
|
|
if(!isFinite(day) || !isFinite(time)) {
|
|
return Number.NaN;
|
|
}
|
|
|
|
return day*msPerDay+time;
|
|
}
|
|
|
|
//15.9.1.14 TimeClip (time)
|
|
function TimeClip(time) {
|
|
if(!isFinite(time) || Math.abs(time) > 8.64e15){
|
|
return Number.NaN;
|
|
}
|
|
|
|
return ToInteger(time);
|
|
}
|
|
|
|
//Test Functions
|
|
//ConstructDate is considered deprecated, and should not be used directly from
|
|
//test262 tests as it's incredibly sensitive to DST start/end dates that
|
|
//vary with geographic location.
|
|
function ConstructDate(year, month, date, hours, minutes, seconds, ms){
|
|
/*
|
|
* 1. Call ToNumber(year)
|
|
* 2. Call ToNumber(month)
|
|
* 3. If date is supplied use ToNumber(date); else use 1
|
|
* 4. If hours is supplied use ToNumber(hours); else use 0
|
|
* 5. If minutes is supplied use ToNumber(minutes); else use 0
|
|
* 6. If seconds is supplied use ToNumber(seconds); else use 0
|
|
* 7. If ms is supplied use ToNumber(ms); else use 0
|
|
* 8. If Result(1) is not NaN and 0 <= ToInteger(Result(1)) <= 99, Result(8) is
|
|
* 1900+ToInteger(Result(1)); otherwise, Result(8) is Result(1)
|
|
* 9. Compute MakeDay(Result(8), Result(2), Result(3))
|
|
* 10. Compute MakeTime(Result(4), Result(5), Result(6), Result(7))
|
|
* 11. Compute MakeDate(Result(9), Result(10))
|
|
* 12. Set the [[Value]] property of the newly constructed object to TimeClip(UTC(Result(11)))
|
|
*/
|
|
var r1 = Number(year);
|
|
var r2 = Number(month);
|
|
var r3 = ((date && arguments.length > 2) ? Number(date) : 1);
|
|
var r4 = ((hours && arguments.length > 3) ? Number(hours) : 0);
|
|
var r5 = ((minutes && arguments.length > 4) ? Number(minutes) : 0);
|
|
var r6 = ((seconds && arguments.length > 5) ? Number(seconds) : 0);
|
|
var r7 = ((ms && arguments.length > 6) ? Number(ms) : 0);
|
|
|
|
var r8 = r1;
|
|
|
|
if(!isNaN(r1) && (0 <= ToInteger(r1)) && (ToInteger(r1) <= 99))
|
|
r8 = 1900+r1;
|
|
|
|
var r9 = MakeDay(r8, r2, r3);
|
|
var r10 = MakeTime(r4, r5, r6, r7);
|
|
var r11 = MakeDate(r9, r10);
|
|
|
|
var retVal = TimeClip(UTC(r11));
|
|
return retVal;
|
|
}
|
|
|
|
|
|
|
|
/**** Python code for initialize the above constants
|
|
// We may want to replicate the following in JavaScript.
|
|
// However, using JS date operations to generate parameters that are then used to
|
|
// test those some date operations seems unsound. However, it isn't clear if there
|
|
//is a good interoperable alternative.
|
|
|
|
# Copyright 2009 the Sputnik authors. All rights reserved.
|
|
# This code is governed by the BSD license found in the LICENSE file.
|
|
|
|
def GetDaylightSavingsTimes():
|
|
# Is the given floating-point time in DST?
|
|
def IsDst(t):
|
|
return time.localtime(t)[-1]
|
|
# Binary search to find an interval between the two times no greater than
|
|
# delta where DST switches, returning the midpoint.
|
|
def FindBetween(start, end, delta):
|
|
while end - start > delta:
|
|
middle = (end + start) / 2
|
|
if IsDst(middle) == IsDst(start):
|
|
start = middle
|
|
else:
|
|
end = middle
|
|
return (start + end) / 2
|
|
now = time.time()
|
|
one_month = (30 * 24 * 60 * 60)
|
|
# First find a date with different daylight savings. To avoid corner cases
|
|
# we try four months before and after today.
|
|
after = now + 4 * one_month
|
|
before = now - 4 * one_month
|
|
if IsDst(now) == IsDst(before) and IsDst(now) == IsDst(after):
|
|
logger.warning("Was unable to determine DST info.")
|
|
return None
|
|
# Determine when the change occurs between now and the date we just found
|
|
# in a different DST.
|
|
if IsDst(now) != IsDst(before):
|
|
first = FindBetween(before, now, 1)
|
|
else:
|
|
first = FindBetween(now, after, 1)
|
|
# Determine when the change occurs between three and nine months from the
|
|
# first.
|
|
second = FindBetween(first + 3 * one_month, first + 9 * one_month, 1)
|
|
# Find out which switch is into and which if out of DST
|
|
if IsDst(first - 1) and not IsDst(first + 1):
|
|
start = second
|
|
end = first
|
|
else:
|
|
start = first
|
|
end = second
|
|
return (start, end)
|
|
|
|
|
|
def GetDaylightSavingsAttribs():
|
|
times = GetDaylightSavingsTimes()
|
|
if not times:
|
|
return None
|
|
(start, end) = times
|
|
def DstMonth(t):
|
|
return time.localtime(t)[1] - 1
|
|
def DstHour(t):
|
|
return time.localtime(t - 1)[3] + 1
|
|
def DstSunday(t):
|
|
if time.localtime(t)[2] > 15:
|
|
return "'last'"
|
|
else:
|
|
return "'first'"
|
|
def DstMinutes(t):
|
|
return (time.localtime(t - 1)[4] + 1) % 60
|
|
attribs = { }
|
|
attribs['start_month'] = DstMonth(start)
|
|
attribs['end_month'] = DstMonth(end)
|
|
attribs['start_sunday'] = DstSunday(start)
|
|
attribs['end_sunday'] = DstSunday(end)
|
|
attribs['start_hour'] = DstHour(start)
|
|
attribs['end_hour'] = DstHour(end)
|
|
attribs['start_minutes'] = DstMinutes(start)
|
|
attribs['end_minutes'] = DstMinutes(end)
|
|
return attribs
|
|
|
|
*********/
|