typescanner
typescanner is a simple library for implementing Type Guard in TypeScript.
It can be easily implemented from basic primitive type guard to type guard for types defined in Type Aliases.
Install
npm i -D typescanner
Example
const Lang = {
ja: "ja",
en: "en",
} as const;
type Lang = typeof Lang[keyof typeof Lang];
const langList = Object.values(Lang);
type Post = {
id: number;
author: string | null;
body: string;
lang: Lang;
isPublic: boolean;
createdAt: Date;
tags?: string[] | null;
};
const isPost = scanner<Post>({
id: number,
author: union(string, Null),
body: string,
lang: list(langList),
isPublic: boolean,
createdAt: date,
tags: optional(array(string), Null),
});
const data = {
id: 1,
author: "taro",
body: "Hello!",
lang: "ja",
isPublic: true,
createdAt: new Date(),
tags: ["tag1", "tag2"],
} as unknown;
const post = scan(data, isPost);
post.body;
Usage
fields
fields
is used when creating a scanner
. fields
returns a Condition
or an array of conditions as defined below.
type Condition<T> = (value: unknown) => value is T;
string;
number;
boolean;
symbol;
bigint;
Undefined;
Null;
data;
union(string, null);
union<string | number>(string, number);
array(string);
array<string | number>(string, number);
optional(string);
optional<string | number>(string, number);
list(["a", "b", "c"]);
scanner
scanner
is a function for implementing type guard for Objects. It returns a Condition
of the type defined in Type Aliase by setting a field to the value of each property.
type Foo = {
a: string;
b: number;
c: boolean;
d: Date;
e: string[];
f?: string;
g: "a" | "b" | "c";
h: string | null;
i: string | number;
};
const isFoo = scanner<Foo>({
a: string,
b: number,
c: boolean,
d: date,
e: array(string),
f: optional(string),
g: list(["a", "b", "c"]),
h: union(string, Null),
i: union<string | number>(string, number),
});
const foo = {
a: "a",
b: 2,
c: true,
d: new Date(),
e: ["a", "b"],
f: "f",
g: "a",
} as unknown;
if (isFoo(foo)) {
foo.a
}
scan
scan
is used with the first argument being the value you want to validate and the second argument being the Condition
.
If the verification is successful, the "narrowed value" will be returned. If it fails, it throws an exception.
const data = scan(foo, isFoo);
data.a
const data = scan(bar, isFoo);
primitive
isString("a")
isNumber(1)
isBoolean(true)
isUndefined(undefined)
isNull(null)
isDate(new Data())
isSymbol(Symbol("a"))
isBigint(BigInt(1))
isArray
isArray(["a", "b"], isString)
isArray<string | number>(["a", 1], isString, isNumber)
isArray(["a", null, undefined], isString, isNull, isUndefined)
isOptional
isOptional("a", isString)
isOptional(undefined, isString)
isList
const Lang = {
ja: "ja",
en: "en",
} as const;
type Lang = typeof Lang[keyof typeof Lang];
const langList = Object.values(Lang);
isList("ja", langList)
Contribution
wellcome
License
MIT