rembrembdocs

Writing all API-code in the same file can become unwieldy. It's easy to merge routers together in order to break them up.

Merging with child routers

routers/user.ts

ts

import { router, publicProcedure } from '../trpc';

import { z } from 'zod';

export const userRouter = router({

list: publicProcedure.query(() => {

`// [..]`

`return [];`

}),

});

routers/post.ts

ts

import { router, publicProcedure } from '../trpc';

import { z } from 'zod';

export const postRouter = router({

create: publicProcedure

`.input(`

  `z.object({`

    `title: z.string(),`

  `}),`

`)`

`.mutation((opts) => {`

  `const { input } = opts;`

  `// [...]`

`}),`

list: publicProcedure.query(() => {

`// ...`

`return [];`

}),

});

routers/_app.ts

ts

import { router } from '../trpc';

import { userRouter } from './user';

import { postRouter } from './post';

const appRouter = router({

user: userRouter,

post: postRouter,

});

appRouter.user

       `(property) user: DecorateCreateRouterOptions<{     list: QueryProcedure<{         input: void;         output: never[];         meta: object;     }>; }>`

appRouter.post

       `(property) post: DecorateCreateRouterOptions<{     create: MutationProcedure<{         input: {             title: string;         };         output: void;         meta: object;     }>;     list: QueryProcedure<{         input: void;         output: never[];         meta: object;     }>; }>`

export type AppRouter = typeof appRouter;

Merging with t.mergeRouters

If you prefer having all procedures flat in one single namespace, you can instead use t.mergeRouters

routers/user.ts

ts

import { router, publicProcedure } from '../trpc';

import { z } from 'zod';

export const userRouter = router({

userList: publicProcedure.query(() => {

`// [..]`

`return [];`

}),

});

routers/post.ts

ts

import { router, publicProcedure } from '../trpc';

import { z } from 'zod';

export const postRouter = router({

postCreate: publicProcedure

`.input(`

  `z.object({`

    `title: z.string(),`

  `}),`

`)`

`.mutation((opts) => {`

  `const { input } = opts;`

  `// [...]`

`}),`

postList: publicProcedure.query(() => {

`// ...`

`return [];`

}),

});

routers/_app.ts

ts

import { mergeRouters } from '../trpc';

import { userRouter } from './user';

import { postRouter } from './post';

const appRouter = mergeRouters(userRouter, postRouter);

     `const appRouter: BuiltRouter<{     ctx: object;     meta: object;     errorShape: DefaultErrorShape;     transformer: false; }, DecorateCreateRouterOptions<{     postCreate: MutationProcedure<{         input: {             title: string;         };         output: void;         meta: object;     }>;     postList: QueryProcedure<{         input: void;         output: never[];         meta: object;     }>; }> & DecorateCreateRouterOptions<{     userList: QueryProcedure<{         input: void;         output: never[];         meta: object;     }>; }>>`

export type AppRouter = typeof appRouter;

Dynamically load routers

You can use the lazy function to dynamically load your routers. This can be useful to reduce cold starts of your application. There's no difference in how you use the router after it's been lazy loaded vs. how you use a normal router.

routers/greeting.ts

ts

import { router, publicProcedure } from '../trpc';

export const greetingRouter = router({

hello: publicProcedure.query(() => 'world'),

});

routers/user.ts

ts

import { router, publicProcedure } from '../trpc';

export const userRouter = router({

list: publicProcedure.query(() => ['John', 'Jane', 'Jim']),

});

routers/_app.ts

ts

import { lazy } from '@trpc/server';

import { router } from '../trpc';

export const appRouter = router({

// Option 1: Short-hand when the module has exactly 1 router exported

greeting: lazy(() => import('./greeting.js')),

// Option 2: if exporting more than 1 router

user: lazy(() => import('./user.js').then((m) => m.userRouter)),

});

export type AppRouter = typeof appRouter;