next-zero-rpc
Type-safe fetch for Next.js API routes — zero dependencies, zero lock-in, ~0.9KB runtime.
npx next-zero-rpc initThat's it. Four files. Full type safety. ~0.9 KB runtime.
What it does
next-zero-rpc gives you compile-time type-safe fetch for your Next.js App Router API routes — without changing how you write backends.
// ✅ Full autocomplete on paths, methods, and response types
const [data, err] = await apiFetch("/api/users/123", { method: "GET" });
// ✅ TypeScript errors on invalid paths
const [data, err] = await apiFetch("/api/typo", { method: "GET" }); // ← compile error
// ✅ TypeScript errors on invalid methods
const [data, err] = await apiFetch("/api/users/123", { method: "DELETE" }); // ← error if DELETE not exported
// ✅ Multi-variable template literals — all segments resolve independently, still fully type-safe
const orgId = "org-1";
const projectId = "proj-42";
const taskId = "task-99";
await apiFetch(`/api/orgs/${orgId}/projects/${projectId}/tasks/${taskId}`, { method: "PATCH" });
// ✅ Error type narrowing — err.code autocompletes only the errors THIS route can return
if (err) {
const code = err.code;
switch (code) {
case "auth:forbidden": // ← only if this route uses createApiError("auth:forbidden", ...)
case "system:database-error": // ← only if this route uses createApiError("system:database-error", ...)
case "system:unknown-error": // ← always included as a fallback from apiFetch itself
break;
default:
assertNever(code); // ← TypeScript errors if you miss a case
}
}The 2026 Ecosystem Comparison
| Feature | next-zero-rpc | tRPC | ts-rest | raw fetch |
|---|---|---|---|---|
| Primary Philosophy | Invisible type bridge | End-to-end framework | Contract-first API | Platform standard |
| Source of Truth | Next.js Route Handlers | tRPC Routers / Procedures | Shared contract.ts file | None |
| Type-safe paths & responses | ||||
| Per-route error narrowing | (via TypeScript generics) | (Global error shapes) | (Standardized HTTP errors) | |
| Next.js App Router Native | (Zero changes to standard handlers) | (Requires tRPC adapters) | (Requires ts-rest adapters) | |
| Input Validation | Bring-your-own | Built-in (Zod heavily favored) | Built-in (Zod favored) | Bring-your-own |
| OpenAPI Generation | (Requires third-party plugins) | (First-class citizen) | ||
| Client Runtime Size | ~0.9 KB | ~15 KB | ~3-5 KB | 0 KB |
| Server Actions Integration | (Tuple-based Go-style returns) | (Excellent RSC/Action support) | Partial (Focus remains on REST) | N/A |
| Non-TS Client Support | (Via standard REST/OpenAPI) | |||
| Ecosystem & Community | Niche / Lightweight | Massive / Enterprise-grade | Very Strong / Standardized | Ubiquitous |
Architectural Breakdown: Which to Choose?
1. next-zero-rpc (The Minimalist Bridge)
- Best for: Teams deeply invested in the Next.js App Router who want type safety without adopting a new framework paradigm.
- The draw: You write standard
export async function GET(req)handlers using two simple response helpers. The library just quietly infers what you wrote. If you ever decide to remove the code generator, your backend routes still run perfectly as standard Next.js endpoints. - The trade-off: You give up the robust middleware pipelines, batched requests, and automatic OpenAPI generation that larger ecosystems provide.
When to use this
Use next-zero-rpc when:
- You're already writing plain Next.js App Router route handlers and want type-safe
fetchcalls without restructuring your backend into tRPC procedures or ts-rest contracts - Per-route error code narrowing matters to you — this is genuinely not available in tRPC or ts-rest out of the box
- You want a tiny client footprint (~0.9 KB) and zero ongoing npm dependencies
The Philosophy
You own the code, not the library. next-zero-rpc is not a locked-in framework—it's a philosophy, a paradigm, and a set of methods for doing things. When you run init, we give you four files. From that moment on, they are yours to modify, extend, or delete.
- Zero vendor lock-in — There is no black-box
node_modulesdependency dictating your architecture. You own the fetch client, the error codes, and the registry generator. If the maintainer disappears tomorrow, you're not blocked. - Zero boilerplate — You write standard Next.js API route handlers using simple response helpers — no decorators, no schema registrations, no complex abstractions. The codegen reads what already exists and builds the type bridge automatically.
- Not a framework — It's a type bridge. It infers what your route handlers return and gives your client code full type safety over those responses.
- Validation is yours — Input validation (Zod, Valibot, Arktype, manual checks) stays inside your route handler where it belongs. This library doesn't impose a validation layer — that's a deliberate design choice.
- Non-invasive — Unlike tRPC or ts-rest, you don't adopt a new API definition pattern. Your routes are regular Next.js routes. The library is invisible.
- Architectural freedom — Want to use Server-Sent Events (SSE), WebSockets, GraphQL, or custom streaming patterns alongside it? Go ahead.
next-zero-rpcjust drops four plain files (apiClient.ts,apiRegistry.ts,responses.ts,update-api-registry.mjs) into your project; it doesn't take over your Next.js server. Study the generated files yourself to verify this claim—there's no hidden magic.
Setup
1. Install
npx next-zero-rpc initThis copies 4 files into lib/next-zero-rpc/ (or src/lib/next-zero-rpc/ if your project uses the src/ directory):
| File | Purpose | Client bundle | Server bundle |
|---|---|---|---|
apiClient.ts |
Type-safe fetch wrapper | (~0.7 KB minified) | |
apiRegistry.ts |
Auto-generated route type registry | (types only) | (types only) |
responses.ts |
Error/success helpers + error codes | (~0.2 KB minified) | (server helpers) |
update-api-registry.mjs |
Code generator + Next.js plugin | (dev only) |
* From responses.ts: Client gets isApiErrorPayload type guard (~50 bytes) and assertNever helper (~50 bytes). Server gets createApiError, createApiSuccess, createServiceError, createServiceSuccess, and HTTP status constants. ERROR_CODES are types only and erased at runtime, though some bundlers may include them until production tree-shaking removes them. Total client runtime: ~0.9 KB minified.
The CLI also:
- Runs the registry generator once to detect existing routes
- Adds an
"infer-api"script to yourpackage.jsonfor manual regeneration
2. Add the plugin to next.config.ts
// If using src/ directory:
import { withApiRegistry } from "./src/lib/next-zero-rpc/update-api-registry.mjs";
// If NOT using src/ directory:
import { withApiRegistry } from "./lib/next-zero-rpc/update-api-registry.mjs";
const nextConfig = {};
export default withApiRegistry(nextConfig);The plugin auto-updates apiRegistry.ts whenever you create, modify, or delete route.ts files during development. In production builds, it runs once and skips the file watcher.
3. Use it
Route handlers — createApiSuccess / createApiError
// app/api/users/[userId]/route.ts
import {
createApiSuccess,
createApiError,
HTTP_STATUS_ERROR,
HTTP_STATUS_SUCCESS,
} from "@/lib/next-zero-rpc/responses";
export async function GET(req: Request, { params }: { params: Promise<{ userId: string }> }) {
const { userId } = await params;
const user = await db.users.find(userId);
if (!user) {
return createApiError("resource:not-found", HTTP_STATUS_ERROR.NOT_FOUND);
}
return createApiSuccess({ id: user.id, name: user.name });
}
export async function DELETE(req: Request, { params }: { params: Promise<{ userId: string }> }) {
const { userId } = await params;
await db.users.delete(userId);
// 204 No Content — no body per HTTP spec
return createApiSuccess();
}Always
exportyour HTTP method handlers (GET,POST,PUT,PATCH,DELETE). The registry generator checks for the presence ofexportin the file content — any route file without exports is skipped and will not appear inKnownRoutesor receive type safety.
Client components — apiFetch
"use client";
import { apiFetch } from "@/lib/next-zero-rpc/apiClient";
const [data, err] = await apiFetch("/api/users/123", { method: "GET" });
if (err) {
console.error(err.code, err.message); // err.code is narrowed to only this route's errors
} else {
console.log(data.name); // ← fully typed, payload returned directly
}Features
Error Type Narrowing
The standout feature: when you check err.code after an apiFetch call, TypeScript narrows the union to only the error codes that specific route handler can actually return — not every error code in the system.
// Route handler returns createApiError("auth:unauthorized", 401)
// or createApiError("validation:missing-required-fields", 400)
const [data, err] = await apiFetch("/api/auth/login", { method: "POST" });
if (err) {
// err.code is: "auth:unauthorized" | "validation:missing-required-fields" | "validation:invalid-payload" | "system:unknown-error"
// NOT the full 37+ error code union — only what this route can produce
// (system:unknown-error is always included as a fallback from apiFetch itself)
}This works because:
createApiErroris generic:createApiError<C extends ErrorCode>(code: C, ...) → NextResponse<ApiErrorPayload<C>>- TypeScript infers the literal
Cfrom each call site in your handler UnwrapNextResponseextracts the union of allApiErrorPayload<C>types from the handler's return type- The client sees only those specific error codes
Go-Style Tuple Returns
Every apiFetch call returns [data, null] on success or [null, error] on failure — never throws.
const [data, err] = await apiFetch("/api/orders/checkout", {
method: "POST",
body: JSON.stringify({ ticketId: "abc", quantity: 2 }),
});
if (err) {
// Handle error — err is fully typed with route-specific error codes
return;
}
// data is fully typed — no casting needed
console.log(data.orderId, data.status);Exhaustive Error Checking
Use assertNever to guarantee you handle every possible error code at compile time:
import { assertNever } from "@/lib/next-zero-rpc/responses";
const [data, err] = await apiFetch("/api/users/123", { method: "GET" });
if (err) {
const code = err.code;
switch (code) {
case "system:database-error":
showToast("Database error, please try again");
break;
case "system:unknown-error":
showToast("Something went wrong");
break;
default:
assertNever(code); // ← TypeScript errors if you miss a case
}
return;
}Dynamic Route Matching
The type system supports Next.js dynamic segments and catch-all routes:
// Dynamic segments: [id], [userId], [slug]
const [data, err] = await apiFetch("/api/users/abc-123", { method: "GET" });
// Catch-all segments: [...catchall]
const [data, err] = await apiFetch("/api/extreme/org1/projects/proj1/tasks/a/b/c", {
method: "POST",
});
// Optional catch-all segments: [[...slug]] (zero or more segments)
const [data, err] = await apiFetch("/api/docs", { method: "GET" }); // zero segments
const [data, err] = await apiFetch("/api/docs/getting-started/intro", { method: "GET" }); // multiple segments
// Query strings are stripped before matching
const [data, err] = await apiFetch("/api/users/123?include=profile", { method: "GET" });Static vs Dynamic Route Precedence
If you have overlapping static and dynamic routes (e.g., /api/users/active and /api/users/[userId]), next-zero-rpc correctly gives exact static matches precedence over dynamic segments at compile time.
// Safely infers the type of the `active` route, completely ignoring the `[userId]` route
const [activeUsers] = await apiFetch("/api/users/active", { method: "GET" });
// Safely infers the type of the `[userId]` route
const [singleUser] = await apiFetch("/api/users/123", { method: "GET" });Route Groups Support
Next.js route groups like (groupName) are natively supported. They are automatically ignored in the URL path mapping and the generated TypeScript types, perfectly matching Next.js behavior:
// File: app/api/(admin)/users/route.ts
// The fetch URL correctly skips the (admin) group
const [data, err] = await apiFetch("/api/users", { method: "GET" });Conflict-safe: You don't have to worry about multiple route groups accidentally mapping to the exact same generated TypeScript identifier. Next.js enforces uniqueness during development and at build time. It will automatically raise a
Conflicting routes at /api/...error if you try to create two identical endpoints (e.g.,(admin)/users/route.tsand(public)/users/route.ts).
HTTP Method Validation
TypeScript only allows methods that your route handler actually exports:
// If route.ts only exports GET and POST:
await apiFetch("/api/auth/login", { method: "POST" }); // ✅
await apiFetch("/api/auth/login", { method: "GET" }); // ✅ (if exported)
await apiFetch("/api/auth/login", { method: "DELETE" }); // ❌ compile errorFunction Overloading on createApiSuccess
createApiSuccess uses TypeScript function overloading for precise return types:
// With data → NextResponse<T>
return createApiSuccess({ id: "123", name: "John" });
// Without data (204) → NextResponse<undefined>
return createApiSuccess();
// Without data (explicit 200 OK) → NextResponse<undefined>
return createApiSuccess(undefined, HTTP_STATUS_SUCCESS.OK);
// The overload enforces: if statusCode is NO_CONTENT, data must be undefined
return createApiSuccess({ id: "123" }, HTTP_STATUS_SUCCESS.NO_CONTENT); // ← type errorSafely Handling Empty Responses
The apiClient.ts uses an incredibly robust approach to handle empty API responses, leveraging how Next.js and JavaScript parse JSON primitives.
If a route needs to return a 204 No Content or an empty 200 OK, createApiSuccess(undefined) intelligently evaluates to new NextResponse(null) to completely strip the HTTP body, avoiding Next.js Response.json(undefined) serialization crashes.
On the client side, apiFetch reads res.text() before attempting to parse JSON. Because valid JSON primitives like 0, null, false, or "" serialize into strings with a length > 0 (e.g. "0", '""'), they evaluate to truthy and are safely passed into JSON.parse. Only a genuinely empty HTTP body resolves to an empty string (""), which evaluates to falsy, allowing the client to safely resolve payload = undefined without throwing a JSON syntax error.
API Reference
responses.ts
Error Codes
Error codes follow a prefix:description convention enforced by the PrefixedError<T> template literal type. The built-in categories are:
| Category | Prefix | Examples |
|---|---|---|
| System | system: |
system:internal-server-error, system:database-error |
| Authentication | auth: |
auth:unauthorized, auth:token-expired |
| Validation | validation: |
validation:invalid-payload, validation:duplicate-entry |
| Resource | resource: |
resource:not-found, resource:already-exists |
| Network | network: |
network:timeout, network:connection-refused |
| Upload | upload: |
upload:file-too-large, upload:quota-exceeded |
To add a custom category:
export const PAYMENT_ERRORS = [
"payment:card-declined",
"payment:insufficient-funds",
"payment:expired-card",
] as const satisfies PrefixedError<"payment">[];
// Then add to ERROR_CODES:
export const ERROR_CODES = [
...SYSTEM_ERRORS,
...AUTH_ERRORS,
...VALIDATION_ERRORS,
...RESOURCE_ERRORS,
...NETWORK_ERRORS,
...UPLOAD_ERRORS,
...PAYMENT_ERRORS, // ← add here
] as const;HTTP Status Constants
Pre-defined, typed HTTP status code objects to prevent magic numbers:
HTTP_STATUS_SUCCESS.OK; // 200
HTTP_STATUS_SUCCESS.CREATED; // 201
HTTP_STATUS_SUCCESS.ACCEPTED; // 202
HTTP_STATUS_SUCCESS.NO_CONTENT; // 204
HTTP_STATUS_ERROR.BAD_REQUEST; // 400
HTTP_STATUS_ERROR.UNAUTHORIZED; // 401
HTTP_STATUS_ERROR.FORBIDDEN; // 403
HTTP_STATUS_ERROR.NOT_FOUND; // 404
HTTP_STATUS_ERROR.METHOD_NOT_ALLOWED; // 405
HTTP_STATUS_ERROR.NOT_ACCEPTABLE; // 406
HTTP_STATUS_ERROR.CONFLICT; // 409
HTTP_STATUS_ERROR.PAYLOAD_TOO_LARGE; // 413
HTTP_STATUS_ERROR.UNPROCESSABLE_ENTITY; // 422
HTTP_STATUS_ERROR.TOO_MANY_REQUESTS; // 429
HTTP_STATUS_ERROR.INTERNAL_SERVER_ERROR; // 500
HTTP_STATUS_ERROR.BAD_GATEWAY; // 502
HTTP_STATUS_ERROR.SERVICE_UNAVAILABLE; // 503
HTTP_STATUS_ERROR.GATEWAY_TIMEOUT; // 504Types
| Type | Description |
|---|---|
ErrorCode |
Union of all error code string literals |
SuccessHttpStatusCode |
200 | 201 | 202 | 204 |
ErrorHttpStatusCode |
400 | 401 | 403 | ... | 504 |
ApiErrorPayload<C> |
{ code: C; details?: Record<string, string[]>; message?: string } — generic over the specific error code |
ServiceError<C> |
{ code: C; message: string; details?: ... } — generic over the specific error code |
ServiceResponse<S, E> |
[S, null] | [null, E] — Go-style tuple for server actions |
Functions
| Function | Signature | Description |
|---|---|---|
createApiError<C> |
(code: C, statusCode, options?) → NextResponse<ApiErrorPayload<C>> |
Create a typed API error response |
createApiSuccess<T> |
(data: T, statusCode?) → NextResponse<T> |
Create a typed API success response |
createApiSuccess |
(undefined, NO_CONTENT) → NextResponse<undefined> |
Overload for 204 No Content |
isApiErrorPayload |
(payload: unknown) → payload is ApiErrorPayload<ErrorCode> |
Runtime type guard for error payloads |
createServiceError<C> |
(code: C, options?) → [null, ServiceError<C>] |
Go-style error for server actions |
createServiceSuccess<T> |
(data?: T) → [T | undefined, null] |
Go-style success for server actions |
assertNever |
(value: never) → never |
Compile-time exhaustiveness guard |
apiClient.ts
apiFetch<Path, Method>(path, options)
Type-safe fetch wrapper that returns Go-style [data, error] tuples.
Type parameters:
Path— validated againstKnownRoutesat compile timeMethod— restricted to only the HTTP methods exported by the matched route
Returns: Promise<[SuccessType, null] | [null, ErrorType | ApiErrorPayload<"system:unknown-error">]>
- Success:
[data, null]wheredatais inferred from the route handler'screateApiSuccesscalls - Error:
[null, error]whereerror.codeis narrowed to only the error codes used in that route'screateApiErrorcalls, plus"system:unknown-error"as a fallback - 204 No Content:
[undefined, null]
Runtime behavior:
- Handles
204 No Content— returns[undefined, null]without parsing body - Parses JSON responses based on
Content-Typeheader - Falls back to text for non-JSON responses
- Validates errors via
isApiErrorPayloadruntime type guard - Catches network errors (offline, CORS) and wraps them as
system:unknown-error
apiRegistry.ts
Auto-generated file containing:
KnownRoutes— A type map from route path strings to theirtypeofmodule typesKnownRouteSegments— Pre-computed segment tuples for each route (codegen output; avoids per-routeSplit<K>work at type-check time)- Path matching types — Template literal types that resolve runtime paths (with dynamic segments) to their registry entries:
Split<S>— Splits the input path string into a tuple of segments (used once per lookup, not per route)FindBySegments<PathSegments>— Matches input segments against pre-computedKnownRouteSegmentsentriesMatchSegment<P, K>— Matches a runtime segment against a route pattern segment (supports[param])MatchSegments<P, K>— Recursively matches all segments (supports[...catchall]and[[...slug]])StripQuery<Path>— Strips query string before matchingFindMatchingRoute<Path>— Resolves a runtime path to itsKnownRouteskey (using an O(1) fast-path for static routes)CheckPath<Path>— Validates a path at compile time, falling back to autocomplete hints on mismatch
update-api-registry.mjs
Code generator and Next.js plugin:
updateApiRegistry()— Scansapp/api/forroute.tsfiles, generates type imports and theKnownRoutes/KnownRouteSegmentsmapswithApiRegistry(nextConfig)— Next.js plugin that runs the generator on startup and watches for changes in development mode- Automatically detects
src/vs root directory layout - Groups imports and type entries by top-level API directory
- De-duplicates watcher setup via
globalThis.__apiRegistryWatcherSetup - Debounces file system events (100ms) to prevent redundant regeneration
How it works
┌─────────────────────────────────────────────────────────────────┐
│ Build / Dev Time │
│ │
│ app/api/***/route.ts ──→ update-api-registry.mjs │
│ │ │
│ ▼ │
│ apiRegistry.ts │
│ (KnownRoutes + KnownRouteSegments maps) │
│ │ │
│ ▼ │
│ apiClient.ts ◄──── TypeScript infers paths, methods, │
│ success types, AND error types │
└─────────────────────────────────────────────────────────────────┘
┌───────────────────────────────────────────────────────────────────┐
│ Runtime (~0.9 KB) │
│ │
│ apiFetch("/api/users/123", { method: "GET" }) │
│ │ │
│ ▼ │
│ fetch(path, options) │
│ │ │
│ ├─ 204? → [undefined, null] │
│ ├─ JSON + ok? → [payload, null] │
│ ├─ JSON + !ok? → [null, ApiErrorPayload] (narrowed) │
│ ├─ non-JSON? → [text, null] │
│ └─ network fail? → [null, { code: "system:unknown-error" }] │
└───────────────────────────────────────────────────────────────────┘
- The
withApiRegistryplugin scans yourapp/api/directory forroute.tsfiles - It generates
import type * as ...statements inapiRegistry.tsmapping each route path to its module type - TypeScript infers the available methods and response types from each route's exports
createApiError<C>preserves the literal error codeCin the return type, enabling per-route error narrowingapiFetchuses these types to validate paths, methods, and infer both success AND error shapes at compile time- At runtime,
apiFetchis just a thinfetchwrapper with Go-style[data, error]returns
The registry auto-updates when you create, modify, or delete route files during development.
Server Actions / Service Layer
For server-side code (server actions, service functions), use the Go-style service helpers:
// services/user.ts
import { createServiceError, createServiceSuccess } from "@/lib/next-zero-rpc/responses";
export async function getUser(userId: string) {
const user = await db.users.find(userId);
if (!user) {
return createServiceError("resource:not-found", { message: "User not found" });
}
return createServiceSuccess({ id: user.id, name: user.name });
}
// Usage in a server action or component:
const [user, err] = await getUser("123");
if (err) {
console.error(err.code, err.message);
return;
}
console.log(user.name);Customization
Since you own all the source files, you can customize anything:
- Add error codes — Add new arrays in
responses.tswith thePrefixedError<"prefix">constraint - Add auth headers — Modify
apiFetchto inject tokens automatically - Add request body types — Extend the type inference in
apiClient.ts - Change the output path — Update
REGISTRY_FILEinupdate-api-registry.mjs - Override the generator — The
withApiRegistryplugin andupdateApiRegistry()function are fully yours to modify
Examples
Try it online
Opens examples/minimal — a self-contained Next.js app with a working apiFetch client and two example API routes.
Run locally
# Clone just the example folder (no git history)
npx degit caocchinh/next-zero-rpc/examples/minimal my-app
cd my-app
npm install
npm run devCLI Usage
npx next-zero-rpc # Install files (same as init)
npx next-zero-rpc init # Install files into your project
npx next-zero-rpc --force # Overwrite existing files
npx next-zero-rpc --help # Show helpRequirements
| Requirement | Minimum Version | Reason |
|---|---|---|
| Next.js | 14.0 | App Router (stable since 14.0) |
| TypeScript | 4.9 | satisfies keyword used in responses.ts |
| Node.js | 18 | Native fetch API required by apiFetch |
License
MIT