Skip to content

Modern utility library and typescript typings for building JSON Schema documents

License

Notifications You must be signed in to change notification settings

triggerdotdev/json-schema-fns

Repository files navigation

json-schema-fns

Modern utility library and typescript typings for building JSON Schema documents dynamically

Features

  • Build JSON Schema documents for various drafts (currently only draft-2020-12 but more coming soon)
  • Strongly typed documents using Typescript
  • Allows you to build correct JSON Schema documents using dynamic data

Usage

Create a simple draft-2020-12 document:

import { s } from "json-schema-fns";

const schema = s.object({
  properties: [s.requiredProperty("foo", s.string()), s.property("bar", s.int())],
});

schema.toSchemaDocument();

Will result in

{
  "$schema": "https://json-schema.org/draft/2020-12/schema#",
  "$id": "https://jsonhero.io/schemas/root.json",
  "type": "object",
  "properties": {
    "foo": { "type": "string" },
    "bar": { "type": "integer" }
  },
  "required": ["foo"]
}

You can also import the types for a specific draft to use, like so:

import { s, Schema, IntSchema, StringSchema, StringFormat } from "json-schema-fns";

function buildIntSchema(maximum: number, minimum: number): IntSchema {
  return s.int({ minimum, maximum });
}

function buildStringFormat(format: JSONStriStringFormatgFormat): StringSchema {
  return s.string({ format });
}

json-schema-fns support all the features of JSON schema:

import { s } from "json-schema-fns";

const phoneNumber = s.def("phoneNumber", s.string({ pattern: "^[0-9]{3}-[0-9]{3}-[0-9]{4}$" }));
const usAddress = s.def(
  "usAddress",
  s.object({
    properties: [s.requiredProperty("zipCode", s.string())],
  }),
);

const ukAddress = s.def(
  "ukAddress",
  s.object({
    properties: [s.requiredProperty("postCode", s.string())],
  }),
);

s.object({
  $id: "/schemas/person",
  title: "Person Profile",
  description: "Attributes of a person object",
  examples: [
    {
      name: "Eric",
      email: "eric@stackhero.dev",
    },
  ],
  $comment: "This is just a preview",
  default: {},
  properties: [
    s.requiredProperty("name", s.string()),
    s.property("email", s.string({ format: "email" })),
    s.property("phoneNumber", s.ref("phoneNumber")),
    s.property("billingAddress", s.oneOf(s.ref("ukAddress"), s.ref("usAddress"))),
    s.patternProperty("^[A-Za-z]$", s.string()),
  ],
  additionalProperties: s.array({
    items: s.number({ minimum: 0, maximum: 5000 }),
  }),
  propertyNames: "^[A-Za-z_][A-Za-z0-9_]*$",
  minProperties: 3,
  maxProperties: 20,
  unevaluatedProperties: false,
  defs: [phoneNumber, usAddress, ukAddress],
}).toSchemaDocument();

Will result in

{
  "$schema": "https://json-schema.org/draft/2020-12/schema",
  "type": "object",
  "$id": "/schemas/person",
  "title": "Person Profile",
  "description": "Attributes of a person object",
  "examples": [
    {
      "name": "Eric",
      "email": "eric@stackhero.dev"
    }
  ],
  "$comment": "This is just a preview",
  "default": {},
  "minProperties": 3,
  "maxProperties": 20,
  "unevaluatedProperties": false,
  "properties": {
    "name": {
      "type": "string"
    },
    "email": {
      "type": "string",
      "format": "email"
    },
    "phoneNumber": {
      "$ref": "#/$defs/phoneNumber"
    },
    "billingAddress": {
      "oneOf": [
        {
          "$ref": "#/$defs/ukAddress"
        },
        {
          "$ref": "#/$defs/usAddress"
        }
      ]
    }
  },
  "required": ["name"],
  "patternProperties": {
    "^[A-Za-z]$": {
      "type": "string"
    }
  },
  "propertyNames": {
    "pattern": "^[A-Za-z_][A-Za-z0-9_]*$"
  },
  "additionalProperties": {
    "type": "array",
    "items": {
      "type": "number",
      "minimum": 0,
      "maximum": 5000
    }
  },
  "$defs": {
    "phoneNumber": {
      "type": "string",
      "pattern": "^[0-9]{3}-[0-9]{3}-[0-9]{4}$"
    },
    "usAddress": {
      "type": "object",
      "properties": {
        "zipCode": {
          "type": "string"
        }
      },
      "required": ["zipCode"]
    },
    "ukAddress": {
      "type": "object",
      "properties": {
        "postCode": {
          "type": "string"
        }
      },
      "required": ["postCode"]
    }
  }
}

API

s

All the builder methods for creating subschemas are available on the s object

import { s } from "json-schema-fns";

