* 🔥 Remove test extensions * 🚧 Add test description * 📘 Expand types * ⚡ Export extensions * ⚡ Export collection * ⚡ Mark all proxies * ✏️ Rename for clarity * ⚡ Export from barrel * ✨ Create datatype completions * ⚡ Mount datatype completions * 🧪 Adjust tests * ⚡ Add `path` prop * 🔥 Remove `()` from completion labels * ⚡ Filter out completions for pseudo-proxies * 🐛 Fix method error * ⚡ Add metrics * ✏️ Improve naming * ✨ Start completion on empty resolvable * ✨ Implement completion previews * ⚡ Break out completion manager * ⚡ Implement in expression editor modal * ✏️ Improve naming * ⚡ Filter out irrelevant completions * ✨ Add preview hint * ✏️ Improve comments * 🎨 Style preview hint * ⚡ Expand `hasNoParams` * ⚡ Add spacing for readability * ⚡ Add error codes * ✏️ Add comment * 🐛 Fix Esc behavior * ⚡ Parse Unicode * ⚡ Throw on invalid `DateTime` * ⚡ Fix second root completion detection * ⚡ Switch message at completable prefix position * 🐛 Fix function names for non-dev build * 🐛 Fix `json` handling * 🔥 Comment out previews * ♻️ Apply feedback * 🔥 Remove extensions * 🚚 Rename extensions * ⚡ Adjust some implementations * 🔥 Remove dummy extensions * 🐛 Fix object regex * ♻️ Apply feedback * ✏️ Fix typos * ✏️ Add `fn is not a function` message * 🔥 Remove check * ✨ Add `isNotEmpty` for objects * 🚚 Rename `global` to `alpha` * 🔥 Remove `encrypt` * ⚡ Restore `is not a function` error * ⚡ Support `week` on `extract()` * 🧪 Fix tests * ⚡ Add validation to some string extensions * ⚡ Validate number arrays in some extensions * 🧪 Fix tests * ✏️ Improve error message * ⏪ Revert extensions framework changes * 🧹 Previews cleanup * ⚡ Condense blank completions * ⚡ Refactor dollar completions * ⚡ Refactor non-dollar completions * ⚡ Refactor Luxon completions * ⚡ Refactor datatype completions * ⚡ Use `DATETIMEUNIT_MAP` * ✏️ Update test description * ⏪ Revert "Use `DATETIMEUNIT_MAP`" This reverts commit 472a77df5cd789905d162f3c3db02ac767b89b4e. * 🧪 Add tests * ♻️ Restore generic extensions * 🔥 Remove logs * 🧪 Expand tests * ✨ Add `Math` completions * ✏️ List breaking change * ⚡ Add doc tooltips * 🐛 Fix node selector regex * 🐛 Fix `context` resolution * 🐛 Allow dollar completions in args * ⚡ Make numeric array methods context-dependent * 📝 Adjust docs * 🐛 Fix selector ref * ⚡ Surface error for valid URL * 🐛 Disallow whitespace in `isEmail` check * 🧪 Fix test for `isUrl` * ⚡ Add comma validator in `toFloat` * ⚡ Add validation to `$jmespath()` * ⏪ Revert valid URL error * ⚡ Adjust `$jmespath()` validation * 🧪 Adjust `isUrl` test * ⚡ Remove `{}` and `[]` from compact * ✏️ Update docs * 🚚 Rename `stripTags` to `removeTags` * ⚡ Do not inject whitespace inside resolvable * ⚡ Make completions aware of `()` * ✏️ Add note * ⚡ Update sorting * ⚡ Hide active node name from node selector * 🔥 Remove `length()` and its aliases * ⚡ Validate non-zero for `chunk` * ✏️ Reword all error messages * 🐛 Fix `$now` and `$today` * ⚡ Simplify with `stripExcessParens` * ⚡ Fold luxon into datatype * 🧪 Clean up tests * 🔥 Remove tests for removed methods * 👕 Fix type * ⬆️ Upgrade lang pack * ⏪ Undo change to `vitest` command * 🔥 Remove unused method * ⚡ Separate `return` line * ✏️ Improve description * 🧪 Expand tests for initial-only completions * 🧪 Add bracket-aware completions * ⚡ Make check for `all()` stricter * ✏️ Adjust explanatory comments * 🔥 Remove unneded copy * 🔥 Remove outdated comment * ⚡ Make naming consistent * ✏️ Update comments * ⚡ Improve URL scheme check * ✏️ Add comment * 🚚 Move extension * ✏️ Update `BREAKING-CHANGES.md` * ✏️ Update upcoming version * ✏️ Fix grammar * ✏️ Shorten message * 🐛 Fix `Esc` behavior * 🐛 Fix `isNumeric` * ✨ Support native methods * 🧪 Skip Pinia tests * ✏️ Shorten description * 🔥 Remove outdated comment * 🧪 Unskip Pinia tests * ✏️ Add comments * 🧪 Expand tests to natives * ✏️ Add clarifying comments * ⚡ Use `setTimeout` to make telemetry non-blocking * 🐛 Account for no active node in cred modal * ✨ Resolve without workflow * 🔥 Remove `Esc` handling on NDV * ⚡ Use `isDateTime` * 🚚 Move `unique` to next phase This array extension takes optional args. * ⚡ Merge export * 🧪 Fix tests * ⏪ Restore check * ✏️ Make breaking change description more accurate * 🧪 Fix e2e tests
265 lines
6.3 KiB
TypeScript
265 lines
6.3 KiB
TypeScript
/* eslint-disable @typescript-eslint/unbound-method */
|
|
/* eslint-disable @typescript-eslint/explicit-member-accessibility */
|
|
import { DateTime } from 'luxon';
|
|
import type { DateTimeUnit, DurationLike, DurationObjectUnits, LocaleOptions } from 'luxon';
|
|
import type { ExtensionMap } from './Extensions';
|
|
|
|
type DurationUnit =
|
|
| 'milliseconds'
|
|
| 'seconds'
|
|
| 'minutes'
|
|
| 'hours'
|
|
| 'days'
|
|
| 'weeks'
|
|
| 'months'
|
|
| 'quarter'
|
|
| 'years';
|
|
type DatePart =
|
|
| 'day'
|
|
| 'week'
|
|
| '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',
|
|
};
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
function isDateTime(date: any): date is DateTime {
|
|
if (date) {
|
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
|
|
return DateTime.isDateTime(date);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function generateDurationObject(durationValue: number, unit: DurationUnit) {
|
|
const convertedUnit = DURATION_MAP[unit] || unit;
|
|
return { [`${convertedUnit}`]: durationValue } as DurationObjectUnits;
|
|
}
|
|
|
|
function beginningOf(date: Date | DateTime, extraArgs: DurationUnit[]): Date {
|
|
const [unit = 'week'] = extraArgs;
|
|
|
|
if (isDateTime(date)) {
|
|
return date.startOf(DATETIMEUNIT_MAP[unit] || unit).toJSDate();
|
|
}
|
|
let datetime = DateTime.fromJSDate(date);
|
|
if (date.getTimezoneOffset() === 0) {
|
|
datetime = datetime.setZone('UTC');
|
|
}
|
|
return datetime.startOf(DATETIMEUNIT_MAP[unit] || unit).toJSDate();
|
|
}
|
|
|
|
function endOfMonth(date: Date | DateTime): Date {
|
|
if (isDateTime(date)) {
|
|
return date.endOf('month').toJSDate();
|
|
}
|
|
return DateTime.fromJSDate(date).endOf('month').toJSDate();
|
|
}
|
|
|
|
function extract(inputDate: Date | DateTime, extraArgs: DatePart[]): number | Date {
|
|
let [part] = extraArgs;
|
|
let date = inputDate;
|
|
if (isDateTime(date)) {
|
|
date = date.toJSDate();
|
|
}
|
|
if (part === 'yearDayNumber') {
|
|
const firstDayOfTheYear = new Date(date.getFullYear(), 0, 0);
|
|
const diff =
|
|
date.getTime() -
|
|
firstDayOfTheYear.getTime() +
|
|
(firstDayOfTheYear.getTimezoneOffset() - date.getTimezoneOffset()) * 60 * 1000;
|
|
return Math.floor(diff / (1000 * 60 * 60 * 24));
|
|
}
|
|
|
|
if (part === 'week') {
|
|
part = 'weekNumber';
|
|
}
|
|
|
|
return DateTime.fromJSDate(date).get((DATETIMEUNIT_MAP[part] as keyof DateTime) || part);
|
|
}
|
|
|
|
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 });
|
|
}
|
|
|
|
function isBetween(date: Date | DateTime, extraArgs: unknown[]): boolean {
|
|
const [first, second] = extraArgs as string[];
|
|
const firstDate = new Date(first);
|
|
const secondDate = new Date(second);
|
|
|
|
if (firstDate > secondDate) {
|
|
return secondDate < date && date < firstDate;
|
|
}
|
|
return secondDate > date && date > firstDate;
|
|
}
|
|
|
|
function isDst(date: Date | DateTime): boolean {
|
|
if (isDateTime(date)) {
|
|
return date.isInDST;
|
|
}
|
|
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();
|
|
}
|
|
|
|
function isWeekend(date: Date | DateTime): boolean {
|
|
enum DAYS {
|
|
saturday = 6,
|
|
sunday = 7,
|
|
}
|
|
if (isDateTime(date)) {
|
|
return [DAYS.saturday, DAYS.sunday].includes(date.weekday);
|
|
}
|
|
return [DAYS.saturday, DAYS.sunday].includes(DateTime.fromJSDate(date).weekday);
|
|
}
|
|
|
|
function minus(date: Date | DateTime, extraArgs: unknown[]): Date | DateTime {
|
|
if (isDateTime(date) && extraArgs.length === 1) {
|
|
return date.minus(extraArgs[0] as DurationLike);
|
|
}
|
|
|
|
const [durationValue = 0, unit = 'minutes'] = extraArgs as [number, DurationUnit];
|
|
|
|
if (isDateTime(date)) {
|
|
return date.minus(generateDurationObject(durationValue, unit)).toJSDate();
|
|
}
|
|
return DateTime.fromJSDate(date).minus(generateDurationObject(durationValue, unit)).toJSDate();
|
|
}
|
|
|
|
function plus(date: Date | DateTime, extraArgs: unknown[]): Date | DateTime {
|
|
if (isDateTime(date) && extraArgs.length === 1) {
|
|
return date.plus(extraArgs[0] as DurationLike);
|
|
}
|
|
|
|
const [durationValue = 0, unit = 'minutes'] = extraArgs as [number, DurationUnit];
|
|
|
|
if (isDateTime(date)) {
|
|
return date.plus(generateDurationObject(durationValue, unit)).toJSDate();
|
|
}
|
|
return DateTime.fromJSDate(date).plus(generateDurationObject(durationValue, unit)).toJSDate();
|
|
}
|
|
|
|
endOfMonth.doc = {
|
|
name: 'endOfMonth',
|
|
returnType: 'Date',
|
|
description: 'Transforms a date to the last possible moment that lies within the month',
|
|
};
|
|
|
|
isDst.doc = {
|
|
name: 'isDst',
|
|
returnType: 'boolean',
|
|
description: 'Checks if a Date is within Daylight Savings Time',
|
|
};
|
|
|
|
isWeekend.doc = {
|
|
name: 'isWeekend',
|
|
returnType: 'boolean',
|
|
description: 'Checks if the Date falls on a Saturday or Sunday',
|
|
};
|
|
|
|
// @TODO_NEXT_PHASE: Surface extensions below which take args
|
|
|
|
beginningOf.doc = {
|
|
name: 'beginningOf',
|
|
returnType: 'Date',
|
|
};
|
|
|
|
extract.doc = {
|
|
name: 'extract',
|
|
returnType: 'number',
|
|
};
|
|
|
|
format.doc = {
|
|
name: 'format',
|
|
returnType: '(?)',
|
|
};
|
|
|
|
isBetween.doc = {
|
|
name: 'isBetween',
|
|
returnType: 'boolean',
|
|
};
|
|
|
|
isInLast.doc = {
|
|
name: 'isInLast',
|
|
returnType: 'boolean',
|
|
};
|
|
|
|
minus.doc = {
|
|
name: 'minus',
|
|
returnType: 'Date',
|
|
};
|
|
|
|
plus.doc = {
|
|
name: 'plus',
|
|
returnType: 'Date',
|
|
};
|
|
|
|
export const dateExtensions: ExtensionMap = {
|
|
typeName: 'Date',
|
|
functions: {
|
|
beginningOf,
|
|
endOfMonth,
|
|
extract,
|
|
isBetween,
|
|
isDst,
|
|
isInLast,
|
|
isWeekend,
|
|
minus,
|
|
plus,
|
|
format,
|
|
},
|
|
};
|