UNPKG

validata

Version:

Type safe data validation and sanitization

64 lines (53 loc) 2.06 kB
import { DateTime, Duration } from 'luxon'; import { basicValidation, Check, Coerce, CommonValidationOptions, Convert, createAsCheck, createIsCheck, createMaybeAsCheck, createMaybeCheck, Validate } from './common'; import { Issue } from './types'; interface CoerceOptions { } interface ValidationOptions extends CommonValidationOptions<DateTime> { maxFuture?: Duration; maxPast?: Duration; } const check: Check<DateTime> = (value): value is DateTime => { return value instanceof DateTime; }; const convert: Convert<DateTime> = (value) => { if (value instanceof Date) { const utc = DateTime.fromJSDate(value, { zone: 'utc' }); if (!utc.isValid) return undefined; return utc; } if (typeof value === 'number' && !Number.isNaN(value)) { const utc = DateTime.fromMillis(value, { zone: 'utc' }); if (!utc.isValid) return undefined; return utc; } if (typeof value === 'string' && value) { const utc = DateTime.fromISO(value, { zone: 'utc' }); if (!utc.isValid) return undefined; return utc; } return undefined; }; const coerce: Coerce<DateTime, CoerceOptions> = () => (next) => (value, path) => { return next(value, path); }; const validate: Validate<DateTime, ValidationOptions> = (value, path, options) => { const result = basicValidation(value, path, options); if (options.maxFuture) { const max = DateTime.utc().plus(options.maxFuture); if (value > max) { result.issues.push(Issue.forPath(path, value, 'max-future', { max })); } } if (options.maxPast) { const min = DateTime.utc().minus(options.maxPast); if (value < min) { result.issues.push(Issue.forPath(path, value, 'max-past', { min })); } } return result; }; export const isDateTime = createIsCheck('date', check, coerce, validate); export const maybeDateTime = createMaybeCheck('date', check, coerce, validate); export const asDateTime = createAsCheck('date', check, convert, coerce, validate); export const maybeAsDateTime = createMaybeAsCheck('date', check, convert, coerce, validate);