284 lines
6.7 KiB
JavaScript
284 lines
6.7 KiB
JavaScript
'use strict';
|
|
|
|
const IMF_DAYS = ['mon', 'tue', 'wed', 'thu', 'fri', 'sat', 'sun'];
|
|
const IMF_SPACES = [4, 7, 11, 16, 25];
|
|
const IMF_MONTHS = [
|
|
'jan',
|
|
'feb',
|
|
'mar',
|
|
'apr',
|
|
'may',
|
|
'jun',
|
|
'jul',
|
|
'aug',
|
|
'sep',
|
|
'oct',
|
|
'nov',
|
|
'dec',
|
|
];
|
|
const IMF_COLONS = [19, 22];
|
|
|
|
const ASCTIME_SPACES = [3, 7, 10, 19];
|
|
|
|
const RFC850_DAYS = [
|
|
'monday',
|
|
'tuesday',
|
|
'wednesday',
|
|
'thursday',
|
|
'friday',
|
|
'saturday',
|
|
'sunday',
|
|
];
|
|
|
|
/**
|
|
* @see https://www.rfc-editor.org/rfc/rfc9110.html#name-date-time-formats
|
|
*
|
|
* @param {string} date
|
|
* @param {Date} [now]
|
|
* @returns {Date | undefined}
|
|
*/
|
|
function parseHttpDate(date, now) {
|
|
// Sun, 06 Nov 1994 08:49:37 GMT ; IMF-fixdate
|
|
// Sun Nov 6 08:49:37 1994 ; ANSI C's asctime() format
|
|
// Sunday, 06-Nov-94 08:49:37 GMT ; obsolete RFC 850 format
|
|
|
|
date = date.toLowerCase();
|
|
|
|
switch (date[3]) {
|
|
case ',':
|
|
return parseImfDate(date);
|
|
case ' ':
|
|
return parseAscTimeDate(date);
|
|
default:
|
|
return parseRfc850Date(date, now);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @see https://httpwg.org/specs/rfc9110.html#preferred.date.format
|
|
*
|
|
* @param {string} date
|
|
* @returns {Date | undefined}
|
|
*/
|
|
function parseImfDate(date) {
|
|
if (date.length !== 29) {
|
|
return undefined;
|
|
}
|
|
|
|
if (!date.endsWith('gmt')) {
|
|
// Unsupported timezone
|
|
return undefined;
|
|
}
|
|
|
|
for (const spaceInx of IMF_SPACES) {
|
|
if (date[spaceInx] !== ' ') {
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
for (const colonIdx of IMF_COLONS) {
|
|
if (date[colonIdx] !== ':') {
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
const dayName = date.substring(0, 3);
|
|
if (!IMF_DAYS.includes(dayName)) {
|
|
return undefined;
|
|
}
|
|
|
|
const dayString = date.substring(5, 7);
|
|
const day = Number.parseInt(dayString);
|
|
if (isNaN(day) || (day < 10 && dayString[0] !== '0')) {
|
|
// Not a number, 0, or it's less than 10 and didn't start with a 0
|
|
return undefined;
|
|
}
|
|
|
|
const month = date.substring(8, 11);
|
|
const monthIdx = IMF_MONTHS.indexOf(month);
|
|
if (monthIdx === -1) {
|
|
return undefined;
|
|
}
|
|
|
|
const year = Number.parseInt(date.substring(12, 16));
|
|
if (isNaN(year)) {
|
|
return undefined;
|
|
}
|
|
|
|
const hourString = date.substring(17, 19);
|
|
const hour = Number.parseInt(hourString);
|
|
if (isNaN(hour) || (hour < 10 && hourString[0] !== '0')) {
|
|
return undefined;
|
|
}
|
|
|
|
const minuteString = date.substring(20, 22);
|
|
const minute = Number.parseInt(minuteString);
|
|
if (isNaN(minute) || (minute < 10 && minuteString[0] !== '0')) {
|
|
return undefined;
|
|
}
|
|
|
|
const secondString = date.substring(23, 25);
|
|
const second = Number.parseInt(secondString);
|
|
if (isNaN(second) || (second < 10 && secondString[0] !== '0')) {
|
|
return undefined;
|
|
}
|
|
|
|
return new Date(Date.UTC(year, monthIdx, day, hour, minute, second));
|
|
}
|
|
|
|
/**
|
|
* @see https://httpwg.org/specs/rfc9110.html#obsolete.date.formats
|
|
*
|
|
* @param {string} date
|
|
* @returns {Date | undefined}
|
|
*/
|
|
function parseAscTimeDate(date) {
|
|
// This is assumed to be in UTC
|
|
|
|
if (date.length !== 24) {
|
|
return undefined;
|
|
}
|
|
|
|
for (const spaceIdx of ASCTIME_SPACES) {
|
|
if (date[spaceIdx] !== ' ') {
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
const dayName = date.substring(0, 3);
|
|
if (!IMF_DAYS.includes(dayName)) {
|
|
return undefined;
|
|
}
|
|
|
|
const month = date.substring(4, 7);
|
|
const monthIdx = IMF_MONTHS.indexOf(month);
|
|
if (monthIdx === -1) {
|
|
return undefined;
|
|
}
|
|
|
|
const dayString = date.substring(8, 10);
|
|
const day = Number.parseInt(dayString);
|
|
if (isNaN(day) || (day < 10 && dayString[0] !== ' ')) {
|
|
return undefined;
|
|
}
|
|
|
|
const hourString = date.substring(11, 13);
|
|
const hour = Number.parseInt(hourString);
|
|
if (isNaN(hour) || (hour < 10 && hourString[0] !== '0')) {
|
|
return undefined;
|
|
}
|
|
|
|
const minuteString = date.substring(14, 16);
|
|
const minute = Number.parseInt(minuteString);
|
|
if (isNaN(minute) || (minute < 10 && minuteString[0] !== '0')) {
|
|
return undefined;
|
|
}
|
|
|
|
const secondString = date.substring(17, 19);
|
|
const second = Number.parseInt(secondString);
|
|
if (isNaN(second) || (second < 10 && secondString[0] !== '0')) {
|
|
return undefined;
|
|
}
|
|
|
|
const year = Number.parseInt(date.substring(20, 24));
|
|
if (isNaN(year)) {
|
|
return undefined;
|
|
}
|
|
|
|
return new Date(Date.UTC(year, monthIdx, day, hour, minute, second));
|
|
}
|
|
|
|
/**
|
|
* @see https://httpwg.org/specs/rfc9110.html#obsolete.date.formats
|
|
*
|
|
* @param {string} date
|
|
* @param {Date} [now]
|
|
* @returns {Date | undefined}
|
|
*/
|
|
function parseRfc850Date(date, now = new Date()) {
|
|
if (!date.endsWith('gmt')) {
|
|
// Unsupported timezone
|
|
return undefined;
|
|
}
|
|
|
|
const commaIndex = date.indexOf(',');
|
|
if (commaIndex === -1) {
|
|
return undefined;
|
|
}
|
|
|
|
if (date.length - commaIndex - 1 !== 23) {
|
|
return undefined;
|
|
}
|
|
|
|
const dayName = date.substring(0, commaIndex);
|
|
if (!RFC850_DAYS.includes(dayName)) {
|
|
return undefined;
|
|
}
|
|
|
|
if (
|
|
date[commaIndex + 1] !== ' ' ||
|
|
date[commaIndex + 4] !== '-' ||
|
|
date[commaIndex + 8] !== '-' ||
|
|
date[commaIndex + 11] !== ' ' ||
|
|
date[commaIndex + 14] !== ':' ||
|
|
date[commaIndex + 17] !== ':' ||
|
|
date[commaIndex + 20] !== ' '
|
|
) {
|
|
return undefined;
|
|
}
|
|
|
|
const dayString = date.substring(commaIndex + 2, commaIndex + 4);
|
|
const day = Number.parseInt(dayString);
|
|
if (isNaN(day) || (day < 10 && dayString[0] !== '0')) {
|
|
// Not a number, or it's less than 10 and didn't start with a 0
|
|
return undefined;
|
|
}
|
|
|
|
const month = date.substring(commaIndex + 5, commaIndex + 8);
|
|
const monthIdx = IMF_MONTHS.indexOf(month);
|
|
if (monthIdx === -1) {
|
|
return undefined;
|
|
}
|
|
|
|
// As of this point year is just the decade (i.e. 94)
|
|
let year = Number.parseInt(date.substring(commaIndex + 9, commaIndex + 11));
|
|
if (isNaN(year)) {
|
|
return undefined;
|
|
}
|
|
|
|
const currentYear = now.getUTCFullYear();
|
|
const currentDecade = currentYear % 100;
|
|
const currentCentury = Math.floor(currentYear / 100);
|
|
|
|
if (year > currentDecade && year - currentDecade >= 50) {
|
|
// Over 50 years in future, go to previous century
|
|
year += (currentCentury - 1) * 100;
|
|
} else {
|
|
year += currentCentury * 100;
|
|
}
|
|
|
|
const hourString = date.substring(commaIndex + 12, commaIndex + 14);
|
|
const hour = Number.parseInt(hourString);
|
|
if (isNaN(hour) || (hour < 10 && hourString[0] !== '0')) {
|
|
return undefined;
|
|
}
|
|
|
|
const minuteString = date.substring(commaIndex + 15, commaIndex + 17);
|
|
const minute = Number.parseInt(minuteString);
|
|
if (isNaN(minute) || (minute < 10 && minuteString[0] !== '0')) {
|
|
return undefined;
|
|
}
|
|
|
|
const secondString = date.substring(commaIndex + 18, commaIndex + 20);
|
|
const second = Number.parseInt(secondString);
|
|
if (isNaN(second) || (second < 10 && secondString[0] !== '0')) {
|
|
return undefined;
|
|
}
|
|
|
|
return new Date(Date.UTC(year, monthIdx, day, hour, minute, second));
|
|
}
|
|
|
|
module.exports = {
|
|
parseHttpDate,
|
|
};
|