tRPC
tRPC allows us to write end-to-end typesafe APIs without any code generation or runtime bloat. It uses TypeScript’s great inference to infer your API router’s type definitions and lets you call your API procedures from your frontend with full typesafety and auto-completion. When using tRPC, your frontend and backend feel closer together than ever before, allowing for an outstanding developer experience.
I built tRPC to allow people to move faster by removing the need for a traditional API-layer, while still having confidence that our apps won't break as we rapidly iterate.
Alex - creator of tRPC @alexdotjs
How do I use tRPC?
tRPC contributor trashh_dev↗ made a killer talk at Next.js Conf↗ about tRPC. We highly recommend you watch it if you haven’t already.
With tRPC, you write TypeScript functions on your backend, and then call them from your frontend. A simple tRPC procedure could look like this:
const userRouter = createTRPCRouter({
getById: publicProcedure.input(z.string()).query(({ ctx, input }) => {
return ctx.prisma.user.findFirst({
where: {
id: input,
},
});
}),
});
This is a tRPC procedure (equivalent to a route handler in a traditional backend) that first validates the input using Zod (which is the same validation library that we use for environment variables) - in this case, it’s making sure that the input is a string. If the input is not a string it will send an informative error instead.
After the input, we chain a resolver function which can be either a query↗, mutation↗, or a subscription↗. In our example, the resolver calls our database using our prisma client and returns the user whose
id
You define your procedures in
routers
users
posts
messages
appRouter
const appRouter = createTRPCRouter({
users: userRouter,
posts: postRouter,
messages: messageRouter,
});
export type AppRouter = typeof appRouter;
Notice that we only need to export our router’s type definitions, which means we are never importing any server code on our client.
Now let’s call the procedure on our frontend. tRPC provides a wrapper for
@tanstack/react-query
import { useRouter } from "next/router";
import { api } from "../../utils/api";
const UserPage = () => {
const { query } = useRouter();
const userQuery = api.users.getById.useQuery(query.id);
return (
<div>
<h1>{userQuery.data?.name}</h1>
</div>
);
};
You’ll immediately notice how good the autocompletion and typesafety is. As soon as you write
api.
Inferring errors
By default,
create-t3-app
Example usage:
function MyComponent() {
const { mutate, error } = api.post.create.useMutation();
return (
<form onSubmit={(e) => {
e.preventDefault();
const formData = new FormData(e.currentTarget);
mutate({ title: formData.get('title') });
}}>
<input name="title" />
{error?.data?.zodError?.fieldErrors.title && (
{/** `mutate` returned with an error on the `title` */}
<span className="mb-8 text-red-500">
{error.data.zodError.fieldErrors.title}
</span>
)}
...
</form>
);
}
Files
tRPC requires quite a lot of boilerplate that
create-t3-app
đź“„ pages/api/trpc/[trpc].ts
pages/api/trpc/[trpc].ts
This is the entry point for your API and exposes the tRPC router. Normally, you won’t touch this file very much, but if you need to, for example, enable CORS middleware or similar, it’s useful to know that the exported
createNextApiHandler
createNextApiHandler
đź“„ server/api/trpc.ts
server/api/trpc.ts
This file is split up in two parts, context creation and tRPC initialization:
- We define the context that is passed to your tRPC procedures. Context is data that all of your tRPC procedures will have access to, and is a great place to put things like database connections, authentication information, etc. In create-t3-app we use two functions, to enable using a subset of the context when we do not have access to the request object.
-
: This is where you define context which doesn’t depend on the request, e.g. your database connection. You can use this function for integration testing or ssg-helpers↗ where you don’t have a request object.
createInnerTRPCContext
-
: This is where you define context which depends on the request, e.g. the user’s session. You request the session using the
createTRPCContext
object, and then pass the session down to theopts.req
function to create the final context.createInnerTRPCContext
- We initialize tRPC and define reusable procedures↗ and middlewares↗. By convention, you shouldn’t export the entire -object but instead, create reusable procedures and middlewares and export those.
t
You’ll notice we use
superjson
Date
Date
đź“„ server/api/routers/*.ts
server/api/routers/*.ts
This is where you define the routes and procedures of your API. By convention, you create separate routers↗ for related procedures.
đź“„ server/api/root.ts
server/api/root.ts
Here we merge↗ all the sub-routers defined in
routers/**
đź“„ utils/api.ts
utils/api.ts
This is the frontend entry point for tRPC. This is where you’ll import the router’s type definition and create your tRPC client along with the react-query hooks. Since we enabled
superjson
You’ll define your tRPC links↗ here, which determines the request flow from the client to the server. We use the “default” httpBatchLink
loggerLink
Lastly, we export a helper type↗ which you can use to infer your types on the frontend.
Create T3 App contributor Christopher Ehrlich↗ made a video about data flows in tRPC↗. This video is recommended if you have used tRPC but still feel a bit unclear about how it works.
How do I call my API externally?
With regular APIs, you can call your endpoints using any HTTP client such as
curl
Postman
fetch
Expose a single procedure externally
If you want to expose a single procedure externally, you’re looking for server side calls↗. That would allow you to create a normal Next.js API endpoint, but reuse the resolver part of your tRPC procedure.
import { type NextApiRequest, type NextApiResponse } from "next";
import { appRouter, createCaller } from "../../../server/api/root";
import { createTRPCContext } from "../../../server/api/trpc";
const userByIdHandler = async (req: NextApiRequest, res: NextApiResponse) => {
// Create context and caller
const ctx = await createTRPCContext({ req, res });
const caller = createCaller(ctx);
try {
const { id } = req.query;
const user = await caller.user.getById(id);
res.status(200).json(user);
} catch (cause) {
if (cause instanceof TRPCError) {
// An error from tRPC occurred
const httpCode = getHTTPStatusCodeFromError(cause);
return res.status(httpCode).json(cause);
}
// Another error occurred
console.error(cause);
res.status(500).json({ message: "Internal server error" });
}
};
export default userByIdHandler;
Exposing every procedure as a REST endpoint
If you want to expose every single procedure externally, checkout the community built plugin trpc-openapi↗. By providing some extra meta-data to your procedures, you can generate an OpenAPI compliant REST API from your tRPC router.
It’s just HTTP Requests
tRPC communicates over HTTP, so it is also possible to call your tRPC procedures using “regular” HTTP requests. However, the syntax can be cumbersome due to the RPC protocol↗ that tRPC uses. If you’re curious, you can check what tRPC requests and responses look like in your browser’s network tab, but we suggest doing this only as an educational exercise and sticking to one of the solutions outlined above.
Comparison to a Next.js API endpoint
Let’s compare a Next.js API endpoint to a tRPC procedure. Let’s say we want to fetch a user object from our database and return it to the frontend. We could write a Next.js API endpoint like this:
import { type NextApiRequest, type NextApiResponse } from "next";
import { prisma } from "../../../server/db";
const userByIdHandler = async (req: NextApiRequest, res: NextApiResponse) => {
if (req.method !== "GET") {
return res.status(405).end();
}
const { id } = req.query;
if (!id || typeof id !== "string") {
return res.status(400).json({ error: "Invalid id" });
}
const examples = await prisma.example.findFirst({
where: {
id,
},
});
res.status(200).json(examples);
};
export default userByIdHandler;
import { useState, useEffect } from "react";
import { useRouter } from "next/router";
const UserPage = () => {
const router = useRouter();
const { id } = router.query;
const [user, setUser] = useState(null);
useEffect(() => {
fetch(`/api/user/${id}`)
.then((res) => res.json())
.then((data) => setUser(data));
}, [id]);
};
Compare this to the tRPC example above and you can see some of the advantages of tRPC:
- Instead of specifying a url for each route, which can become annoying to debug if you move something, your entire router is an object with autocomplete.
- You don’t need to validate which HTTP method was used.
- You don’t need to validate that the request query or body contains the correct data in the procedure, because Zod takes care of this.
- Instead of creating a response, you can throw errors and return a value or object as you would in any other TypeScript function.
- Calling the procedure on the frontend provides autocompletion and type safety.
Useful snippets
Here are some snippets that might come in handy.
Enabling CORS
If you need to consume your API from a different domain, for example in a monorepo that includes a React Native app, you might need to enable CORS:
import { type NextApiRequest, type NextApiResponse } from "next";
import { createNextApiHandler } from "@trpc/server/adapters/next";
import { appRouter } from "~/server/api/root";
import { createTRPCContext } from "~/server/api/trpc";
import cors from "nextjs-cors";
const handler = async (req: NextApiRequest, res: NextApiResponse) => {
// Enable cors
await cors(req, res);
// Create and call the tRPC handler
return createNextApiHandler({
router: appRouter,
createContext: createTRPCContext,
})(req, res);
};
export default handler;
Optimistic updates
Optimistic updates are when we update the UI before the API call has finished. This gives the user a better experience because they don’t have to wait for the API call to finish before the UI reflects the result of their action. However, apps that value data correctness highly should avoid optimistic updates as they are not a “true” representation of backend state. You can read more on the React Query docs↗.
const MyComponent = () => {
const listPostQuery = api.post.list.useQuery();
const utils = api.useUtils();
const postCreate = api.post.create.useMutation({
async onMutate(newPost) {
// Cancel outgoing fetches (so they don't overwrite our optimistic update)
await utils.post.list.cancel();
// Get the data from the queryCache
const prevData = utils.post.list.getData();
// Optimistically update the data with our new post
utils.post.list.setData(undefined, (old) => [...old, newPost]);
// Return the previous data so we can revert if something goes wrong
return { prevData };
},
onError(err, newPost, ctx) {
// If the mutation fails, use the context-value from onMutate
utils.post.list.setData(undefined, ctx.prevData);
},
onSettled() {
// Sync with server once mutation has settled
utils.post.list.invalidate();
},
});
};
Sample Integration Test
Here is a sample integration test that uses Vitest↗ to check that your tRPC router is working as expected, the input parser infers the correct type, and that the returned data matches the expected output.
import { type inferProcedureInput } from "@trpc/server";
import { expect, test } from "vitest";
import { appRouter, type AppRouter } from "~/server/api/root";
import { createInnerTRPCContext } from "~/server/api/trpc";
test("example router", async () => {
const ctx = await createInnerTRPCContext({ session: null });
const caller = appRouter.createCaller(ctx);
type Input = inferProcedureInput<AppRouter["example"]["hello"]>;
const input: Input = {
text: "test",
};
const example = await caller.example.hello(input);
expect(example).toMatchObject({ greeting: "Hello test" });
});
If your procedure is protected, you can pass in a mocked
session
test("protected example router", async () => {
const ctx = await createInnerTRPCContext({
session: {
user: { id: "123", name: "John Doe" },
expires: "1",
},
});
const caller = appRouter.createCaller(ctx);
// ...
});
Useful Resources
Resource | Link |
---|---|
tRPC Docs | https://www.trpc.io↗ |
Bunch of tRPC Examples | https://github.com/trpc/trpc/tree/next/examples↗ |
React Query Docs | https://tanstack.com/query/latest/docs↗ |