From a33d782e01fe361b7cc47eccbd1af3f283472215 Mon Sep 17 00:00:00 2001 From: Erik Brinkman Date: Sun, 21 Feb 2021 19:58:20 -0500 Subject: [PATCH 1/3] Update README.md --- README.md | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/README.md b/README.md index ec11b9be7..1215f53dc 100644 --- a/README.md +++ b/README.md @@ -274,6 +274,8 @@ if (validate(data)) { With JSON Type Definition schema: +In JavaScript: + ```javascript const Ajv = require("ajv").default @@ -292,6 +294,22 @@ const valid2 = validate({}) // false, foo is required const valid3 = validate({foo: 1, bar: 2}) // false, bar is additional ``` +In TypeScript: + +```typescript +import {JTDSchemaType} from "ajv" + +type MyData = {foo: number} + +// Optional schema type annotation for schema to match MyData type. +// To use JTDSchemaType set `strictNullChecks: true` in tsconfig `compilerOptions`. +const schema: JTDSchemaType = { + properties: { + foo: {type: "float64"}, + }, +} +``` + See [this test](./spec/types/json-schema.spec.ts) for an advanced example, [API reference](./docs/api.md) and [Options](./docs/api.md#options) for more details. Ajv compiles schemas to functions and caches them in all cases (using schema itself as a key for Map) or another function passed via options), so that the next time the same schema is used (not necessarily the same object instance) it won't be compiled again. From 579ed947b381384a088e37d9331c5d26c6743b13 Mon Sep 17 00:00:00 2001 From: Erik Brinkman Date: Sun, 21 Feb 2021 20:38:14 -0500 Subject: [PATCH 2/3] Update json-type-definition.md --- docs/json-type-definition.md | 68 ++++++++++++++++++++++++++++++++++++ 1 file changed, 68 insertions(+) diff --git a/docs/json-type-definition.md b/docs/json-type-definition.md index 4a8dc0015..d24ceb3b3 100644 --- a/docs/json-type-definition.md +++ b/docs/json-type-definition.md @@ -22,6 +22,7 @@ const ajv = new AjvJTD() - [values](#values-schema-form) (for dictionary) - [ref](#ref-schema-form) (to reference a schema in definitions) - [empty](#empty-schema-form) (for any data) +- [JTDSchemaType](#jtdschematype) - [Extending JTD](#extending-jtd) - [metadata](#metadata-schema-member) - [union](#union-keyword) @@ -313,6 +314,73 @@ Unlike JSON Schema, JTD does not allow to reference: Empty JTD schema defines the data instance that can be of any type, including JSON `null` (even if `nullable` member is not present). It cannot have any member other than `nullable` and `metadata`. +## JTDSchemaType + +The type `JTDSchemaType` can be used to validate that the written schema matches the type you expect to validate. This type is strict such that if typescript compiles, you should require no further type guards. The downside of this is that the types that `JTDSchemaType` can verify are limited to the types that JTD can verify. If a type doesn't verify, `JTDSchemaType` should resolve to `never`, throwing an error when you try to assign to it. This means that types like `1 | 2 | 3`, or general untagged unions (outside of unions of string literals) cannot be used with `JTDSchemaType`. + +### Most Schemas + +Most straightforward types should work with `JTDSchemaType`, e.g. +```typescript +interface MyType { + num: number; + optionalStr?: string; + nullableEnum: "1" | "2"; + values: Record; +} + +const schema: JTDSchemaType = { + properties: { + num: { type: "float64" }, + nullableEnum: { enum: ["1", "2"], nullable: true }, + values: { values: { type: "int32" } }, + }, + optionalProperties: { + optionalStr: { type: "string" }, + } +} +``` +will compile. Using `schema` with AJV will guarantee type safety. + +### Ref Schemas + +Ref schemas are a little more advanced, because the types of every definition must be specified in advance. +A simple ref schema is relatively straightforward: +```typescript +const schema: JTDSchemaType<{ val: number }, { num: number }> = { + definitions: { + num: { type: "float64" } + }, + properties: { + val: { ref: "num" } + }, +} +``` +note that the type of all definitions was included as a second argument to `JTDSchemaType`. + +This also works for recursive schemas: +```typescript +type LinkedList = { val: number, next?: LinkedList } +const schema: JTDSchemaType = { + definitions: { + node: { + properties: { + val: { type: "float64" }, + }, + optionalProperties: { + next: { ref: "node" }, + }, + }, + }, + ref: "node", +} +``` + +### Notable Omissions + +`JTDSchemaType` currently validats that if the schema compiles it will verify an accurate type, but there are a few places with potentially unexpected behavior. +`JTDSchemaType` doesn't verify the schema is correct. It won't reject schemas that definitions anywhere by the root, and it won't reject discriminator schemas that still define the descriminator in mapping properties. It also won't verify that enum schemas have every enum member as this isn't generally feasible in typescript yet. + ## Extending JTD ### Metadata schema member From fc5666a0d54210aca16eea394a6c8474cb56d4f7 Mon Sep 17 00:00:00 2001 From: Erik Brinkman Date: Mon, 22 Feb 2021 18:55:47 -0500 Subject: [PATCH 3/3] Update json-type-definition.md --- docs/json-type-definition.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/docs/json-type-definition.md b/docs/json-type-definition.md index d24ceb3b3..7e074c86e 100644 --- a/docs/json-type-definition.md +++ b/docs/json-type-definition.md @@ -325,14 +325,14 @@ Most straightforward types should work with `JTDSchemaType`, e.g. interface MyType { num: number; optionalStr?: string; - nullableEnum: "1" | "2"; + nullableEnum: "v1.0" | "v1.2" | null; values: Record; } const schema: JTDSchemaType = { properties: { num: { type: "float64" }, - nullableEnum: { enum: ["1", "2"], nullable: true }, + nullableEnum: { enum: ["v1.0", "v1.2"], nullable: true }, values: { values: { type: "int32" } }, }, optionalProperties: {