Skip to content
This repository was archived by the owner on Apr 8, 2022. It is now read-only.

sparebytes/typesmith

Folders and files

NameName
Last commit message
Last commit date

Latest commit

19d86d9 Β· Nov 1, 2019
Jun 18, 2019
Jun 26, 2019
Jun 25, 2019
Oct 11, 2019
Nov 1, 2019
Jun 18, 2019
Jun 18, 2019
Jun 19, 2019
Jun 18, 2019
Jun 18, 2019
Oct 11, 2019
Jun 18, 2019
Oct 11, 2019
Jun 21, 2019
Jun 18, 2019
Jun 18, 2019
Oct 11, 2019
Jun 18, 2019
Oct 11, 2019
Jul 3, 2019
Oct 11, 2019

Repository files navigation

Typesmith: Easy Runtime Validation

Typesmith logo

Inspired by typescript-is.

Transforms typescript interfaces, classes and other types into runtime validation functions. It leverages these fine libraries:

  • ts-json-schema-generator - Generate JSON Schema from Typescript AST
  • ajv - Validates an object against JSON Schema
  • ajv-keywords - Additional validation capabilities via custom JSON Schema keywords

πŸ’Ώ Quick Start Guide

  1. Install typesmith

    npm i --save typesmith
  2. Install ttypescript

    npm i --save-dev ttypescript
  3. Edit your tsconfig.json to use the plugin

    {
      "compilerOptions": {
        "plugins": [{ "transform": "typesmith/transformer" }]
      }
    }
  4. Replace tsc ... commands with ttsc ... commands.

πŸ“ Example

import { assertTypeFn, DateString } from "typesmith";

interface Person {
  /** @minLength 1 */
  firstName: string;
  lastName: string;
  dateOfBirth: DateString;
}

const assertPerson = assertTypeFn<Person>();

const jillSmith = { firstName: "Jill", lastName: "Smith", dateOfBirth: "1990-12-31" };
const janeDoe = { firstName: "Jane", lastName: "Doe", dateOfBirth: "1990-12-31" };
const invalidPerson = {};

// Validation Successful Result
assertPerson(jillSmith).isSuccess === true;
assertPerson(jillSmith).unwrap() === jillSmith;
assertPerson(jillSmith).getErrors() === null;
assertPerson(jillSmith).getOrElse(janeDoe) === jillSmith;
assertPerson(jillSmith).getOrElseL(errors => janeDoe) === jillSmith;

// Validation Failure Result
assertPerson(invalidPerson).isSuccess === false;
assertPerson(invalidPerson).unwrap(); // Throws Error
assertPerson(invalidPerson).getErrors().length > 0;
assertPerson(invalidPerson).getOrElse(janeDoe) === janeDoe;
assertPerson(invalidPerson).getOrElseL(errors => janeDoe) === janeDoe;

πŸŽ›οΈ Options

  • allErrors: (default: true) returns all errors instead of returning only the first one
  • removeAdditional: (default: false) remove additional properties
  • useDefaults: (default: false) replace missing or undefined properties and items with the values from corresponding default keywords
  • coerceTypes: (default: false) change data type of data to match type keyword
  • lazyCompile: (default: true) wait to compile validation function until first use

Options are specifiable at the global and type level, EG:

import { assertTypeFn, settings, Validatable } from "typesmith";

// Globally:
settings.updateGlobalValidationOptions({ removeAdditional: true });

// Type-Level
assertTypeFn<{ name: string }>({ coerceTypes: true });
@Validatable({ coerceTypes: true }) class Foo {}

πŸ’£ Caveats

Compiler crashes in rare but specific circumstances:

  • Mixin classes:
    // Crashes
    @Validatable()
    export class PersonSearchRequest extends SearchRequestOf(PersonFilters) {}
  • Anonymous recursive generic types:
    // Okay
    type NumberBTree = BTree<number>;
    assertTypeFn<NumberBTree>();
    
    // (Works as of v0.9.8)
    assertTypeFn<BTree<number>>();

AJV's type coercion ineffective for primitives

With AJV's coerceTypes enabled, coerced primitives will validate but will return the original value. EG:

// Coercion of boxed value works
assertTypeFn<number>({value: "12"}).unwrap().value === 12;
// Coercion of primitive doesn't
assertTypeFn<number>("12").unwrap() === "12"

@Validate() w/ Generic Type Parameters

Defaults must be provided to classes with generic type paramters. EG:

// Okay
@Validate() class Foo<A = any> {}

// Crashes
@Validate() class Foo<A> {}

Dates

TODO

note: try using { DateInstance, DateTimeFlex, DateTimeString } from "typesmith"

βœ” Todo

Features

  • feat: allow strings to be coerced to other primitives without using AJV's built-in coercion since it is too lenient. EG, false shouldn't be coerced to 0
  • feat: use ajv-pack for precompiled validation functions. ajv-pack has restrictions so this would need to be optional.

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Packages

No packages published