Nexpresst is a lightweight TypeScript utility designed to build Express-like API routes in Next.js applications. It leverages the Next.js App Router's file-based routing system, providing a structured way to handle HTTP methods, middleware, and response processing—all with strong TypeScript support.
- Express-like Routing: Use familiar patterns from Express to create API routes in Next.js.
- Express Middleware Adapter: You can leverage existing Express-compatible middleware like
helmet
,compression
,csurf
andcors
in your Next.js API routes with theexpressMiddlewareAdapter
. - Custom Middleware Support: Define global and route-specific middleware for fine-grained request handling.
- Strong TypeScript Support: Utilize TypeScript generics for type-safe request handlers and middleware, ensuring robust and predictable API interactions.
- Easy Integration: Seamlessly integrate with Next.js's App Router and next/server module for a smooth development experience.
To install nexpresst
in your Next.js project, run:
npm install nexpresst
Start by creating a function dynamically generating an apiRouter instance in your Next.js application. This function will serve as the central point for managing routes and applying global middleware.
// @/lib/api-router.ts
import { NextRequest } from 'next/server';
import { ApiRouter, TNextContext } from 'nexpresst';
export const apiRouter = (req: NextRequest, ctx: TNextContext) => new ApiRouter(req, ctx);
You can optionally add global middleware using the .use()
method.
Since Next.js does not parse request bodies out of the box, nexpresst
provides ready-to-use middleware to handle such scenarios.
import { NextRequest } from 'next/server';
import { ApiRouter, TNextContext, queryParser, jsonParser } from 'nexpresst';
export const apiRouter = (req: NextRequest, ctx: TNextContext) =>
new ApiRouter(req, ctx)
.use(queryParser) // Appends a query object to the request, accessible via `req.query`
.use(jsonParser); // Parses the request body as JSON, accessible via `req.payload`
You can use your custom implementations if you prefer, but these are solid starters to get the job done initially. 😎
Define route handlers using the IRouteHandler
interface.
To use these handlers, pass them to the handle()
method of the apiRouter
instance inside the Next.js HTTP method function by directly returning the instance.
🔺 Note: You must still export a function with a valid HTTP method name such as GET
, POST
etc. from your route.ts
file. Because it is a strict requirement by Next.js.
Example: Handling Requests
// @/app/api/posts/route.ts
import { apiRouter } from '@/lib/api-router';
import { NextRequest } from 'next/server';
import { IRouteHandler, TNextContext } from 'nexpresst';
// Define a GET handler
const getPostsHandler: IRouteHandler = async (req, res) => {
// Your logic here
// return res.send({ message: 'Hello from posts' }) // statusCode defaults to 200
// or set it explicitly
return res.statusCode(200).send({ message: 'Hello from posts' });
};
// Export GET function for Next.js routing
export function GET(req: NextRequest, ctx: TNextContext) {
return apiRouter(req, ctx).handle(getPostsHandler);
}
You can handle all other HTTP methods with the above syntax.
A powerful feature in nexpresst
is the expressMiddlewareAdapter
, which allows you to use popular Express-compatible middleware in your Next.js routes. This opens up a world of existing middleware solutions from the Express ecosystem.
import { NextRequest } from 'next/server';
import helmet from 'helmet';
import cors from 'cors';
import compression from 'compression';
import { ApiRouter, TNextContext, expressMiddlewareAdapter } from 'nexpresst';
export const apiRouter = (req: NextRequest, ctx: TNextContext) =>
new ApiRouter(req, ctx)
.use(expressMiddlewareAdapter(compression())) // Using compression middleware
.use(expressMiddlewareAdapter(cors({ origin: 'http://localhost:3000' }))) // Adding CORS middleware
.use(expressMiddlewareAdapter(helmet())); // Adding helmet for security headers
Use the IMiddlewareHandler
interface to create custom middleware. Here, we'll create a basic logger
middleware.
// @/lib/middlewares/logger.ts
import { IMiddlewareHandler } from 'nexpresst';
export const logger: IMiddlewareHandler = async (req, res, next) => {
console.log(`${req.method} -- ${req.url}`);
return next();
};
You can create custom global middleware and register it with your global router instance to apply it to all incoming requests. Alternatively, middleware can be applied on a per-route basis.
Example: Global Usage
// @/lib/api-router.ts
export const apiRouter = (req: NextRequest, ctx: TNextContext) =>
new ApiRouter(req, ctx).use(logger).use(otherMiddleware);
// Alternatively, you can use the following syntax for registering middleware
export const apiRouter = (req: NextRequest, ctx: TNextContext) =>
new ApiRouter(req, ctx).use(logger, otherMiddleware);
These two middleware will be applied to all incoming requests.
Example: Route-specific Usage
// @/app/api/posts/route.ts
const getPostsHandler: IRouteHandler = async (req, res) => {
return res.statusCode(200).send({ message: 'Hello from posts' });
};
export function GET(req: NextRequest, ctx: TNextContext) {
return apiRouter(req, ctx).use(logger).use(otherMiddleware).handle(getPostsHandler);
// Alternatively, you can use the following syntax for registering middleware
return apiRouter(req, ctx).use(logger, otherMiddleware).handle(getPostsHandler);
}
These two middleware will only be applied to this specific route.
ℹ️ Note that you can always create multiple instances of the ApiRouter class with different configurations, allowing you to register each instance with different middleware for more fine-tuned control.
// @/lib/api-router.ts
export const protectedRouter = (req: NextRequest, ctx: TNextContext) =>
new ApiRouter(req, ctx).use(logger).use(protect);
export const publicRouter = (req: NextRequest, ctx: TNextContext) =>
new ApiRouter(req, ctx).use(logger);
Then, use the corresponding router instance in your relevant routes as follows:
// @/app/api/users/posts/route.ts
import { protectedRouter } from '@/lib/api-router';
// Users are only allowed to see their own posts
export function GET(req: NextRequest, ctx: TNextContext) {
return protectedRouter(req, ctx).handle(someProtectedHandler);
}
// @/app/api/posts/route.ts
import { publicRouter } from '@/lib/api-router';
// Everyone can see the posts
export function GET(req: NextRequest, ctx: TNextContext) {
return publicRouter(req, ctx).handle(someProtectedHandler);
}
This approach ensures that different routes are handled according to their specific middleware requirements.
Nexpresst leverages TypeScript to provide strong typing for both middleware and route handlers.
The IRouteHandler
interface allows you to define the types for path parameters, query parameters, request payloads, response payloads, and session objects to ensure type safety.
// @/app/api/posts/route.ts
import { IRouteHandler } from 'nexpresst';
const example: IRouteHandler<
{ id: string }, // Path parameters (e.g., /posts/:id)
{ search: string }, // Query parameters (e.g., /posts?search=term)
{ title: string }, // Request payload (e.g., { title: "New Post" })
{ message: string } // Response payload (e.g., { message: "Success" })
{ user: object } // Request session, if any
> = (req, res, next) => {
const { id } = req.params;
const { search } = req.query;
const { title } = req.payload;
const { user } = req.session;
// Your handler logic here
return res.statusCode(200).send({ message: `Post ${id} updated with title: ${title}` });
};
The IMiddlewareHandler
interface allows you to define the types for path parameters, query parameters, request payload, response payloads and session objects to ensure type safety.
// @/lib/middlewares/example.ts
import { IMiddlewareHandler } from 'nexpresst';
const example: IMiddlewareHandler<
{ id: string }, // Path parameters (e.g., /posts/:id)
{ search: string }, // Query parameters (e.g., /posts?search=term)
{ title: string } // Request payload (e.g., { title: "New Post" })
unknown // Response payload (e.g., { message: "Success" })
{ user: object } // Request session, if any
> = (req, res, next) => {
const { id } = req.params;
const { search } = req.query;
const { title } = req.payload;
const { user } = req.session;
/**
* If you passed a response payload type, you can return a response satisfying this type.
* Otherwise, you can call the next function to proceed.
*/
// your middleware logic here
return next();
};
You can optionally register an onError
middleware with global router to handle errors gracefully.
// @/lib/middlewares/error-handler.ts
import { IMiddlewareHandler } from 'nexpresst';
type TErrorResponse = { name: string; message: string };
// Example error handler middleware
const errorHandler: IMiddlewareHandler<unknown, unknown, unknown, TErrorResponse> = (
req,
res,
next,
) => {
return next().catch((err: unknown) => {
/**
* This is just a simple demonstration of how to handle errors.
* Add your custom error logging and response handling logic here.
*/
if (err instanceof Error) {
return res.statusCode(500).send({ name: err.name, message: err.message });
}
return res
.statusCode(500)
.send({ name: 'INTERNAL_SERVER_ERROR', message: 'Something went wrong' });
});
};
And then in your api-router.ts
file:
// @/lib/api-router.ts
import { errorHandler } from '@/lib/middlewares';
export const apiRouter = (req: NextRequest, ctx: TNextContext) =>
new ApiRouter(req, ctx)
.onError(errorHandler) // Register errorHandler middleware with your global router instance
.use(middleware, anotherMiddleware); // Add other middlewares
In Next.js, the file-based routing system automatically provides a default error page for requests made to non-existent endpoints. However, in modern REST APIs, relying on a generic 404
page isn't ideal.
Nexpresst offers a more flexible solution that allows you to handle 404 errors in a custom and developer-friendly way.
To address this issue, start by creating a catch-all segment at the root of your api
folder:
📦app
┣ 📂api
┃ ┣ 📂[[...params]] <--------- Catch-all route
┃ ┃ ┗ 📜route.ts
┃ ┗ 📂posts
┃ ┃ ┣ 📜route.ts
┣ 📜favicon.ico
┣ 📜globals.css
┣ 📜layout.tsx
┗ 📜page.tsx
In the [[...params]]/route.ts
file, add the following code:
import { apiRouter } from '@/lib/api-router';
import { exportAllHttpMethods, IRouteHandler } from 'nexpresst';
const notFoundHandler: IRouteHandler = async (req, res) => {
console.log(req.params); // // Access to params passed as a string[]
// Define your custom 404 logic here
return res.statusCode(404).end();
};
export const { GET, POST, PUT, DELETE, PATCH, HEAD } = exportAllHttpMethods(
apiRouter,
notFoundHandler,
);
With this setup, any requests to non-existent API routes will trigger the notFoundHandler
, allowing you to customize the 404
response according to your specific requirements.
For a full example, check out the GitHub repository with a complete implementation.
Contributions are welcome! If you find a bug or have a feature request, please open an issue.
To contribute to this project, follow these steps:
- Fork this repository.
- Create a new branch (
git checkout -b feature/<some-feature>
). - Make your changes.
- Commit your changes (
git commit -m "feat: add some feature"
). - Push to the branch (
git push origin feature-branch
). - Open a pull request.
This project is licensed under the MIT License. See the LICENSE file for details.