Quickstart
We highly encourage you to check out the Example Apps to get a feel of tRPC and getting up & running as seamless as possible.
Installation
⚠️ Requirements: tRPC requires TypeScript > 4.1 as it relies on Template Literal Types.
npm install @trpc/server
For implementing tRPC endpoints and routers. Install in your server codebase.
npm install @trpc/client @trpc/server
For making typesafe API calls from your client. Install in your client codebase (@trpc/server is a peer dependency of @trpc/client).
npm install @trpc/react react-query@3
For generating a powerful set of React hooks for querying your tRPC API. Powered by react-query.
npm install @trpc/next
A set of utilies for integrating tRPC with Next.js.
Installation Snippets
npm:
npm install @trpc/server @trpc/client @trpc/react react-query@3 @trpc/next
yarn:
yarn add @trpc/server @trpc/client @trpc/react react-query@3 @trpc/next
Defining a router
Let's walk through the steps of building a typesafe API with tRPC. To start, this API will only contain two endpoints:
getUser(id: string) => { id: string; name: string; }
createUser(data: {name:string}) => { id: string; name: string; }
Create a router instance
First we define a router somewhere in our server codebase:
import { initTRPC } from '@trpc/server';
const t = initTRPC()();
const appRouter = t.router({});
// only export *type signature* of router!
// to avoid accidentally importing your API
// into client-side code
export type AppRouter = typeof appRouter;
Add a query procedure
Use t.procedure.query() to add a query procedure/endpoint to the router. Methods:
- input: Optional. This should be a function that validates/casts the input of this procedure and either returns a strongly typed value (if valid) or throws an error (if invalid). Alternatively you can pass a Zod, Superstruct or Yup schema.
- query: This is the actual implementation of the procedure (a "resolver"). It's a function with a single- reqargument. The validated input is passed into- req.inputand the context is in- req.ctx(more about context later!)
The following would create a query procedure called getUser that takes a single string argument and returns a user object:
import { initTRPC } from '@trpc/server';
const t = initTRPC()();
const appRouter = t.router({
  getUser: t.procedure
    .input((val: unknown) => {
      if (typeof val === 'string') return val;
      throw new Error(`Invalid input: ${typeof val}`);
    })
    .query((req) => {
      req.input; // string
      return { id: req.input, name: 'Bilbo' };
    }),
});
export type AppRouter = typeof appRouter;
Add a mutation procedure
Similarly to GraphQL, tRPC makes a distinction between query and mutation procedures. Let's add a createUser mutation:
createUser(payload: {name: string}) => {id: string; name: string};
import { initTRPC } from '@trpc/server';
import { z } from 'zod';
const t = initTPRC()();
const appRouter = t.router({
  getUser: t.procedure
    .input((val: unknown) => {
      if (typeof val === 'string') return val;
      throw new Error(`Invalid input: ${typeof val}`);
    })
    .query((req) => {
      req.input; // string
      return { id: req.input, name: 'Bilbo' };
    }),
  createUser: t.procedure
    .input(z.object({ name: z.string() }))
    .mutation((req) => {
      // use your ORM of choice
      return await UserModel.create({
        data: req.input,
      });
    })
});
export type AppRouter = typeof appRouter;
Next steps
tRPC includes more sophisticated client-side tooling designed for React projects generally and Next.js specifically. Read the appropriate guide next: