Skip to main content

Define Contract

Use the @ts-rest/core package to define a contract. Nesting routers can help organize your resources. For example, /users/:id/posts could have a nested router contract.users.posts. This is the path that you'd use on the client to query the API.

Breaking down the contract to sub-routers also allows you to split up the backend implementation. For example, in Nest.js you could have multiple controllers for the sub-routers.

You can define your contract fields such as body, query, pathParams, and headers using a plain Typescript through the c.type helper, or you can use Zod objects.

import { initContract } from '@ts-rest/core';

const c = initContract();
export const contract = c.router({
createPost: {
method: 'POST',
path: '/posts',
// ^ Note! This is the full path on the server, not just the sub-path of a route
responses: {
201: c.type<Post>(),
},
body: z.object({
title: z.string(),
content: z.string(),
published: z.boolean().optional(),
description: z.string().optional(),
}),
summary: 'Create a post',
metadata: { role: 'user' } as const,
},
getPosts: {
method: 'GET',
path: '/posts',
responses: {
200: c.type<{ posts: Post[]; total: number }>(),
},
headers: z.object({
pagination: z.string().optional(),
}),
query: z.object({
take: z.string().transform(Number).optional(),
skip: z.string().transform(Number).optional(),
search: z.string().optional(),
}),
summary: 'Get all posts',
metadata: { role: 'guest' } as const,
},
});

Query Parameters

All query parameters, by default, need to have an input type of string since query strings inherently cannot be typed, however, ts-rest allows you to encode query parameters as JSON values. This will allow you to use input types other than strings in your contracts.

const c = initContract();
export const contract = c.router({
getPosts: {
...,
query: z.object({
take: z.number().default(10),
skip: z.number().default(0),
search: z.string().optional(),
}),
}
});

Check the relevant sections to see how to enable JSON query encoding/decoding on the client or server.

Path Parameters

You can define URL path parameters in your contract using a Zod object with the path and pathParams keys.

const c = initContract();
export const contract = c.router({
getPost: {
...,
path: '/api/posts/:id',
pathParams: z.object({
id: z.string(),
}),
}
});

Since URLs are just strings, any type other than string would need to be coerced or transformed into the required type. Using zod's .coerce for example:

const c = initContract();
export const contract = c.router({
getPost: {
...,
path: '/api/posts/:id',
pathParams: z.object({
id: z.coerce.number(),
}),
}
});

Headers

You can define headers in your contract, however, they must have an input type of string, as they cannot be typed otherwise. You can use Zod transforms or coercion to transform any string values to different types if needed.

const c = initContract();
export const contract = c.router({
getPosts: {
...,
headers: z.object({
authorization: z.string(),
pagination: z.coerce.number().optional(),
}),
}
});

You can also define base headers for all routes in a contract and its sub-contracts, this is useful for things like authorization headers. This will force the client to always pass

const c = initContract();
export const contract = c.router(
{
// ...endpoints
},
{
baseHeaders: z.object({
authorization: z.string(),
}),
}
);

Responses

To define your response types, they need to be defined as a map of status codes to response types.

Responses are assumed by default to be JSON responses, however, you can define other response types using c.otherResponse and passing in the content type header value and body type or Zod schema.

const c = initContract();

export const contract = c.router({
createPost: {
...,
responses: {
201: z.object({
id: z.string(),
title: z.string(),
content: z.string(),
published: z.boolean(),
description: z.string(),
}),
404: c.type<{ message: string }>(),
500: c.otherResponse({
contentType: 'text/plain',
body: z.literal('Server Error'),
})
},
...,
},
});

Common Responses

APIs often have shared common response schemas, specifically for error responses. You can define these common responses in the contract options.

const c = initContract();
export const contract = c.router(
{
// ...endpoints
},
{
commonResponses: {
404: c.type<{ message: 'Not Found'; reason: string }>(),
500: c.otherResponse({
contentType: 'text/plain',
body: z.literal('Server Error'),
}),
},
}
);

Strict Response Status Codes

To help with incremental adoption, ts-rest, by default, will allow any response status code to be returned from the server even if it is not defined in the contract.

As a result, the response types on the client will include all possible HTTP status codes, even ones that are not defined in the contract with those mapping to a body type of unknown.

If you would like to disable this functionality and only allow the response status codes defined in the contract, you can set the strictStatusCodes option to true when initializing the contract.

const c = initContract();
export const contract = c.router(
{
// ...endpoints
},
{
strictStatusCodes: true,
}
);

You can also set this option on a per-route basis which will also override the global option.

const c = initContract();
export const contract = c.router({
getPosts: {
...,
strictStatusCodes: true,
}
});

Combining Contracts

You can combine contracts to create a single contract, helpful if you want many sub-contracts, especially if they are huge.

const c = initContract();

export const postContract = c.router({
getPosts: {
method: 'GET',
path: '/posts',
responses: {
200: c.type<{ posts: Post[]; total: number }>(),
},
query: z.object({
take: z.string().transform(Number).optional(),
skip: z.string().transform(Number).optional(),
search: z.string().optional(),
}),
summary: 'Get all posts',
},
});

export const contract = c.router({
posts: postContract,
});

Metadata

You can attach metadata with any type to your contract routes that can be accessed anywhere throughout ts-rest where you have access to the contract route object.

const c = initContract();
export const contract = c.router({
getPosts: {
...,
metadata: { role: 'guest' } as const,
}
});
caution

As the contract is not only used on the server, but on the client as well, it will also be part of your client-side bundle. You should not put any sensitive information in the metadata.

Intellisense

For intellisense on your contract types, you can use JSDoc Reference.

const c = initContract();

export const contract = c.router({
getPosts: {
method: 'GET',
path: '/posts',
responses: {
200: c.type<{ posts: Post[]; total: number }>(),
},
query: z.object({
/**
* @type {string} - UTC timestamp in milliseconds
*/
beginDate: z.string(),
/**
* @type {string} - UTC timestamp in milliseconds
*/
endDate: z.string(),
}),
summary: 'Get posts within time-range',
},
});

Options

These configuration options allow you to modify how your contract functions.

Base Headers

You can assign baseHeaders which will be merged with the contract headers. Here's how to set it:

const c = initContract();
export const contract = c.router(
{
// ...endpoints
},
{
baseHeaders: z.object({
authorization: z.string(),
}),
}
);

Path Prefix

The pathPrefix option allows you to add a prefix to paths, allowing more modular and reusable routing logic. This option is applied recursively, allowing the application of prefixes to nested contracts. In addition, when hovering over the contract, the prefixed path will appear at the beginning of the path for ease of use.

Here is an example of how to use the pathPrefix option. In this example, the resulting path is /api/v1/mypath.

const c = initContract();
export const contract = c.router(
{
getPost: {
path: '/mypath',
//... Your Contract
},
},
{
pathPrefix: '/api/v1',
}
);

You can also use this feature in nested contracts, as shown below. In this case, the resulting path is /v1/posts/mypath, with the pathPrefix of the nested contract following the pathPrefix of the parent contract.

const nestedContract = c.router(
{
getPost: {
path: '/mypath',
//... Your Contract
},
},
{
pathPrefix: '/posts',
}
);

const parentContract = c.router(
{
posts: nestedContract,
},
{
pathPrefix: '/v1',
}
);