Typer is a comprehensive TypeScript validation library that provides robust type checking, schema validation, and runtime type safety. Built with modern TypeScript features including generics, type guards, and advanced type inference.
Infer<typeof schema> derives the static type from the runtime schema (3.2+)is/isType — ~70M is() ops/sec and ~74M isType() ops/sec on hot literals (3.2.2+)npm install @illavv/run_typer
import { Typer } from '@illavv/run_typer';
const typer = new Typer();
// 3.1+: pass a literal alias and the return type is inferred — no <generic> needed
const message = typer.isType('string', 'Hello World'); // message: string
const count = typer.isType('number', 42); // count: number
// Type guards narrow automatically from the literal alias
if (typer.is(userInput, 'string')) {
// userInput is now typed as string
console.log(userInput.toUpperCase());
}
// Non-throwing variant
const result = typer.safeParse('number', userInput);
if (result.success) {
// result.data: number
}
import { Typer, type Infer } from '@illavv/run_typer';
const typer = new Typer();
// One literal drives both runtime validation and the static type
const user = typer.parse(
{ id: 'number', name: 'string', email: 'string?' },
payload,
);
// user is typed: { id: number; name: string; email?: string | null }
// Reusable schemas with derived types — no `as const` needed
const userSchema = typer.schema({
id: 'number',
name: 'string',
email: 'string?',
tags: ['string'],
address: { city: 'string', zip: 'string?' },
});
type User = Infer<typeof userSchema>;
const u = typer.parse(userSchema, payload); // throws + typed as User
const r = typer.safeParse(userSchema, payload); // { success, data: User } | { error }
Mix-and-match validator functions inside a schema (any slot accepts a
Validator<T> from a built-in helper or your own):
const strict = typer.schema({
id: typer.isPositiveInteger,
name: typer.isNonEmptyString,
color: (v) => typer.isHexColor(v),
role: 'admin|user|guest',
});
import { Typer } from '@illavv/run_typer';
const typer = new Typer();
// Type guards (return boolean)
console.log(typer.is<string>("Hello", "string")); // true
console.log(typer.is<number>(123, "number")); // true
console.log(typer.is<boolean>(true, "boolean")); // true
console.log(typer.is<unknown[]>([], "array")); // true
console.log(typer.is<object>({}, "object")); // true
// Type validation (throws on error)
const str = typer.isType<string>('string', 'Hello'); // Returns 'Hello' typed as string
const num = typer.isType<number>('number', 42); // Returns 42 typed as number
const typer = new Typer();
// Multiple type validation
const value = typer.isType<string | number>(['string', 'number'], 'Hello');
// Specific validations
const email = typer.isEmail('user@example.com');
const phone = typer.isPhoneNumber('+1234567890');
const url = typer.isURL('https://example.com');
// Array validations
const numbers = typer.isArrayOf<number>('number', [1, 2, 3]);
const nonEmpty = typer.isNonEmptyArray<string>(['a', 'b']);
// Range and constraints
const age = typer.isInRange(18, 65, 25);
const positiveInt = typer.isPositiveInteger(42);
const typer = new Typer();
// Modern JavaScript types
const bigIntVal = typer.isType<bigint>('bigint', BigInt(123));
const buffer = typer.isType<ArrayBuffer>('arraybuffer', new ArrayBuffer(8));
const typedArray = typer.isType<Int32Array>('typedarray', new Int32Array(4));
const dataView = typer.isType<DataView>('dataview', new DataView(buffer));
// Collections
const map = typer.isType<Map<string, number>>('map', new Map());
const set = typer.isType<Set<string>>('set', new Set());
// Special validations
const jsonStr = typer.isType<string>('json', '{"valid": "json"}');
const validDate = typer.isType<Date>('date', new Date());
const regex = typer.isType<RegExp>('regexp', /pattern/);
const typer = new Typer();
const userSchema = {
name: "string",
age: "number",
email: "string?", // Optional field
address: {
street: "string",
city: "string",
zip: "number|string" // Union types
},
hobbies: ["string"], // Array of strings
isActive: "boolean"
};
const userData = {
name: "John Doe",
age: 30,
address: {
street: "123 Main St",
city: "New York",
zip: 10001
},
hobbies: ["reading", "coding"],
isActive: true
};
const result = typer.checkStructure(userSchema, userData);
if (result.isValid) {
console.log("✅ Valid user data");
} else {
console.log("❌ Validation errors:", result.errors);
}
// Strict mode (rejects extra properties)
const strictResult = typer.checkStructure(userSchema, userData, '', true);
const typer = new Typer();
// Register a custom validator
typer.registerType("positive", (value) => {
if (typeof value !== "number" || value <= 0) {
throw new TypeError("Value must be a positive number");
}
return value;
});
// Register email validator with override
typer.registerType("email", (value) => {
const email = typer.isType<string>('string', value);
if (!/^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(email)) {
throw new TypeError('Invalid email format');
}
return email;
}, true); // Override existing if present
// Use custom type
console.log(typer.is(10, "positive")); // true
console.log(typer.is(-5, "positive")); // false
// List all types
console.log(typer.listTypes());
// Export/Import types
const typesJson = typer.exportTypes();
typer.importTypes(typesJson);
// Remove custom type
typer.unregisterType("positive");
const typer = new Typer();
// Single parameter function
const safeMultiply = typer.expect(
(x: number) => x * 2,
{
paramTypes: ['number'],
returnType: ['number']
}
);
console.log(safeMultiply(4)); // 8
// safeMultiply("hello"); // Throws TypeError
// Multiple parameters
const safeAdd = typer.expect(
(x: number, y: number) => x + y,
{
paramTypes: ['number', 'number'],
returnType: ['number']
}
);
// Async function support
const asyncFunc = typer.expect(
async (x: number): Promise<string> => x.toString(),
{
paramTypes: ['number'],
returnType: ['string']
}
);
// Multiple return types
const flexibleFunc = typer.expect(
(x: boolean) => x ? 42 : "string",
{
paramTypes: ['boolean'],
returnType: ['number', 'string']
}
);
is<T>(value: unknown, types: string | string[]): value is TType guard that returns boolean. Safe for TypeScript type narrowing.
isType<T>(types: string | string[], value: unknown): TValidates type and returns the value cast to T. Throws TypeError on failure.
asString(value: unknown): stringValidates and returns a string. Alias for isType<string>('string', value).
asNumber(value: unknown): numberValidates and returns a number.
asBoolean(value: unknown): booleanValidates and returns a boolean.
asArray<T>(value: unknown): T[]Validates and returns an array.
asObject<T>(value: unknown): TValidates and returns an object.
isEmail(value: unknown): stringValidates email format.
isURL(value: unknown): stringValidates URL format.
isPhoneNumber(value: unknown): stringValidates international phone numbers (7-15 digits, ITU-T E.164 standard).
isArrayOf<T>(elementType: string, value: unknown): T[]Validates array with specific element type.
isNonEmptyString(value: unknown): stringValidates non-empty strings.
isNonEmptyArray<T>(value: unknown): T[]Validates non-empty arrays.
isOneOf<T>(values: readonly T[], value: unknown): TValidates value is one of specified options.
isInRange(min: number, max: number, value: unknown): numberValidates number within range.
isInteger(value: unknown): numberValidates integer values.
isPositiveNumber(value: unknown): numberValidates positive numbers.
isPositiveInteger(value: unknown): numberValidates positive integers.
isNegativeNumber(value: unknown): numberValidates negative numbers.
isNegativeInteger(value: unknown): numberValidates negative integers.
isFiniteNumber(value: unknown): numberStricter than isType('number', x): rejects NaN and Infinity. (3.1+)
isSafeInteger(value: unknown): numberValidates integers within Number.MIN/MAX_SAFE_INTEGER. (3.1+)
isUUID(value: unknown): stringValidates UUID strings (RFC 4122, versions 1–5). (3.1+)
isIPv4(value: unknown): string / isIPv6(value: unknown): stringValidates IPv4 / IPv6 addresses. (3.1+)
isHexColor(value: unknown): stringValidates CSS hex colors (#RGB, #RGBA, #RRGGBB, #RRGGBBAA). (3.1+)
isISODate(value: unknown): DateValidates an ISO 8601 string and returns the parsed Date. (3.1+)
isBase64(value: unknown, opts?: { urlSafe?: boolean, requirePadding?: boolean }): stringValidates Base64 strings. (3.1+)
isPlainObject<T>(value: unknown): TValidates a plain object literal (rejects class instances, arrays, Map, Set). (3.1+)
isPromise<T>(value: unknown): Promise<T>Validates a Promise / thenable. (3.1+)
isInstanceOf<T>(ctor: new (...args: never[]) => T, value: unknown): TType-safe instanceof check. (3.1+)
matches(regex: RegExp, value: unknown): stringValidates a string matches the given regex. (3.1+)
isLength<T>(bounds: { min?: number; max?: number }, value: unknown): TValidates the length of a string or array. (3.1+)
isEmpty(value: unknown): unknown / isNonEmpty<T>(value: unknown): TPolymorphic emptiness check across string, array, Map, Set, object. (3.1+)
nullable<T>(validator: Validator<T>): Validator<T | null>Wraps a validator so null is also accepted.
optional<T>(validator: Validator<T>): Validator<T | undefined>Wraps a validator so undefined is also accepted.
union<T extends readonly unknown[]>(...validators): Validator<T[number]>Tries each validator in order; succeeds on the first match.
safeParse<K>(types, value): ParseResult<TypeMap[K]>Same input as isType, returns a discriminated union
{ success: true, data } | { success: false, error: TypeError } instead
of throwing.
parse<S>(schemaOrTypeOrValidator, value): T (3.2+)Universal entry point. Accepts a type alias, an array of aliases, a
Validator<T> function, or a Schema object. Throws TypeError on
failure; on success returns the value typed via Infer<S> (for
schemas) or TypeMap[K] (for type aliases).
safeParse<S>(schemaOrTypeOrValidator, value): ParseResult<T> (3.2+ for schema overload)Non-throwing variant. Returns
{ success: true, data } | { success: false, error: TypeError }.
schema<const S>(definition: S): S (3.2+)Identity helper that preserves literal types of a schema declared in a
variable. Use it to derive Infer<typeof schema> without as const.
Infer<S> (3.2+, type-only export)Derives a TypeScript type from a runtime schema literal — handles
?-suffix optionals, |-unions, array elements, nested objects, and
embedded Validator<T> functions.
checkStructure(schema: Record<string, unknown>, obj: Record<string, unknown>, path?: string, strictMode?: boolean): StructureValidationReturnValidates object structure against schema. Returns {isValid: boolean, errors: string[]}.
Lower-level than parse — kept for backward compatibility and for the
strict-mode entry point.
registerType(name: string, validator: (value: unknown) => unknown, override?: boolean): voidRegisters custom type validator.
unregisterType(name: string): voidRemoves registered type.
listTypes(): string[]Returns all registered type names.
exportTypes(): stringExports types as JSON string.
importTypes(json: string): voidImports types from JSON string.
expect(func: Function, types: TyperExpectTypes): FunctionWraps function with type checking for parameters and return value.
validate(schema: Record<string, string | string[]>, obj: Record<string, unknown>): string[]Validates object against simple schema, returns error array.
assert(value: unknown, expectedType: string | string[]): voidLogs warning if type assertion fails.
Primitives: string, number, boolean, bigint, symbol, undefined, null
Objects: object, array, function, date, regexp, map, set
Advanced: arraybuffer, dataview, typedarray, json, domelement
Aliases: Short forms like s/str for string, n/num for number, etc.
Typer has 100% line / 100% function / 99.3% statement / 95.7% branch coverage with 290 comprehensive tests covering:
npm test # Run tests
npm run test:watch # Watch mode
npm run test:coverage # Coverage report
npm run build # Build all formats
npm run build:docs # Build with documentation
Outputs:
dist/Typer.min.js - UMD formatdist/Typer.esm.min.js - ES modulesdist/Typer.cjs.min.js - CommonJSdist/Typer.d.ts - TypeScript definitionsMIT License - see LICENSE file for details.
Contributions are welcome! Please:
Michael Lavigna
Built with ❤️ and TypeScript