pdf-lib
Version:
Create and modify PDF files with JavaScript
229 lines (202 loc) • 7.06 kB
text/typescript
/* tslint:disable:ban-types */
import { values as objectValues } from 'src/utils/objects';
export const backtick = (val: any) => `\`${val}\``;
export const singleQuote = (val: any) => `'${val}'`;
type Primitive = string | number | boolean | undefined | null;
// prettier-ignore
const formatValue = (value: any) => {
const type = typeof value;
if (type ==='string') return singleQuote(value);
else if (type ==='undefined') return backtick(value);
else return value;
};
export const createValueErrorMsg = (
value: any,
valueName: string,
values: Primitive[],
) => {
const allowedValues = new Array(values.length);
for (let idx = 0, len = values.length; idx < len; idx++) {
const v = values[idx];
allowedValues[idx] = formatValue(v);
}
const joinedValues = allowedValues.join(' or ');
// prettier-ignore
return `${backtick(valueName)} must be one of ${joinedValues}, but was actually ${formatValue(value)}`;
};
export const assertIsOneOf = (
value: any,
valueName: string,
allowedValues: Primitive[] | { [key: string]: Primitive },
) => {
if (!Array.isArray(allowedValues)) {
allowedValues = objectValues(allowedValues);
}
for (let idx = 0, len = allowedValues.length; idx < len; idx++) {
if (value === allowedValues[idx]) return;
}
throw new TypeError(createValueErrorMsg(value, valueName, allowedValues));
};
export const assertIsOneOfOrUndefined = (
value: any,
valueName: string,
allowedValues: Primitive[] | { [key: string]: Primitive },
) => {
if (!Array.isArray(allowedValues)) {
allowedValues = objectValues(allowedValues);
}
assertIsOneOf(value, valueName, allowedValues.concat(undefined));
};
export const assertIsSubset = (
values: any[],
valueName: string,
allowedValues: Primitive[] | { [key: string]: Primitive },
) => {
if (!Array.isArray(allowedValues)) {
allowedValues = objectValues(allowedValues);
}
for (let idx = 0, len = values.length; idx < len; idx++) {
assertIsOneOf(values[idx], valueName, allowedValues);
}
};
export const getType = (val: any) => {
if (val === null) return 'null';
if (val === undefined) return 'undefined';
if (typeof val === 'string') return 'string';
if (isNaN(val)) return 'NaN';
if (typeof val === 'number') return 'number';
if (typeof val === 'boolean') return 'boolean';
if (typeof val === 'symbol') return 'symbol';
if (typeof val === 'bigint') return 'bigint';
if (val.constructor && val.constructor.name) return val.constructor.name;
if (val.name) return val.name;
if (val.constructor) return String(val.constructor);
return String(val);
};
export type TypeDescriptor =
| 'null'
| 'undefined'
| 'string'
| 'number'
| 'boolean'
| 'symbol'
| 'bigint'
| DateConstructor
| ArrayConstructor
| Uint8ArrayConstructor
| ArrayBufferConstructor
| FunctionConstructor
| [Function, string];
export const isType = (value: any, type: TypeDescriptor) => {
if (type === 'null') return value === null;
if (type === 'undefined') return value === undefined;
if (type === 'string') return typeof value === 'string';
if (type === 'number') return typeof value === 'number' && !isNaN(value);
if (type === 'boolean') return typeof value === 'boolean';
if (type === 'symbol') return typeof value === 'symbol';
if (type === 'bigint') return typeof value === 'bigint';
if (type === Date) return value instanceof Date;
if (type === Array) return value instanceof Array;
if (type === Uint8Array) return value instanceof Uint8Array;
if (type === ArrayBuffer) return value instanceof ArrayBuffer;
if (type === Function) return value instanceof Function;
return value instanceof (type as [Function, string])[0];
};
export const createTypeErrorMsg = (
value: any,
valueName: string,
types: TypeDescriptor[],
) => {
const allowedTypes = new Array(types.length);
for (let idx = 0, len = types.length; idx < len; idx++) {
const type = types[idx];
if (type === 'null') allowedTypes[idx] = backtick('null');
if (type === 'undefined') allowedTypes[idx] = backtick('undefined');
if (type === 'string') allowedTypes[idx] = backtick('string');
else if (type === 'number') allowedTypes[idx] = backtick('number');
else if (type === 'boolean') allowedTypes[idx] = backtick('boolean');
else if (type === 'symbol') allowedTypes[idx] = backtick('symbol');
else if (type === 'bigint') allowedTypes[idx] = backtick('bigint');
else if (type === Array) allowedTypes[idx] = backtick('Array');
else if (type === Uint8Array) allowedTypes[idx] = backtick('Uint8Array');
else if (type === ArrayBuffer) allowedTypes[idx] = backtick('ArrayBuffer');
else allowedTypes[idx] = backtick((type as [Function, string])[1]);
}
const joinedTypes = allowedTypes.join(' or ');
// prettier-ignore
return `${backtick(valueName)} must be of type ${joinedTypes}, but was actually of type ${backtick(getType(value))}`;
};
export const assertIs = (
value: any,
valueName: string,
types: TypeDescriptor[],
) => {
for (let idx = 0, len = types.length; idx < len; idx++) {
if (isType(value, types[idx])) return;
}
throw new TypeError(createTypeErrorMsg(value, valueName, types));
};
export const assertOrUndefined = (
value: any,
valueName: string,
types: TypeDescriptor[],
) => {
assertIs(value, valueName, types.concat('undefined'));
};
export const assertEachIs = (
values: any[],
valueName: string,
types: TypeDescriptor[],
) => {
for (let idx = 0, len = values.length; idx < len; idx++) {
assertIs(values[idx], valueName, types);
}
};
export const assertRange = (
value: any,
valueName: string,
min: number,
max: number,
) => {
assertIs(value, valueName, ['number']);
assertIs(min, 'min', ['number']);
assertIs(max, 'max', ['number']);
max = Math.max(min, max);
if (value < min || value > max) {
// prettier-ignore
throw new Error(`${backtick(valueName)} must be at least ${min} and at most ${max}, but was actually ${value}`);
}
};
export const assertRangeOrUndefined = (
value: any,
valueName: string,
min: number,
max: number,
) => {
assertIs(value, valueName, ['number', 'undefined']);
if (typeof value === 'number') assertRange(value, valueName, min, max);
};
export const assertMultiple = (
value: any,
valueName: string,
multiplier: number,
) => {
assertIs(value, valueName, ['number']);
if (value % multiplier !== 0) {
// prettier-ignore
throw new Error(`${backtick(valueName)} must be a multiple of ${multiplier}, but was actually ${value}`);
}
};
export const assertInteger = (value: any, valueName: string) => {
if (!Number.isInteger(value)) {
throw new Error(
`${backtick(valueName)} must be an integer, but was actually ${value}`,
);
}
};
export const assertPositive = (value: number, valueName: string) => {
if (![1, 0].includes(Math.sign(value))) {
// prettier-ignore
throw new Error(`${backtick(valueName)} must be a positive number or 0, but was actually ${value}`);
}
};