Standard Schema Spec
A consortium of schema library authors have collaborated to craft a standard interface for schema libraries to benefit the entire JavaScript ecosystem. Standard Schema provides third-party libraries a uniform integration to automatically support multiple schema libraries at once, without adding a single runtime dependency. This simplifies implementation, prevents vendor lock-in, and enables innovation, especially for smaller schema libraries with new ideas. For more information on the origins and use cases of Standard Schema, see background.
The Interface
The Standard Schema interface is a set of validation-related properties that must be defined under a key called ~standard
.
export interface StandardSchemaV1<Input = unknown, Output = Input> {
readonly "~standard": {
/**
* The version number of the standard.
*/
readonly version: 1;
/**
* The vendor name of the schema library.
*/
readonly vendor: string;
/**
* Validates unknown input values.
*/
readonly validate: (
value: unknown,
) => Result<Output> | Promise<Result<Output>>;
/**
* Inferred types associated with the schema.
*/
readonly types?: Types<Input, Output> | undefined;
};
}
~standard
contains the Standard Schema properties and can be used to test whether an object is a Standard Schema.version
defines the version number of the standard. This can be used in the future to distinguish between different versions of the standard.vendor
stores the name of the schema library. This can be useful for performing vendor-specific operations in special cases.validate
is a function that validates unknown input and returns the output of the schema if the input is valid or an array of issues otherwise. This can be discriminated by checking whether theissues
property isundefined
.types
is used to associate type metadata with the schema. This property should be declared on the schema's type, but is not required to exist at runtime. Authors implementing a schema using a class are encouraged to use TypeScript'sdeclare
keyword or other means to avoid runtime overhead.InferInput
andInferOutput
can be used to extract their corresponding types.
Implementation
Two parties are required for Standard Schema to work. First, the schema libraries that implement the standard interface, and second, the third-party libraries that accept schemas as part of their API that follow the standard interface.
Schema Library
Schemas libraries that want to support Standard Schema must implement its interface. This includes adding the ~standard
property. To make this process easier, schema libraries can optionally extend their interface from the StandardSchemaV1
interface.
It doesn't matter whether your schema library returns plain objects, functions, or class instances. The only thing that matters is that the
~standard
property is defined somehow.
import type { StandardSchemaV1 } from "@standard-schema/spec";
// Step 1: Define the schema interface
interface StringSchema extends StandardSchemaV1<string> {
type: "string";
message: string;
}
// Step 2: Implement the schema interface
function string(message: string = "Invalid type"): StringSchema {
return {
type: "string",
message,
"~standard": {
version: 1,
vendor: "valizod",
validate(value) {
return typeof value === "string"
? { value }
: { issues: [{ message }] };
},
},
};
}
Instead of implementing the StandardSchemaV1
interface natively into your library code, you can also just add it on top and reuse your existing functions and methods within the validate
function.
Third Party
Other than for schema library authors, we recommend third party authors to install the @standard-schema/spec
package when implementing Standard Schema into their libraries. This package provides the StandardSchemaV1
interface and the InferInput
and InferOutput
utility types.
npm install @standard-schema/spec --save-dev # npm
yarn add @standard-schema/spec --dev # yarn
pnpm add @standard-schema/spec --dev # pnpm
bun add @standard-schema/spec --dev # bun
deno add jsr:@standard-schema/spec --dev # deno
Alternatively, you can also copy and paste the types into your project.
After that you can accept any schemas that implement the Standard Schema interface as part of your API. We recommend using a generic that extends the StandardSchemaV1
interface in most cases to be able to infer the type information of the schema.
import type { StandardSchemaV1 } from "@standard-schema/spec";
// Step 1: Define the schema generic
function createEndpoint<TSchema extends StandardSchemaV1, TOutput>(
// Step 2: Use the generic to accept a schema
schema: TSchema,
// Step 3: Infer the output type from the generic
handler: (data: StandardSchemaV1.InferOutput<TSchema>) => Promise<TOutput>,
) {
return async (data: unknown) => {
// Step 4: Use the schema to validate data
const result = await schema["~standard"].validate(data);
// Step 5: Process the validation result
if (result.issues) {
throw new Error(result.issues[0].message ?? "Validation failed");
}
return handler(result.value);
};
}
Common Tasks
There are two common tasks that third-party libraries perform after validation fails. The first is to flatten the issues by creating a dot path to more easily associate the issues with the input data. This is commonly used in form libraries. The second is to throw an error that contains all the issue information.
Get Dot Path
To generate a dot path, simply map and join the keys of an issue path, if available.
import type { StandardSchemaV1 } from "@standard-schema/spec";
async function getFormErrors(schema: StandardSchemaV1, data: unknown) {
const result = await schema["~standard"].validate(data);
const formErrors: string[] = [];
const fieldErrors: Record<string, string[]> = {};
if (result.issues) {
for (const issue of result.issues) {
const dotPath = issue.path
?.map((item) => (typeof item === "object" ? item.key : item))
.join(".");
if (dotPath) {
if (fieldErrors[dotPath]) {
fieldErrors[dotPath].push(issue.message);
} else {
fieldErrors[dotPath] = [issue.message];
}
} else {
formErrors.push(issue.message);
}
}
}
return { formErrors, fieldErrors };
}
Schema Error
To throw an error that contains all issue information, simply pass the issues of the failed schema validation to a SchemaError
class. The SchemaError
class extends the Error
class with an issues
property that contains all the issues.
import type { StandardSchemaV1 } from "@standard-schema/spec";
class SchemaError extends Error {
public readonly issues: ReadonlyArray<StandardSchemaV1.Issue>;
constructor(issues: ReadonlyArray<StandardSchemaV1.Issue>) {
super(issues[0].message);
this.name = "SchemaError";
this.issues = issues;
}
}
async function validateInput<TSchema extends StandardSchemaV1>(
schema: TSchema,
data: unknown,
): Promise<StandardSchemaV1.InferOutput<TSchema>> {
const result = await schema["~standard"].validate(data);
if (result.issues) {
throw new SchemaError(result.issues);
}
return result.value;
}
Ecosystem
These are the libraries that have already implemented the Standard Schema interface. Feel free to add your library to the list in ascending order by creating a pull request.
Schema Libraries
- ArkType: TypeScript's 1:1 validator, optimized from editor to runtime ⛵
- Arri Schema: Type safe validator and schema builder that can be compiled to other languages
- Valibot: The modular and type safe schema library for validating structural data 🤖
- Zod (v3.24+): TypeScript-first schema validation with static type inference
Third Parties
- Formwerk: A Vue.js Framework for building high-quality, accessible, delightful forms.
- GQLoom: Weave GraphQL schema and resolvers using Standard Schema.
- Nuxt UI: A UI Library for Modern Web Apps, powered by Vue & Tailwind CSS.
- oRPC: Typesafe API's Made Simple 🪄
- TanStack Form: 🤖 Headless, performant, and type-safe form state management for TS/JS, React, Vue, Angular, Solid, and Lit.
- TanStack Router: A fully type-safe React router with built-in data fetching, stale-while revalidate caching and first-class search-param APIs.
- tRPC: 🧙♀️ Move Fast and Break Nothing. End-to-end typesafe APIs made easy.
- UploadThing: File uploads for modern web devs
Background
The Problem
Validation is an essential building block for almost any application. Therefore, it was no surprise to see more and more JavaScript frameworks and libraries start to natively support specific schema libraries. Frameworks like Astro and libraries like the OpenAI SDK have adopted Zod in recent months to streamline the experience for their users. But to be honest, the current situation is far from perfect. Either only a single schema library gets first-party support, because the implementation and maintenance of multiple schema libraries is too complicated and time-consuming, or the choice falls on an adapter or resolver pattern, which is more cumbersome to implement for both sides.
For this reason, Colin McDonnell, the creator of Zod, came up with the idea of a standard interface for schema libraries. This interface should be minimal, easy to implement, but powerful enough to support the most important features of popular schema libraries. The goal was to make it easier for other libraries to accept user-defined schemas as part of their API, in a library-agnostic way. After much thought and consideration, Standard Schema was born.
Use Cases
The first version of Standard Schemas aims to address the most common use cases of schema libraries today. This includes API libraries like tRPC and JavaScript frameworks like Astro and Qwik who secure the client/server communication in a type safe way using schemas. Or projects like the T3 Stack, which uses schemas to validate environment variables. It also includes UI libraries like Nuxt UI and form libraries like Reach Hook Form, which use schemas to validate user inputs. Especially with the rise of TypeScript, schemas became the de facto standard as they drastically improved the developer experience by providing the type information and validation in a single source of truth.
At the moment, Standard Schema deliberately tries to cover only the most common use cases. However, we believe that other use cases, such as integrating schema libraries into AI SDKs like Vercel AI or the OpenAI SDK to generate structured output, can also benefit from a standard interface.
FAQ
These are the most frequently asked questions about Standard Schema. If your question is not listed, feel free to create an issue.
Do I need to include @standard-schema/spec
as a dependency?
No. The @standard-schema/spec
package is completely optional. You can just copy and paste the types into your project, or manually add the ~standard
properties to your existing types. But you can include @standard-schema/spec
as a dev dependency and consume it exclusively with import type
. The @standard-schema/spec
package contains no runtime code and only exports types.
Why did you choose to prefix the ~standard
property with ~
?
The goal of prefixing the key with ~
is to both avoid conflicts with existing API surfaces and to de-prioritize these keys in auto-complete. The ~
character is one of the few ASCII characters that occurs after A-Za-z0-9
lexicographically, so VS Code puts these suggestions at the bottom of the list.
Why don't you use symbols for the keys instead of the ~
prefix?
In TypeScript, using a plain Symbol
inline as a key always collapses to a simple symbol
type. This would cause conflicts with other schema properties that use symbols.
const object = {
[Symbol.for('~output')]: 'some data',
};
// { [k: symbol]: string }
By contrast, declaring the symbol externally makes it "nominally typed". This means that the key is sorted in autocomplete under the variable name (e.g. testSymbol
below). Thus, these symbol keys don't get sorted to the bottom of the autocomplete list, unlike ~
-prefixed string keys.
What should I do if I only accept synchronous validation?
The ~validate
function does not necessarily have to return a Promise
. If you only accept synchronous validation, you can simply throw an error if the returned value is an instance of the Promise
class.
import type { StandardSchemaV1 } from "@standard-schema/spec";
function validateInput(schema: StandardSchemaV1, data: unknown) {
const result = schema["~standard"].validate(data);
if (result instanceof Promise) {
throw new TypeError('Schema validation must be synchronous');
}
// ...
}