Or if you want to import a specific dialect:

import { s } from "json-schema-fns/2020";

All builder methods return a SchemaBuilder, and you can generate the JSON schema created by the builder using toSchemaDocument like so

s.object().toSchemaDocument();

Which will result in the following document

{
  "$schema": "https://json-schema.org/draft/2020-12/schema",
  "type": "object"
}

If you don't want the $schema property, use toSchema instead:

s.object().toSchema(); // { "type": "object" }

All builder methods also support the options in AnnotationSchema:

s.object({
  $id: "#/foobar",
  $comment: "This is a comment",
  default: {},
  title: "FooBar Object Schema",
  description: "This is the FooBar schema description",
  examples: [{ foo: "bar" }],
  deprecated: true,
  readOnly: true,
  writeOnly: false,
}).toSchema();

Produces the schema

{
  "type": "object",
  "$id": "#/foobar",
  "$comment": "This is a comment",
  "default": {},
  "title": "FooBar Object Schema",
  "description": "This is the FooBar schema description",
  "examples": [{ "foo": "bar" }],
  "deprecated": true,
  "readOnly": true,
  "writeOnly": false
}

s.object(options: ObjectOptions)

Builds a schema of type object, accepting a single argument of type ObjectOptions

ObjectOptions.properties

An array of optional and required properties

s.object({
  properties: [
    s.property("name", s.string()),
    s.requiredProperty("email", s.string({ format: "email" })),
  ],
}).toSchema();

Produces the schema

{
  "type": "object",
  "properties": {
    "name": { "type": "string" },
    "email": { "type": "string", "format": "email" }
  },
  "required": ["email"]
}

You can also add patternProperties

s.object({ properties: [s.patternProperty("^S_", s.string())] }).toSchema();

which produces the schema

{
  "type": "object",
  "patternProperties": {
    "^S_": { "type": "string" }
  }
}

ObjectOptions.additonalProperties

Add an additonalProperties schema:

s.object({ additionalProperties: s.number() }).toSchema();

Produces the schema

{
  "type": "object",
  "additionalProperties": {
    "type": "number"
  }
}

ObjectOptions.propertyNames

Add a propertyNames pattern:

s.object({ propertyNames: "^[A-Za-z_][A-Za-z0-9_]*$" }).toSchema();

Produces the schema

{
  "type": "object",
  "propertyNames": {
    "pattern": "^[A-Za-z_][A-Za-z0-9_]*$"
  }
}

ObjectOptions.minProperties and ObjectOptions.maxProperties

Validate the number of properties in an object using min/maxProperties

s.object({ minProperties: 4, maxProperties: 10 }).toSchema();

Produces the schema

{
  "type": "object",
  "minProperties": 4,
  "maxProperties": 10
}

ObjectOptions.unevaluatedProperties

Specify the handling of unevaluatedProperties

s.object({ unevaluatedProperties: false }).toSchema();

Produces the schema

{
  "type": "object",
  "unevaluatedProperties": false
}

s.array(options: ArrayOptions)

Builds a schema of type array, accepting a single argument of type ArrayOptions

ArrayOptions.items

Define the items schema for an array:

s.array({ items: s.string() }).toSchema();

Produces the schema

{
  "type": "array",
  "items": { "type": "string" }
}

ArrayOptions.minItems and ArrayOptions.maxItems

Define the array length

s.array({ contains: { schema: s.number(), min: 1, max: 3 }).toSchema();

Produces the schema

{
  "type": "array",
  "contains": { "type": "number" },
  "minContains": 1,
  "maxContains": 3
}

ArrayOptions.prefixItems

Allows you to perform tuple validation:

s.array({ prefixItems: [s.string(), s.number()] }).toSchema();

Produces the schema

{
  "type": "array",
  "prefixItems": [{ "type": "string" }, { "type": "number" }]
}

ArrayOptions.unevaluatedItems

Define the schema for unevaluatedItems

s.array({ unevaluatedItems: s.object() }).toSchema();

Produces the schema

{
  "type": "array",
  "unevaluatedItems": { "type": "object" }
}

ArrayOptions.contains

Define the schema contains

s.array({ contains: { schema: s.number(), min: 1, max: 3 }).toSchema();

Produces the schema

{
  "type": "array",
  "contains": { "type": "number" },
  "minContains": 1,
  "maxContains": 3
}

string

integer and number

boolean

nil

nullable

anyOf | allOf | oneOf

ifThenElse and ifThen

not

def and ref

$const

$enumerator

$true and $false

Roadmap

  • Support draft-04
  • Support draft-06
  • Support draft-07
  • Support draft/2019-09

About

Modern utility library and typescript typings for building JSON Schema documents

Resources

License

Code of conduct

Stars

Watchers

Forks

Packages

No packages published