ts-runtime-typecheck
A collection of TypeScript functions for converting unknown values into strictly typed values.
Contents
Installation
npm install ts-runtime-typecheck
Type Casts
Type casts take an unknown object as an argument, and return a typed object as the result. These functions take the form as{TYPE}
, for example asNumber
. If the input object does not match the required type the function will throw. This does not perform any coercion on the value, passing a string of a number to asNumber will cause it to throw.
import { asNumber } from 'ts-runtime-typecheck';
function square (input: unknown): number {
const value: number = asNumber(input);
return value * value;
}
square(10)
square()
square('10')
These functions are meant to primarily validate questionable values that are expected to be in a well defined structure. Such as network responses, interfacing with untyped javascript or reading data back from a file. If you are looking to validate a type, without throwing an error then take a look at Type Checks.
Fallback values
The standard type cast functions take a second optional parameter, which is a fallback value. In the situation that the input is nullish ( undefined | null
) and the fallback parameter has been defined the function will return the fallback parameter instead of throwing. This is very helpful for validating the input of an optional value, and providing a default value.
import { asString } from 'ts-runtime-typecheck';
function printName (name: unknown) {
const value: string = asString(name, 'Dave');
console.log(`Hello ${value}, how are you today?`);
}
printName()
printName('James')
printName(42)
In the situation you want to preserve the optionally of a value, but still validate the type there exists an alternate function for each type cast. These take the form asOpt{TYPE}
. Unlike the standard methods they do not take an optional fallback value, but when a nullish value is passed in they will always emit undefined
. If the input is not nullish, then it behaves the same as the standard type casts. If the type condition is met then it emit the value, otherwise it will throw.
Special case: asDefined
Another common situation is that you have an optional value, with a well defined type, but it shouldn't be optional at that time. TypeScript will allow you to cast the value to a non-optional type using !
, but this is often discouraged in style guides as it can hide real errors. This is solved by the asDefined
function, which removes the optionality from a type union. As with the other type casts this can take a fallback value, and will throw if the condition is not met. However, the output type matches the input type with nullish subtracted.
import { asDefined } from 'ts-runtime-typecheck';
function setup (useComplexType: boolean = false, complexInst?: ComplexType) {
if (useComplexType) {
const inst: ComplexType = asDefined(complexInst);
inst.doComplexThing();
}
else {
doSimpleThing();
}
}
Recursive Array/Object Casts
Validating that a value is an array or object is easy enough, but how about the contents? asArrayRecursive
and asObjectRecursive
allow for deep type casting through a user specified element cast. For example, to cast to an array of strings:
import { asString, asArrayRecursive } from 'ts-runtime-typecheck';
function main (obj: unknown) {
const asStringArray = asArrayRecursive(asString);
const arr: string[] = asStringArray(obj);
}
Or an array of numeric dictionaries:
import { asNumber, asRecordRecursive, asArrayRecursive } from 'ts-runtime-typecheck';
function main (obj: unknown) {
const asNumericRecord = asRecordRecursive(asNumber);
const asArrayOfNumericRecords = asArrayRecursive(asNumericRecord);
const arr: string[] = asArrayOfNumericRecords([
{
a: 12,
b: 42
},
{
n: 90
}
]);
}
Type Checks
Type checks take an unknown object as an argument, and return a boolean indicating if the given value matches the required type. These functions take the form is{TYPE}
In the correct situation TypeScript is capable of refining the type of a value through the use of these functions and flow analysis.
import { isNumber } from 'ts-runtime-typecheck';
export function printSq (input: unknown) {
if (isNumber(input)) {
console.log(`${value} * ${value} = ${value * value}`);
}
else {
console.log('Invalid input', value);
}
}
Type Coerce
Type coercion functions take an unknown object as an argument, and convert it into a specific type. These functions take the format make{TYPE}
. Unlike the other functions this only works for small subset of types: number, string and boolean. They make a best effort to convert the type, but if the input is not suitable then they will throw. For instance passing a non-numeric string to makeNumber
will cause it to throw, as will passing a string that is not "true" | "false"
to makeBoolean
. While these functions will take any input value, this is to allow the input of values that have not been validated, actually the only valid input types for all 3 functions are number | string | boolean
. The intention here is to allow useful conversion, but prevent accidentally passing complex types.
There is an argument that makeString
could support using the toString
method of an object
, but the default toString
method returns the useless [object Object]
string. It is possible to detect if an object has implemented it's own toString
method, but is it correct to use it in this situation? That depends on the intention of the programmer. In the absence of a clear answer the line has been drawn at only accepting primitives.
import { makeNumber } from 'ts-runtime-typecheck';
makeNumber('80')
makeNumber(80)
makeNumber(true)
makeNumber(false)
makeNumber('hello')
makeNumber({
toString () { return 'hello' }
})
JSON Types
Dealing with validating JSON objects can often be frustrating, so to make it a little easier JSON specific types and checks are provided. Using the JSONValue
type in your code will ensure that TS statically analyses any literal values as serializable to JSON.
import type { JSONArray, JSONObject, JSONValue } from 'ts-runtime-typecheck';
const a: JSONArray = [12, 'hello'];
const b: JSONObject = {
num: 12,
str: 'hello'
};
const c: JSONValue = 12;
const d: JSONValue = new Error('hi');
For dynamic data isJSONValue
and asJSONValue
provide recursive type validation on a value.
Type checks and casts are provided for JSONArray
s and JSONObject
s, with the caveat that they only accept JSONValue
s. This is to avoid needing to recursively validate the object.
import { asJSONValue, isJSONObject, isJSONArray } from 'ts-runtime-typecheck';
import type { JSONValue } from 'ts-runtime-typecheck';
function main (a: unknown) {
const obj: JSONValue = asJSONValue(a);
if (isJSONArray(obj)) {
}
else if (isJSONObject(obj)) {
}
else {
}
}
One other caveat of JSONValue
is that it does not guarantee that the value is not cyclic. It is not possible to serialize cyclic object with JSON, but they are otherwise valid. Using isJSONValue
or asJSONValue
on a cyclic object will fail.
import { asJSONValue } from 'ts-runtime-typecheck';
import type { Dictionary } from 'ts-runtime-typecheck';
const almost_right: Dictionary = {};
almost_right.self = almost_right;
const obj = asJSONValue(almost_right);
Reference
Reference: Type Casts
-
asString
Cast unknown
to string
. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asNumber
Cast unknown
to number
. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asIndex
Cast unknown
to Index
. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asIndexable
Cast unknown
to Indexable
. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asBoolean
Cast unknown
to boolean
. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asArray
Cast unknown
to Array<unknown>
. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asRecord
Cast unknown
to Dictionary<unknown>
. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asFunction
Cast unknown
to UnknownFunction
. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asDefined
Cast Type | Nullish
to Type
, where Type
is a generic parameter. Does not accept a fallback value.
-
asJSONValue
Cast unknown
to JSONValue
. This function recursively validates the value, and hence will fail if given a cyclic value. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asJSONObject
Cast JSONValue
to JSONObject
. Unlike asJSONValue
this does not perform recursive validation, hence it only accepts a JSONValue
so that the sub-elements are of a known type. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asJSONArray
Cast JSONValue
to JSONArray
. Unlike asJSONValue
this does not perform recursive validation, hence it only accepts a JSONValue
so that the sub-elements are of a known type. Accepts an optional fallback value that is emitted if the value is nullish and fallback is defined.
-
asArrayRecursive
Takes a Type Cast function for Type
and returns a new Type Cast function for Array<Type>
where type is a generic parameter. The emitted Type Cast function accepts an optional fallback value that is emitted if the value is nullish and fallback is defined. Refer to Recursive Array/Object casts for examples.
-
asObjectRecursive
Takes a Type Cast function for Type
and returns a new Type Cast function for Dictionary<Type>
where type is a generic parameter. The emitted Type Cast function accepts an optional fallback value that is emitted if the value is nullish and fallback is defined. Refer to Recursive Array/Object casts for examples.
Reference: Optional Type Casts
Reference: Type Checks
-
isRecord
Takes an unknown
value and returns a boolean indicating if the value is of the type Dictionary<unknown>
.
-
isFunction
Takes an unknown
value and returns a boolean indicating if the value is of the type UnknownFunction
.
-
isBoolean
Takes an unknown
value and returns a boolean indicating if the value is of the type boolean
.
-
isString
Takes an unknown
value and returns a boolean indicating if the value is of the type string
.
-
isNumber
Takes an unknown
value and returns a boolean indicating if the value is of the type number
.
-
isIndex
Takes an unknown
value and returns a boolean indicating if the value is of the type Index
.
-
isArray
Takes an unknown
value and returns a boolean indicating if the value is of the type Array<unknown>
.
-
isUndefined
Takes an unknown
value and returns a boolean indicating if the value is of the type undefined
.
-
isNullish
Takes an unknown
value and returns a boolean indicating if the value is of the type Nullish
.
-
isDefined
Takes an unknown
value and returns a boolean indicating if the value is not of the type Nullish
.
-
isJSONValue
Takes an unknown
value and returns a boolean indicating if the value is of the type JSONValue
.
-
isJSONArray
Takes an JSONValue
value and returns a boolean indicating if the value is of the type JSONArray
.
-
isJSONObject
Takes an JSONValue
value and returns a boolean indicating if the value is of the type JSONObject
.
Reference: Type Coerce
-
makeString
Takes an unknown
value and converts it to it's textual representation. A value that cannot be cleaning converted will trigger an error.
-
makeNumber
Takes an unknown
value and converts it to it's numerical representation. A value that cannot be cleaning converted will trigger an error.
-
makeBoolean
Takes an unknown
value and converts it to it's boolean representation. A value that cannot be cleaning converted will trigger an error.
Reference: Types
-
JSONValue
A union of all the JSON compatible types: JSONArray
, JSONObject
, number
, string
, boolean
, null
.
-
JSONObject
An alias to Dictionary<JSONValue>
which can represent any JSON Object
value.
-
JSONArray
An alias to Array<JSONValue>
which can represent any JSON Array
value.
-
Dictionary
An alias to Record<string, Type>
where Type
is a generic parameter that default to unknown
. This type can be used to represent a typical key-value map constructed from a JS Object
. Where possible use Map
instead, as it is specifically designed for this purpose and has better protection against null errors in TS.
-
Index
A union of the number
and string
types that represent a value that could be used to index an element within a JS Object
.
-
Indexable
An alias to Record<Index, Type>
where Type
is a generic parameter that default to unknown
. This type can be used to represent an unknown key-value object that can be indexed using a number
or string
. It is intended to be used to ease the transition of JS project to TS. Where possible use Dictionary
or preferably Map
instead, as it is specifically designed for this purpose and has better protection against null errors in TS.
-
Nullish
A union of undefined
and null
. Generally preferable to either null
or undefined
on non-validated input. However, be aware of varying behavior between these 2 types in JS around optional members, default parameters and equality.
-
Optional
A union of Type
and Nullish
where Type
is a generic parameter.
-
UnknownFunction
A stricter alternative to the type Function
. It accepts any number of unknown parameters, and returns an unknown valid. Allowing you to reference an untyped function in a slightly safer manner. This does not provide any arity or type checks for the parameters.
-
UnknownAsyncFunction
Identical to UnknownFunction
in all ways but 1, it returns Promise<unknown>
instead.
Changelog
v1.0.0
v1.1.0
- Documentation update.
- Fix: asDefined was returning
unknown
. - Breaking change: rename ObjectDict to Dictionary.
- Add: Nullish type (
null | undefined
). - Change: Dictionary no longer contains
T | undefined
union. - Change: Optional type now also includes
null
in the type union.