Presi (short for praesidium, Latin meaning protection or guard) is a libary that lets you build interfaces and get out of the box runtime type checking and JSON deserialization. Presi was inspired by DRY, prop-types, and Scala class objects to let you only define an interface once but still allow you to:
- Use the defined interface to statically type check objects in your code,
- Check the type of arbitrary objects that come from unsafe IO sources (e.g. user input).
npm install --save presi
import {I, Z} from 'presi';
When you need to define an interface, i.e.
interface User {
name: string;
age: number;
}
instead do
class User extends I({
name: Z.string,
age: Z.number,
}) {}
-
User
is the interface with a string,name
, and a number,age
. -
I
is a mixin which creates type forUser
and attaches the deserialization function.
From there you can use User
like a regular interface. But additionally, you
can use new User(your unknown object)
which creates a User
object if given
valid data or throws if not.
const user1: User = { name: 'A', age: 1 }; // succeeds
const user2: User = { name: 1, age: 1 }; // fails typecheck
const user3: User = { name: 'A' }; // fails typecheck
const data = {
name: 100,
age: 1,
}
const user4 = new User(data); // throws
In essence,
class _ extends I({
becomes a substitute for
interface _ {
Z, which stands for deserializer, is a set of 1 and 2 arity functions that either check that it's argument is a primative or, when given a type spec, checks that it's argument follows that spec.
import {G, I, Z} from 'presi';
class ObjectDef extends I({
// stringValue: string
stringValue: Z.string,
// numberValue: number
numberValue: Z.number,
// booleanValue: boolean
booleanValue: Z.boolean,
// literalKeyValue: 'Key'
literalKeyValue: Z.literal('Key'),
// literal100Value: 100
literal100Value: Z.literal(100),
// optionalStringValue: string | undefined
optionalStringValue: Z.optional(Z.string),
// optionalLiteralPValue: 'P' | undefined (Z.o is shorthand for Z.optional)
optionalLiteralPValue: Z.o(Z.literal('P')),
// arrayOfNumbers: number[]
arrayOfNumbers: Z.array(Z.number),
// tupleNumNum: [number, number]
tupleNumNum: Z.tuple([Z.number, Z.number]),
// nestedObject: { z: { z?: string } }
nestedObject: Z.object({ z: Z.Object({z: Z.optional(Z.string) }) }),
// stringOrNumber: string | number
stringOrNumber: Z.oneOf(Z.string, Z.number),
// numberOrNull: number | null
numberOrNull: Z.oneOf(Z.number, Z.null),
// tripleTuple: [string, number, string[]]
tripleTuple: Z.tuple([Z.string, Z.number, Z.array(Z.string)]),
}) {}
class objectComposition extends I{
// objectDef: ObjectDef
objectDef: ObjectDef._,
}
- Structure once philosophy.
- Nearly free deserialization.
- Can take output from JSON.parse.
- Run time type checking.
- Usable types in TypeScript for compile time type checking.
- Classes give free structure, type, and function.
- "Structure once" means the means of deserialization needs incorporated in the declaration.
- Due to converting a user created interface to generic then back again, extraneous keys will not be type checked.
- Currently does not support reflection.