2023-11-27 06:33:21 -08:00
|
|
|
import { ExpressionExtensionError } from '../errors/expression-extension.error';
|
2023-07-31 02:00:48 -07:00
|
|
|
|
2023-01-27 05:56:56 -08:00
|
|
|
import { DateTime } from 'luxon';
|
2023-03-30 04:59:59 -07:00
|
|
|
import type {
|
|
|
|
DateTimeUnit,
|
|
|
|
DurationLike,
|
|
|
|
DurationObjectUnits,
|
|
|
|
LocaleOptions,
|
|
|
|
WeekdayNumbers,
|
|
|
|
} from 'luxon';
|
2023-01-10 05:06:12 -08:00
|
|
|
import type { ExtensionMap } from './Extensions';
|
2023-02-15 01:50:16 -08:00
|
|
|
import { convertToDateTime } from './utils';
|
2023-01-10 05:06:12 -08:00
|
|
|
|
|
|
|
type DurationUnit =
|
|
|
|
| 'milliseconds'
|
|
|
|
| 'seconds'
|
|
|
|
| 'minutes'
|
|
|
|
| 'hours'
|
|
|
|
| 'days'
|
|
|
|
| 'weeks'
|
|
|
|
| 'months'
|
|
|
|
| 'quarter'
|
|
|
|
| 'years';
|
|
|
|
type DatePart =
|
|
|
|
| 'day'
|
2023-02-02 03:35:38 -08:00
|
|
|
| 'week'
|
2023-01-10 05:06:12 -08:00
|
|
|
| 'month'
|
|
|
|
| 'year'
|
|
|
|
| 'hour'
|
|
|
|
| 'minute'
|
|
|
|
| 'second'
|
|
|
|
| 'millisecond'
|
|
|
|
| 'weekNumber'
|
|
|
|
| 'yearDayNumber'
|
|
|
|
| 'weekday';
|
|
|
|
|
|
|
|
const DURATION_MAP: Record<string, DurationUnit> = {
|
|
|
|
day: 'days',
|
|
|
|
month: 'months',
|
|
|
|
year: 'years',
|
|
|
|
week: 'weeks',
|
|
|
|
hour: 'hours',
|
|
|
|
minute: 'minutes',
|
|
|
|
second: 'seconds',
|
|
|
|
millisecond: 'milliseconds',
|
|
|
|
ms: 'milliseconds',
|
|
|
|
sec: 'seconds',
|
|
|
|
secs: 'seconds',
|
|
|
|
hr: 'hours',
|
|
|
|
hrs: 'hours',
|
|
|
|
min: 'minutes',
|
|
|
|
mins: 'minutes',
|
|
|
|
};
|
|
|
|
|
|
|
|
const DATETIMEUNIT_MAP: Record<string, DateTimeUnit> = {
|
|
|
|
days: 'day',
|
|
|
|
months: 'month',
|
|
|
|
years: 'year',
|
|
|
|
hours: 'hour',
|
|
|
|
minutes: 'minute',
|
|
|
|
seconds: 'second',
|
|
|
|
milliseconds: 'millisecond',
|
|
|
|
hrs: 'hour',
|
|
|
|
hr: 'hour',
|
|
|
|
mins: 'minute',
|
|
|
|
min: 'minute',
|
|
|
|
secs: 'second',
|
|
|
|
sec: 'second',
|
|
|
|
ms: 'millisecond',
|
|
|
|
};
|
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
function isDateTime(date: unknown): date is DateTime {
|
|
|
|
return date ? DateTime.isDateTime(date) : false;
|
2023-01-10 05:06:12 -08:00
|
|
|
}
|
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
function generateDurationObject(durationValue: number, unit: DurationUnit): DurationObjectUnits {
|
2023-01-10 05:06:12 -08:00
|
|
|
const convertedUnit = DURATION_MAP[unit] || unit;
|
2023-06-15 01:07:47 -07:00
|
|
|
return { [`${convertedUnit}`]: durationValue };
|
2023-01-10 05:06:12 -08:00
|
|
|
}
|
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
function beginningOf(date: Date | DateTime, extraArgs: DurationUnit[]): Date | DateTime {
|
|
|
|
const [rawUnit = 'week'] = extraArgs;
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
const unit = DATETIMEUNIT_MAP[rawUnit] || rawUnit;
|
|
|
|
|
|
|
|
if (isDateTime(date)) return date.startOf(unit);
|
|
|
|
|
|
|
|
return DateTime.fromJSDate(date).startOf(unit).toJSDate();
|
2023-01-10 05:06:12 -08:00
|
|
|
}
|
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
function endOfMonth(date: Date | DateTime): Date | DateTime {
|
|
|
|
if (isDateTime(date)) return date.endOf('month');
|
|
|
|
|
2023-01-10 05:06:12 -08:00
|
|
|
return DateTime.fromJSDate(date).endOf('month').toJSDate();
|
|
|
|
}
|
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
function extract(date: Date | DateTime, args: DatePart[]): number {
|
|
|
|
let [part = 'week'] = args;
|
|
|
|
|
2023-01-10 05:06:12 -08:00
|
|
|
if (part === 'yearDayNumber') {
|
2023-06-15 01:07:47 -07:00
|
|
|
date = isDateTime(date) ? date.toJSDate() : date;
|
|
|
|
|
2023-01-10 05:06:12 -08:00
|
|
|
const firstDayOfTheYear = new Date(date.getFullYear(), 0, 0);
|
2023-06-15 01:07:47 -07:00
|
|
|
|
2023-01-10 05:06:12 -08:00
|
|
|
const diff =
|
|
|
|
date.getTime() -
|
|
|
|
firstDayOfTheYear.getTime() +
|
|
|
|
(firstDayOfTheYear.getTimezoneOffset() - date.getTimezoneOffset()) * 60 * 1000;
|
2023-06-15 01:07:47 -07:00
|
|
|
|
2023-01-10 05:06:12 -08:00
|
|
|
return Math.floor(diff / (1000 * 60 * 60 * 24));
|
|
|
|
}
|
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
if (part === 'week') part = 'weekNumber';
|
|
|
|
|
|
|
|
const unit = (DATETIMEUNIT_MAP[part] as keyof DateTime) || part;
|
|
|
|
|
|
|
|
if (isDateTime(date)) return date.get(unit);
|
2023-02-02 03:35:38 -08:00
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
return DateTime.fromJSDate(date).get(unit);
|
2023-01-10 05:06:12 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
function format(date: Date | DateTime, extraArgs: unknown[]): string {
|
|
|
|
const [dateFormat, localeOpts = {}] = extraArgs as [string, LocaleOptions];
|
|
|
|
if (isDateTime(date)) {
|
|
|
|
return date.toFormat(dateFormat, { ...localeOpts });
|
|
|
|
}
|
|
|
|
return DateTime.fromJSDate(date).toFormat(dateFormat, { ...localeOpts });
|
|
|
|
}
|
|
|
|
|
2023-02-15 01:50:16 -08:00
|
|
|
function isBetween(
|
|
|
|
date: Date | DateTime,
|
|
|
|
extraArgs: Array<string | Date | DateTime>,
|
|
|
|
): boolean | undefined {
|
|
|
|
if (extraArgs.length !== 2) {
|
|
|
|
throw new ExpressionExtensionError('isBetween(): expected exactly two args');
|
|
|
|
}
|
|
|
|
|
|
|
|
const [first, second] = extraArgs;
|
|
|
|
|
|
|
|
const firstDate = convertToDateTime(first);
|
|
|
|
const secondDate = convertToDateTime(second);
|
|
|
|
|
|
|
|
if (!firstDate || !secondDate) {
|
|
|
|
return;
|
|
|
|
}
|
2023-01-10 05:06:12 -08:00
|
|
|
|
|
|
|
if (firstDate > secondDate) {
|
|
|
|
return secondDate < date && date < firstDate;
|
|
|
|
}
|
|
|
|
return secondDate > date && date > firstDate;
|
|
|
|
}
|
|
|
|
|
2023-01-23 05:28:17 -08:00
|
|
|
function isDst(date: Date | DateTime): boolean {
|
|
|
|
if (isDateTime(date)) {
|
|
|
|
return date.isInDST;
|
|
|
|
}
|
2023-01-10 05:06:12 -08:00
|
|
|
return DateTime.fromJSDate(date).isInDST;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isInLast(date: Date | DateTime, extraArgs: unknown[]): boolean {
|
|
|
|
const [durationValue = 0, unit = 'minutes'] = extraArgs as [number, DurationUnit];
|
|
|
|
|
|
|
|
const dateInThePast = DateTime.now().minus(generateDurationObject(durationValue, unit));
|
|
|
|
let thisDate = date;
|
|
|
|
if (!isDateTime(thisDate)) {
|
|
|
|
thisDate = DateTime.fromJSDate(thisDate);
|
|
|
|
}
|
|
|
|
return dateInThePast <= thisDate && thisDate <= DateTime.now();
|
|
|
|
}
|
|
|
|
|
2023-03-30 04:59:59 -07:00
|
|
|
const WEEKEND_DAYS: WeekdayNumbers[] = [6, 7];
|
2023-01-23 05:28:17 -08:00
|
|
|
function isWeekend(date: Date | DateTime): boolean {
|
2023-03-30 04:59:59 -07:00
|
|
|
const { weekday } = isDateTime(date) ? date : DateTime.fromJSDate(date);
|
|
|
|
return WEEKEND_DAYS.includes(weekday);
|
2023-01-10 05:06:12 -08:00
|
|
|
}
|
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
function minus(
|
|
|
|
date: Date | DateTime,
|
|
|
|
args: [DurationLike] | [number, DurationUnit],
|
|
|
|
): Date | DateTime {
|
|
|
|
if (args.length === 1) {
|
|
|
|
const [arg] = args;
|
2023-01-16 05:01:58 -08:00
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
if (isDateTime(date)) return date.minus(arg);
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
return DateTime.fromJSDate(date).minus(arg).toJSDate();
|
2023-01-10 05:06:12 -08:00
|
|
|
}
|
2023-06-15 01:07:47 -07:00
|
|
|
|
|
|
|
const [durationValue = 0, unit = 'minutes'] = args;
|
|
|
|
|
|
|
|
const duration = generateDurationObject(durationValue, unit);
|
|
|
|
|
|
|
|
if (isDateTime(date)) return date.minus(duration);
|
|
|
|
|
|
|
|
return DateTime.fromJSDate(date).minus(duration).toJSDate();
|
2023-01-10 05:06:12 -08:00
|
|
|
}
|
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
function plus(
|
|
|
|
date: Date | DateTime,
|
|
|
|
args: [DurationLike] | [number, DurationUnit],
|
|
|
|
): Date | DateTime {
|
|
|
|
if (args.length === 1) {
|
|
|
|
const [arg] = args;
|
2023-01-16 05:01:58 -08:00
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
if (isDateTime(date)) return date.plus(arg);
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-06-15 01:07:47 -07:00
|
|
|
return DateTime.fromJSDate(date).plus(arg).toJSDate();
|
2023-01-10 05:06:12 -08:00
|
|
|
}
|
2023-06-15 01:07:47 -07:00
|
|
|
|
|
|
|
const [durationValue = 0, unit = 'minutes'] = args;
|
|
|
|
|
|
|
|
const duration = generateDurationObject(durationValue, unit);
|
|
|
|
|
|
|
|
if (isDateTime(date)) return date.plus(duration);
|
|
|
|
|
|
|
|
return DateTime.fromJSDate(date).plus(duration).toJSDate();
|
2023-01-10 05:06:12 -08:00
|
|
|
}
|
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
endOfMonth.doc = {
|
|
|
|
name: 'endOfMonth',
|
|
|
|
returnType: 'Date',
|
2023-02-27 20:34:03 -08:00
|
|
|
description: 'Transforms a date to the last possible moment that lies within the month.',
|
2023-11-20 05:40:28 -08:00
|
|
|
docURL: 'https://docs.n8n.io/code/builtin/data-transformation-functions/dates/#date-endOfMonth',
|
2023-02-02 03:35:38 -08:00
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
isDst.doc = {
|
|
|
|
name: 'isDst',
|
|
|
|
returnType: 'boolean',
|
2023-02-27 20:34:03 -08:00
|
|
|
description: 'Checks if a Date is within Daylight Savings Time.',
|
2023-11-20 05:40:28 -08:00
|
|
|
docURL: 'https://docs.n8n.io/code/builtin/data-transformation-functions/dates/#date-isDst',
|
2023-02-02 03:35:38 -08:00
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
isWeekend.doc = {
|
|
|
|
name: 'isWeekend',
|
|
|
|
returnType: 'boolean',
|
2023-02-27 20:34:03 -08:00
|
|
|
description: 'Checks if the Date falls on a Saturday or Sunday.',
|
2023-11-20 05:40:28 -08:00
|
|
|
docURL: 'https://docs.n8n.io/code/builtin/data-transformation-functions/dates/#date-isWeekend',
|
2023-02-02 03:35:38 -08:00
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
beginningOf.doc = {
|
|
|
|
name: 'beginningOf',
|
2023-02-27 20:34:03 -08:00
|
|
|
description: 'Transform a Date to the start of the given time period. Default unit is `week`.',
|
2023-02-02 03:35:38 -08:00
|
|
|
returnType: 'Date',
|
2023-02-27 20:34:03 -08:00
|
|
|
args: [{ name: 'unit?', type: 'DurationUnit' }],
|
2023-11-20 05:40:28 -08:00
|
|
|
docURL: 'https://docs.n8n.io/code/builtin/data-transformation-functions/dates/#date-beginningOf',
|
2023-02-02 03:35:38 -08:00
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
extract.doc = {
|
|
|
|
name: 'extract',
|
2023-02-27 20:34:03 -08:00
|
|
|
description: 'Extracts the part defined in `datePart` from a Date. Default unit is `week`.',
|
2023-02-02 03:35:38 -08:00
|
|
|
returnType: 'number',
|
2023-02-27 20:34:03 -08:00
|
|
|
args: [{ name: 'datePart?', type: 'DurationUnit' }],
|
2023-11-20 05:40:28 -08:00
|
|
|
docURL: 'https://docs.n8n.io/code/builtin/data-transformation-functions/dates/#date-extract',
|
2023-02-02 03:35:38 -08:00
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
format.doc = {
|
|
|
|
name: 'format',
|
2023-02-27 20:34:03 -08:00
|
|
|
description: 'Formats a Date in the given structure.',
|
|
|
|
returnType: 'string',
|
|
|
|
args: [{ name: 'fmt', type: 'TimeFormat' }],
|
2023-11-20 05:40:28 -08:00
|
|
|
docURL: 'https://docs.n8n.io/code/builtin/data-transformation-functions/dates/#date-format',
|
2023-02-02 03:35:38 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
isBetween.doc = {
|
|
|
|
name: 'isBetween',
|
2023-02-27 20:34:03 -08:00
|
|
|
description: 'Checks if a Date is between two given dates.',
|
2023-02-02 03:35:38 -08:00
|
|
|
returnType: 'boolean',
|
2023-02-27 20:34:03 -08:00
|
|
|
args: [
|
|
|
|
{ name: 'date1', type: 'Date|string' },
|
|
|
|
{ name: 'date2', type: 'Date|string' },
|
|
|
|
],
|
2023-11-20 05:40:28 -08:00
|
|
|
docURL: 'https://docs.n8n.io/code/builtin/data-transformation-functions/dates/#date-isBetween',
|
2023-02-02 03:35:38 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
isInLast.doc = {
|
|
|
|
name: 'isInLast',
|
2023-02-27 20:34:03 -08:00
|
|
|
description: 'Checks if a Date is within a given time period. Default unit is `minute`.',
|
2023-02-02 03:35:38 -08:00
|
|
|
returnType: 'boolean',
|
2023-02-27 20:34:03 -08:00
|
|
|
args: [
|
|
|
|
{ name: 'n', type: 'number' },
|
|
|
|
{ name: 'unit?', type: 'DurationUnit' },
|
|
|
|
],
|
2023-11-20 05:40:28 -08:00
|
|
|
docURL: 'https://docs.n8n.io/code/builtin/data-transformation-functions/dates/#date-isInLast',
|
2023-02-02 03:35:38 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
minus.doc = {
|
|
|
|
name: 'minus',
|
2023-02-27 20:34:03 -08:00
|
|
|
description: 'Subtracts a given time period from a Date. Default unit is `minute`.',
|
2023-02-02 03:35:38 -08:00
|
|
|
returnType: 'Date',
|
2023-02-27 20:34:03 -08:00
|
|
|
args: [
|
|
|
|
{ name: 'n', type: 'number' },
|
|
|
|
{ name: 'unit?', type: 'DurationUnit' },
|
|
|
|
],
|
2023-11-20 05:40:28 -08:00
|
|
|
docURL: 'https://docs.n8n.io/code/builtin/data-transformation-functions/dates/#date-minus',
|
2023-02-02 03:35:38 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
plus.doc = {
|
|
|
|
name: 'plus',
|
2023-02-27 20:34:03 -08:00
|
|
|
description: 'Adds a given time period to a Date. Default unit is `minute`.',
|
2023-02-02 03:35:38 -08:00
|
|
|
returnType: 'Date',
|
2023-02-27 20:34:03 -08:00
|
|
|
args: [
|
|
|
|
{ name: 'n', type: 'number' },
|
|
|
|
{ name: 'unit?', type: 'DurationUnit' },
|
|
|
|
],
|
2023-11-20 05:40:28 -08:00
|
|
|
docURL: 'https://docs.n8n.io/code/builtin/data-transformation-functions/dates/#date-plus',
|
2023-02-02 03:35:38 -08:00
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
|
|
|
export const dateExtensions: ExtensionMap = {
|
|
|
|
typeName: 'Date',
|
|
|
|
functions: {
|
|
|
|
beginningOf,
|
|
|
|
endOfMonth,
|
|
|
|
extract,
|
|
|
|
isBetween,
|
|
|
|
isDst,
|
|
|
|
isInLast,
|
|
|
|
isWeekend,
|
|
|
|
minus,
|
|
|
|
plus,
|
|
|
|
format,
|
|
|
|
},
|
|
|
|
};
|