next-zod-route
A fork from next-safe-route that uses zod for schema validation.
next-zod-route
is a utility library for Next.js that provides type-safety and schema validation for Route Handlers/API Routes.
Features
- ✅ Schema Validation: Automatically validates request parameters, query strings, and body content with built-in error handling.
- 🧷 Type-Safe: Works with full TypeScript type safety for parameters, query strings, and body content.
- 😌 Easy to Use: Simple and intuitive API that makes defining route handlers a breeze.
- 🔄 Flexible Response Handling: Return Response objects directly or return plain objects that are automatically converted to JSON responses.
- 🧪 Fully Tested: Extensive test suite to ensure everything works reliably.
Installation
npm install next-zod-route zod
Or using your preferred package manager:
pnpm add next-zod-route zod
yarn add next-zod-route zod
Usage
// app/api/hello/route.ts
import { createZodRoute } from 'next-zod-route';
import { z } from 'zod';
const paramsSchema = z.object({
id: z.string(),
});
const querySchema = z.object({
search: z.string().optional(),
});
const bodySchema = z.object({
field: z.string(),
});
export const GET = createZodRoute()
.params(paramsSchema)
.query(querySchema)
.handler((request, context) => {
// Next.js passes params as a promise, but next-zod-route unwraps it for you
const { id } = context.params;
const { search } = context.query;
// Return a Response object directly
return Response.json({ id, search }), { status: 200 };
// Or return a plain object that will be converted to a JSON response
// return { id, search };
});
export const POST = createZodRoute()
.params(paramsSchema)
.query(querySchema)
.body(bodySchema)
.handler((request, context) => {
// Next.js 15 use promise, but with .params we already unwrap the promise for you
const { id } = context.params;
const { search } = context.query;
const { field } = context.body;
return Response.json({ id, search, field }), { status: 200 };
});
To define a route handler in Next.js:
- Import
createZodRoute
and zod
.
- Define validation schemas for params, query, and body as needed.
- Use
createZodRoute()
to create a route handler, chaining params
, query
, and body
methods.
- Implement your handler function, accessing validated and type-safe params, query, and body through
context
.
Supported Body Formats
next-zod-route
supports multiple request body formats out of the box:
- JSON: Automatically parses and validates JSON bodies.
- URL Encoded: Supports
application/x-www-form-urlencoded
data.
- Multipart Form Data: Supports
multipart/form-data
, enabling file uploads and complex form data parsing.
The library automatically detects the content type and parses the body accordingly. For GET and DELETE requests, body parsing is skipped.
Response Handling
You can return responses in two ways:
- Return a Response object directly:
return Response.json({ data: 'value' }, { status: 200 });
- Return a plain object that will be automatically converted to a JSON response with status 200:
return { data: 'value' };
Advanced Usage
Middleware
You can add middleware to your route handler with the use
method. Middleware functions can add data to the context that will be available in your handler.
const authMiddleware = async ({ request }) => {
// Get the token from the request headers
const token = request.headers.get('authorization')?.split(' ')[1];
// Validate the token and get the user
const user = await validateToken(token);
// Return the user to be added to the context
return { user };
};
const permissionsMiddleware = async () => {
return { permissions: ['read', 'write'] };
};
export const GET = createZodRoute()
.use(authMiddleware)
.use(permissionsMiddleware)
.handler((request, context) => {
// Access middleware data from context.data
const { user, permissions } = context.data;
return Response.json({ user, permissions });
});
Middleware functions should return an object. The returned object will be merged with the context's data property.
Custom Error Handler
You can specify a custom error handler function to handle errors thrown in your route handler:
import { createZodRoute } from 'next-zod-route';
// Create a custom error class
class CustomError extends Error {
constructor(
message: string,
public status: number = 400,
) {
super(message);
this.name = 'CustomError';
}
}
// Create a route with a custom error handler
const safeRoute = createZodRoute({
handleServerError: (error: Error) => {
if (error instanceof CustomError) {
return new Response(JSON.stringify({ message: error.message }), { status: error.status });
}
// Default error response
return new Response(JSON.stringify({ message: 'Internal server error' }), { status: 500 });
},
});
export const GET = safeRoute.handler((request, context) => {
// This error will be caught by the custom error handler
throw new CustomError('Something went wrong', 400);
});
By default, if no custom error handler is provided, the library will return a generic "Internal server error" message with a 500 status code to avoid information leakage.
Validation Errors
When validation fails, the library returns appropriate error responses:
- Invalid params:
{ message: 'Invalid params' }
with status 400
- Invalid query:
{ message: 'Invalid query' }
with status 400
- Invalid body:
{ message: 'Invalid body' }
with status 400
Tests
Tests are written using Vitest. To run the tests, use the following command:
pnpm test
Contributing
Contributions are welcome! For major changes, please open an issue first to discuss what you would like to change.
License
This project is licensed under the MIT License - see the LICENSE file for details.