Improve is.integer & is.safeInteger TypeScript type

This commit is contained in:
Younho Choo 2022-01-17 20:16:58 +09:00
parent de50144b95
commit d2e65aa5b8
2 changed files with 34 additions and 13 deletions

View file

@ -2,7 +2,7 @@
/// <reference lib="dom"/>
/// <reference types="node"/>
import {Class, TypedArray, ObservableLike, Primitive} from './types';
import {Class, TypedArray, ObservableLike, Primitive, Integer} from './types';
const typedArrayTypeNames = [
'Int8Array',
@ -258,8 +258,8 @@ is.nan = (value: unknown) => Number.isNaN(value as number);
is.primitive = (value: unknown): value is Primitive => is.null_(value) || isPrimitiveTypeName(typeof value);
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.integer = <T extends number>(value: T): value is Integer<T> => Number.isInteger(value);
is.safeInteger = <T extends number>(value: T): value is Integer<T> => Number.isSafeInteger(value);
is.plainObject = <Value = unknown>(value: unknown): value is Record<PropertyKey, Value> => {
// From: https://github.com/sindresorhus/is-plain-obj/blob/main/index.js
@ -279,7 +279,7 @@ export interface ArrayLike<T> {
readonly length: number;
}
const isValidLength = (value: unknown): value is number => is.safeInteger(value) && value >= 0;
const isValidLength = <T extends number>(value: T): value is Integer<T> => is.safeInteger(value) && value >= 0;
is.arrayLike = <T = unknown>(value: unknown): value is ArrayLike<T> => !is.nullOrUndefined(value) && !is.function_(value) && isValidLength((value as ArrayLike<T>).length);
is.inRange = (value: number, range: number | number[]): value is number => {
@ -336,7 +336,7 @@ is.nodeStream = (value: unknown): value is NodeStream => is.object(value) && is.
is.infinite = (value: unknown): value is number => value === Infinity || value === -Infinity;
const isAbsoluteMod2 = (remainder: number) => (value: number): value is number => is.integer(value) && Math.abs(value % 2) === remainder;
const isAbsoluteMod2 = (remainder: number) => <T extends number>(value: T): value is Integer<T> => is.integer(value) && Math.abs(value % 2) === remainder;
is.evenInteger = isAbsoluteMod2(0);
is.oddInteger = isAbsoluteMod2(1);
@ -509,8 +509,8 @@ interface Assert {
falsy: (value: unknown) => asserts value is unknown;
nan: (value: unknown) => asserts value is unknown;
primitive: (value: unknown) => asserts value is Primitive;
integer: (value: unknown) => asserts value is number;
safeInteger: (value: unknown) => asserts value is number;
integer: <T extends number>(value: T) => asserts value is Integer<T>;
safeInteger: <T extends number>(value: T) => asserts value is Integer<T>;
plainObject: <Value = unknown>(value: unknown) => asserts value is Record<PropertyKey, Value>;
typedArray: (value: unknown) => asserts value is TypedArray;
arrayLike: <T = unknown>(value: unknown) => asserts value is ArrayLike<T>;
@ -534,8 +534,8 @@ interface Assert {
urlSearchParams: (value: unknown) => asserts value is URLSearchParams;
// Numbers.
evenInteger: (value: number) => asserts value is number;
oddInteger: (value: number) => asserts value is number;
evenInteger: <T extends number>(value: T) => asserts value is Integer<T>;
oddInteger: <T extends number>(value: T) => asserts value is Integer<T>;
// Two arguments.
directInstanceOf: <T>(instance: unknown, class_: Class<T>) => asserts instance is T;
@ -610,8 +610,8 @@ export const assert: Assert = {
falsy: (value: unknown): asserts value is unknown => assertType(is.falsy(value), AssertionTypeDescription.falsy, value),
nan: (value: unknown): asserts value is unknown => assertType(is.nan(value), AssertionTypeDescription.nan, value),
primitive: (value: unknown): asserts value is Primitive => assertType(is.primitive(value), AssertionTypeDescription.primitive, value),
integer: (value: unknown): asserts value is number => assertType(is.integer(value), AssertionTypeDescription.integer, value),
safeInteger: (value: unknown): asserts value is number => assertType(is.safeInteger(value), AssertionTypeDescription.safeInteger, value),
integer: <T extends number>(value: T): asserts value is Integer<T> => assertType(is.integer(value), AssertionTypeDescription.integer, value),
safeInteger: <T extends number>(value: T): asserts value is Integer<T> => assertType(is.safeInteger(value), AssertionTypeDescription.safeInteger, value),
plainObject: <Value = unknown>(value: unknown): asserts value is Record<PropertyKey, Value> => assertType(is.plainObject(value), AssertionTypeDescription.plainObject, value),
typedArray: (value: unknown): asserts value is TypedArray => assertType(is.typedArray(value), AssertionTypeDescription.typedArray, value),
arrayLike: <T = unknown>(value: unknown): asserts value is ArrayLike<T> => assertType(is.arrayLike(value), AssertionTypeDescription.arrayLike, value),
@ -635,8 +635,8 @@ export const assert: Assert = {
urlSearchParams: (value: unknown): asserts value is URLSearchParams => assertType(is.urlSearchParams(value), 'URLSearchParams', value),
// Numbers.
evenInteger: (value: number): asserts value is number => assertType(is.evenInteger(value), AssertionTypeDescription.evenInteger, value),
oddInteger: (value: number): asserts value is number => assertType(is.oddInteger(value), AssertionTypeDescription.oddInteger, value),
evenInteger: <T extends number>(value: T): asserts value is Integer<T> => assertType(is.evenInteger(value), AssertionTypeDescription.evenInteger, value),
oddInteger: <T extends number>(value: T): asserts value is Integer<T> => assertType(is.oddInteger(value), AssertionTypeDescription.oddInteger, value),
// Two arguments.
directInstanceOf: <T>(instance: unknown, class_: Class<T>): asserts instance is T => assertType(is.directInstanceOf(instance, class_), AssertionTypeDescription.directInstanceOf, instance),

View file

@ -47,3 +47,24 @@ export interface ObservableLike {
subscribe(observer: (value: unknown) => void): void;
[Symbol.observable](): ObservableLike;
}
/**
A `number` that is an integer.
You can't pass a `bigint` as they are already guaranteed to be integers.
Use-case: Validating and documenting parameters.
@example
```
import {Integer} from 'type-fest';
declare function setYear<T extends number>(length: Integer<T>): void;
```
@see NegativeInteger
@see NonNegativeInteger
@category Numeric
*/
// `${bigint}` is a type that matches a valid bigint literal without the `n` (ex. 1, 0b1, 0o1, 0x1)
// Because T is a number and not a string we can effectively use this to filter out any numbers containing decimal points
export type Integer<Type extends number> = `${Type}` extends `${bigint}` ? Type : never;