Improve the TypeScript types (#80)
This commit is contained in:
parent
2502442404
commit
3c847be5a0
3 changed files with 70 additions and 52 deletions
113
source/index.ts
113
source/index.ts
|
|
@ -7,18 +7,8 @@
|
||||||
// tslint:disable-next-line
|
// tslint:disable-next-line
|
||||||
const URLGlobal = typeof URL === 'undefined' ? require('url').URL : URL;
|
const URLGlobal = typeof URL === 'undefined' ? require('url').URL : URL;
|
||||||
|
|
||||||
type TypedArray = Int8Array | Uint8Array | Uint8ClampedArray | Int16Array | Uint16Array | Int32Array | Uint32Array | Float32Array | Float64Array;
|
|
||||||
type Primitive = null | undefined | string | number | boolean | Symbol;
|
|
||||||
|
|
||||||
export interface ArrayLike {
|
|
||||||
length: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export type Class<T = unknown> = new(...args: any[]) => T;
|
export type Class<T = unknown> = new(...args: any[]) => T;
|
||||||
|
|
||||||
type DomElement = object & { nodeType: 1; nodeName: string };
|
|
||||||
type NodeStream = object & { pipe: Function };
|
|
||||||
|
|
||||||
export const enum TypeName {
|
export const enum TypeName {
|
||||||
null = 'null',
|
null = 'null',
|
||||||
boolean = 'boolean',
|
boolean = 'boolean',
|
||||||
|
|
@ -58,11 +48,9 @@ export const enum TypeName {
|
||||||
|
|
||||||
const toString = Object.prototype.toString;
|
const toString = Object.prototype.toString;
|
||||||
const isOfType = <T>(type: string) => (value: unknown): value is T => typeof value === type;
|
const isOfType = <T>(type: string) => (value: unknown): value is T => typeof value === type;
|
||||||
const isBuffer = (input: unknown): input is Buffer => !is.nullOrUndefined(input) && !is.nullOrUndefined((input as Buffer).constructor) && is.function_((input as Buffer).constructor.isBuffer) && (input as Buffer).constructor.isBuffer(input);
|
|
||||||
|
|
||||||
const getObjectType = (value: unknown): TypeName | null => {
|
const getObjectType = (value: unknown): TypeName | null => {
|
||||||
const objectName = toString.call(value).slice(8, -1);
|
const objectName = toString.call(value).slice(8, -1);
|
||||||
|
|
||||||
if (objectName) {
|
if (objectName) {
|
||||||
return objectName as TypeName;
|
return objectName as TypeName;
|
||||||
}
|
}
|
||||||
|
|
@ -72,7 +60,8 @@ const getObjectType = (value: unknown): TypeName | null => {
|
||||||
|
|
||||||
const isObjectOfType = <T>(type: TypeName) => (value: unknown): value is T => getObjectType(value) === type;
|
const isObjectOfType = <T>(type: TypeName) => (value: unknown): value is T => getObjectType(value) === type;
|
||||||
|
|
||||||
function is(value: unknown): TypeName { // tslint:disable-line:only-arrow-functions
|
// tslint:disable-next-line: no-use-before-declare
|
||||||
|
function is(value: unknown): TypeName {
|
||||||
switch (value) {
|
switch (value) {
|
||||||
case null:
|
case null:
|
||||||
return TypeName.null;
|
return TypeName.null;
|
||||||
|
|
@ -102,11 +91,11 @@ function is(value: unknown): TypeName { // tslint:disable-line:only-arrow-functi
|
||||||
return TypeName.Observable;
|
return TypeName.Observable;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Array.isArray(value)) {
|
if (is.array(value)) {
|
||||||
return TypeName.Array;
|
return TypeName.Array;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isBuffer(value)) {
|
if (is.buffer(value)) {
|
||||||
return TypeName.Buffer;
|
return TypeName.Buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -122,25 +111,24 @@ function is(value: unknown): TypeName { // tslint:disable-line:only-arrow-functi
|
||||||
return TypeName.Object;
|
return TypeName.Object;
|
||||||
}
|
}
|
||||||
|
|
||||||
// tslint:disable-next-line:strict-type-predicates
|
// tslint:disable-next-line: strict-type-predicates
|
||||||
const isObject = (value: unknown): value is object => typeof value === 'object';
|
const isObject = (value: unknown): value is object => typeof value === 'object';
|
||||||
|
|
||||||
is.undefined = isOfType<undefined>('undefined');
|
is.undefined = isOfType<undefined>('undefined');
|
||||||
is.string = isOfType<string>('string');
|
is.string = isOfType<string>('string');
|
||||||
is.number = isOfType<number>('number');
|
is.number = isOfType<number>('number');
|
||||||
is.function_ = isOfType<Function>('function');
|
is.function_ = isOfType<Function>('function');
|
||||||
// tslint:disable-next-line:strict-type-predicates
|
// tslint:disable-next-line: strict-type-predicates
|
||||||
is.null_ = (value: unknown): value is null => value === null;
|
is.null_ = (value: unknown): value is null => value === null;
|
||||||
is.class_ = (value: unknown): value is Class => is.function_(value) && value.toString().startsWith('class ');
|
is.class_ = (value: unknown): value is Class => is.function_(value) && value.toString().startsWith('class ');
|
||||||
is.boolean = (value: unknown): value is boolean => value === true || value === false;
|
is.boolean = (value: unknown): value is boolean => value === true || value === false;
|
||||||
is.symbol = isOfType<Symbol>('symbol');
|
is.symbol = isOfType<symbol>('symbol');
|
||||||
// tslint:enable:variable-name
|
|
||||||
|
|
||||||
is.numericString = (value: unknown): boolean =>
|
is.numericString = (value: unknown): value is string =>
|
||||||
is.string(value) && value.length > 0 && !Number.isNaN(Number(value));
|
is.string(value) && value.length > 0 && !Number.isNaN(Number(value));
|
||||||
|
|
||||||
is.array = Array.isArray;
|
is.array = Array.isArray;
|
||||||
is.buffer = isBuffer;
|
is.buffer = (value: unknown): value is Buffer => !is.nullOrUndefined(value) && !is.nullOrUndefined((value as Buffer).constructor) && is.function_((value as Buffer).constructor.isBuffer) && (value as Buffer).constructor.isBuffer(value);
|
||||||
|
|
||||||
is.nullOrUndefined = (value: unknown): value is null | undefined => is.null_(value) || is.undefined(value);
|
is.nullOrUndefined = (value: unknown): value is null | undefined => is.null_(value) || is.undefined(value);
|
||||||
is.object = (value: unknown): value is object => !is.nullOrUndefined(value) && (is.function_(value) || isObject(value));
|
is.object = (value: unknown): value is object => !is.nullOrUndefined(value) && (is.function_(value) || isObject(value));
|
||||||
|
|
@ -188,7 +176,7 @@ is.dataView = isObjectOfType<DataView>(TypeName.DataView);
|
||||||
is.directInstanceOf = <T>(instance: unknown, klass: Class<T>): instance is T => Object.getPrototypeOf(instance) === klass.prototype;
|
is.directInstanceOf = <T>(instance: unknown, klass: Class<T>): instance is T => Object.getPrototypeOf(instance) === klass.prototype;
|
||||||
is.urlInstance = (value: unknown): value is URL => isObjectOfType<URL>(TypeName.URL)(value);
|
is.urlInstance = (value: unknown): value is URL => isObjectOfType<URL>(TypeName.URL)(value);
|
||||||
|
|
||||||
is.urlString = (value: unknown) => {
|
is.urlString = (value: unknown): value is string => {
|
||||||
if (!is.string(value)) {
|
if (!is.string(value)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
@ -201,12 +189,15 @@ is.urlString = (value: unknown) => {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// TODO: Use the `not` operator with a type guard here when it's available.
|
||||||
|
// Example: `is.truthy = (value: unknown): value is (not false | not 0 | not '' | not undefined | not null) => Boolean(value);`
|
||||||
is.truthy = (value: unknown) => Boolean(value);
|
is.truthy = (value: unknown) => Boolean(value);
|
||||||
|
// Example: `is.falsy = (value: unknown): value is (not true | 0 | '' | undefined | null) => Boolean(value);`
|
||||||
is.falsy = (value: unknown) => !value;
|
is.falsy = (value: unknown) => !value;
|
||||||
|
|
||||||
is.nan = (value: unknown) => Number.isNaN(value as number);
|
is.nan = (value: unknown) => Number.isNaN(value as number);
|
||||||
|
|
||||||
const primitiveTypes = new Set([
|
const primitiveTypeOfTypes = new Set([
|
||||||
'undefined',
|
'undefined',
|
||||||
'string',
|
'string',
|
||||||
'number',
|
'number',
|
||||||
|
|
@ -214,12 +205,15 @@ const primitiveTypes = new Set([
|
||||||
'symbol'
|
'symbol'
|
||||||
]);
|
]);
|
||||||
|
|
||||||
is.primitive = (value: unknown): value is Primitive => is.null_(value) || primitiveTypes.has(typeof value);
|
// TODO: This should be able to be `not object` when the `not` operator is out
|
||||||
|
export type Primitive = null | undefined | string | number | boolean | symbol;
|
||||||
|
|
||||||
|
is.primitive = (value: unknown): value is Primitive => is.null_(value) || primitiveTypeOfTypes.has(typeof value);
|
||||||
|
|
||||||
is.integer = (value: unknown): value is number => Number.isInteger(value as number);
|
is.integer = (value: unknown): value is number => Number.isInteger(value as number);
|
||||||
is.safeInteger = (value: unknown): value is number => Number.isSafeInteger(value as number);
|
is.safeInteger = (value: unknown): value is number => Number.isSafeInteger(value as number);
|
||||||
|
|
||||||
is.plainObject = (value: unknown) => {
|
is.plainObject = (value: unknown): value is {[key: string]: unknown} => {
|
||||||
// From: https://github.com/sindresorhus/is-plain-obj/blob/master/index.js
|
// From: https://github.com/sindresorhus/is-plain-obj/blob/master/index.js
|
||||||
let prototype;
|
let prototype;
|
||||||
|
|
||||||
|
|
@ -240,9 +234,10 @@ const typedArrayTypes = new Set([
|
||||||
TypeName.Float64Array
|
TypeName.Float64Array
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
export type TypedArray = Int8Array | Uint8Array | Uint8ClampedArray | Int16Array | Uint16Array | Int32Array | Uint32Array | Float32Array | Float64Array;
|
||||||
|
|
||||||
is.typedArray = (value: unknown): value is TypedArray => {
|
is.typedArray = (value: unknown): value is TypedArray => {
|
||||||
const objectType = getObjectType(value);
|
const objectType = getObjectType(value);
|
||||||
|
|
||||||
if (objectType === null) {
|
if (objectType === null) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
@ -250,10 +245,15 @@ is.typedArray = (value: unknown): value is TypedArray => {
|
||||||
return typedArrayTypes.has(objectType);
|
return typedArrayTypes.has(objectType);
|
||||||
};
|
};
|
||||||
|
|
||||||
const isValidLength = (value: unknown) => is.safeInteger(value) && value > -1;
|
export interface ArrayLike<T> {
|
||||||
is.arrayLike = (value: unknown): value is ArrayLike => !is.nullOrUndefined(value) && !is.function_(value) && isValidLength((value as ArrayLike).length);
|
readonly length: number;
|
||||||
|
readonly [index: number]: T;
|
||||||
|
}
|
||||||
|
|
||||||
is.inRange = (value: number, range: number | number[]) => {
|
const isValidLength = (value: unknown) => is.safeInteger(value) && value >= 0;
|
||||||
|
is.arrayLike = (value: unknown): value is ArrayLike<unknown> => !is.nullOrUndefined(value) && !is.function_(value) && isValidLength((value as ArrayLike<unknown>).length);
|
||||||
|
|
||||||
|
is.inRange = (value: number, range: number | number[]): value is number => {
|
||||||
if (is.number(range)) {
|
if (is.number(range)) {
|
||||||
return value >= Math.min(0, range) && value <= Math.max(range, 0);
|
return value >= Math.min(0, range) && value <= Math.max(range, 0);
|
||||||
}
|
}
|
||||||
|
|
@ -274,10 +274,15 @@ const DOM_PROPERTIES_TO_CHECK = [
|
||||||
'nodeValue'
|
'nodeValue'
|
||||||
];
|
];
|
||||||
|
|
||||||
is.domElement = (value: unknown): value is DomElement => is.object(value) && (value as DomElement).nodeType === NODE_TYPE_ELEMENT && is.string((value as DomElement).nodeName) &&
|
is.domElement = (value: unknown): value is Element => is.object(value) && (value as Element).nodeType === NODE_TYPE_ELEMENT && is.string((value as Element).nodeName) &&
|
||||||
!is.plainObject(value) && DOM_PROPERTIES_TO_CHECK.every(property => property in (value as DomElement));
|
!is.plainObject(value) && DOM_PROPERTIES_TO_CHECK.every(property => property in (value as Element));
|
||||||
|
|
||||||
is.observable = (value: unknown) => {
|
export interface ObservableLike {
|
||||||
|
subscribe(observer: (value: unknown) => void): void;
|
||||||
|
[Symbol.observable](): ObservableLike;
|
||||||
|
}
|
||||||
|
|
||||||
|
is.observable = (value: unknown): value is ObservableLike => {
|
||||||
if (!value) {
|
if (!value) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
@ -293,34 +298,44 @@ is.observable = (value: unknown) => {
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type NodeStream = object & {readonly pipe: Function};
|
||||||
|
|
||||||
is.nodeStream = (value: unknown): value is NodeStream => !is.nullOrUndefined(value) && isObject(value) as unknown && is.function_((value as NodeStream).pipe) && !is.observable(value);
|
is.nodeStream = (value: unknown): value is NodeStream => !is.nullOrUndefined(value) && isObject(value) as unknown && is.function_((value as NodeStream).pipe) && !is.observable(value);
|
||||||
|
|
||||||
is.infinite = (value: unknown) => value === Infinity || value === -Infinity;
|
is.infinite = (value: unknown): value is number => value === Infinity || value === -Infinity;
|
||||||
|
|
||||||
const isAbsoluteMod2 = (rem: number) => (value: number) => is.integer(value) && Math.abs(value % 2) === rem;
|
const isAbsoluteMod2 = (rem: number) => (value: number): value is number => is.integer(value) && Math.abs(value % 2) === rem;
|
||||||
is.evenInteger = isAbsoluteMod2(0);
|
is.evenInteger = isAbsoluteMod2(0);
|
||||||
is.oddInteger = isAbsoluteMod2(1);
|
is.oddInteger = isAbsoluteMod2(1);
|
||||||
|
|
||||||
|
is.emptyArray = (value: unknown): value is never[] => is.array(value) && value.length === 0;
|
||||||
|
is.nonEmptyArray = (value: unknown): value is unknown[] => is.array(value) && value.length > 0;
|
||||||
|
|
||||||
|
is.emptyString = (value: unknown): value is '' => is.string(value) && value.length === 0;
|
||||||
|
|
||||||
|
// TODO: Use `not ''` when the `not` operator is available.
|
||||||
|
is.nonEmptyString = (value: unknown): value is string => is.string(value) && value.length > 0;
|
||||||
|
|
||||||
const isWhiteSpaceString = (value: unknown) => is.string(value) && /\S/.test(value) === false;
|
const isWhiteSpaceString = (value: unknown) => is.string(value) && /\S/.test(value) === false;
|
||||||
|
is.emptyStringOrWhitespace = (value: unknown): value is string => is.emptyString(value) || isWhiteSpaceString(value);
|
||||||
|
|
||||||
is.emptyArray = (value: unknown) => is.array(value) && value.length === 0;
|
is.emptyObject = (value: unknown): value is {[key: string]: never} => is.object(value) && !is.map(value) && !is.set(value) && Object.keys(value).length === 0;
|
||||||
is.nonEmptyArray = (value: unknown) => is.array(value) && value.length > 0;
|
|
||||||
|
|
||||||
is.emptyString = (value: unknown) => is.string(value) && value.length === 0;
|
// TODO: Use `not` operator here to remove `Map` and `Set` from type guard:
|
||||||
is.nonEmptyString = (value: unknown) => is.string(value) && value.length > 0;
|
// - https://github.com/Microsoft/TypeScript/pull/29317
|
||||||
is.emptyStringOrWhitespace = (value: unknown) => is.emptyString(value) || isWhiteSpaceString(value);
|
is.nonEmptyObject = (value: unknown): value is {[key: string]: unknown} => is.object(value) && !is.map(value) && !is.set(value) && Object.keys(value).length > 0;
|
||||||
|
|
||||||
is.emptyObject = (value: unknown) => is.object(value) && !is.map(value) && !is.set(value) && Object.keys(value).length === 0;
|
is.emptySet = (value: unknown): value is Set<never> => is.set(value) && value.size === 0;
|
||||||
is.nonEmptyObject = (value: unknown) => is.object(value) && !is.map(value) && !is.set(value) && Object.keys(value).length > 0;
|
is.nonEmptySet = (value: unknown): value is Set<unknown> => is.set(value) && value.size > 0;
|
||||||
|
|
||||||
is.emptySet = (value: unknown) => is.set(value) && value.size === 0;
|
is.emptyMap = (value: unknown): value is Map<never, never> => is.map(value) && value.size === 0;
|
||||||
is.nonEmptySet = (value: unknown) => is.set(value) && value.size > 0;
|
is.nonEmptyMap = (value: unknown): value is Map<unknown, unknown> => is.map(value) && value.size > 0;
|
||||||
|
|
||||||
is.emptyMap = (value: unknown) => is.map(value) && value.size === 0;
|
export type Predicate = (value: unknown) => boolean;
|
||||||
is.nonEmptyMap = (value: unknown) => is.map(value) && value.size > 0;
|
|
||||||
|
|
||||||
type ArrayMethod = (fn: (value: unknown, index: number, array: unknown[]) => boolean, thisArg?: unknown) => boolean;
|
type ArrayMethod = (fn: (value: unknown, index: number, array: unknown[]) => boolean, thisArg?: unknown) => boolean;
|
||||||
const predicateOnArray = (method: ArrayMethod, predicate: unknown, values: unknown[]) => {
|
|
||||||
|
const predicateOnArray = (method: ArrayMethod, predicate: Predicate, values: unknown[]) => {
|
||||||
if (is.function_(predicate) === false) {
|
if (is.function_(predicate) === false) {
|
||||||
throw new TypeError(`Invalid predicate: ${JSON.stringify(predicate)}`);
|
throw new TypeError(`Invalid predicate: ${JSON.stringify(predicate)}`);
|
||||||
}
|
}
|
||||||
|
|
@ -329,12 +344,12 @@ const predicateOnArray = (method: ArrayMethod, predicate: unknown, values: unkno
|
||||||
throw new TypeError('Invalid number of values');
|
throw new TypeError('Invalid number of values');
|
||||||
}
|
}
|
||||||
|
|
||||||
return method.call(values, predicate as any);
|
return method.call(values, predicate);
|
||||||
};
|
};
|
||||||
|
|
||||||
// tslint:disable variable-name
|
// tslint:disable variable-name
|
||||||
is.any = (predicate: unknown, ...values: unknown[]) => predicateOnArray(Array.prototype.some, predicate, values);
|
is.any = (predicate: Predicate, ...values: unknown[]): boolean => predicateOnArray(Array.prototype.some, predicate, values);
|
||||||
is.all = (predicate: unknown, ...values: unknown[]) => predicateOnArray(Array.prototype.every, predicate, values);
|
is.all = (predicate: Predicate, ...values: unknown[]): boolean => predicateOnArray(Array.prototype.every, predicate, values);
|
||||||
// tslint:enable variable-name
|
// tslint:enable variable-name
|
||||||
|
|
||||||
// Some few keywords are reserved, but we'll populate them for Node.js users
|
// Some few keywords are reserved, but we'll populate them for Node.js users
|
||||||
|
|
|
||||||
|
|
@ -867,7 +867,7 @@ test('is.any', t => {
|
||||||
t.false(is.any(is.integer, true, 'lol', {}));
|
t.false(is.any(is.integer, true, 'lol', {}));
|
||||||
|
|
||||||
t.throws(() => {
|
t.throws(() => {
|
||||||
is.any(null, true);
|
is.any(null as any, true);
|
||||||
});
|
});
|
||||||
|
|
||||||
t.throws(() => {
|
t.throws(() => {
|
||||||
|
|
@ -882,7 +882,7 @@ test('is.all', t => {
|
||||||
t.false(is.all(is.set, new Map(), {}));
|
t.false(is.all(is.set, new Map(), {}));
|
||||||
|
|
||||||
t.throws(() => {
|
t.throws(() => {
|
||||||
is.all(null, true);
|
is.all(null as any, true);
|
||||||
});
|
});
|
||||||
|
|
||||||
t.throws(() => {
|
t.throws(() => {
|
||||||
|
|
|
||||||
|
|
@ -1,3 +1,6 @@
|
||||||
{
|
{
|
||||||
"extends": "tslint-xo"
|
"extends": "tslint-xo",
|
||||||
|
"rules": {
|
||||||
|
"interface-over-type-literal": false
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue