next-smoothie
v0.4.0
Published
A compact, zero-dependency library that constructs Next.js 13+ App Route Handlers from decorated classes
Downloads
17
Readme
Quick start
Set up a regular Next.js project with App routerusing CLI and this instruction.
Install the library: npm i next-smoothie
or yarn add next-smoothie
.
Create the first controller:
// /src/controllers/UserController.ts
import { get, post, prefix } from 'next-smoothie';
import type { NextRequest } from 'next/server';
@prefix('users')
export default class UserController {
@get() // Handles GET requests to '/api/users'
static getHelloWorld() {
return { hello: 'world' };
}
@post('hello/:id/world') // Handles POST requests to '/api/users/hello/:id/world'
static postHelloWorld(req: NextRequest, { id }: { id: string }) {
const q = req.nextUrl.searchParams.get('q');
const body = await req.json();
return { id, q, body };
}
}
Finally, create the catch-all route with an optional slug ([[...slug]]
) and call activateControllers
with all your controllers. The slug is never used so you may want to keep it empty ([[...]]
).
// /src/app/api/[[...]]/route.ts
import { activateControllers } from 'next-smoothie';
import UserController from '../../../controllers/UserController';
export const { GET, POST } = activateControllers([UserController]);
After that you can load the data using any fetching library.
fetch('/api/users');
fetch(`/api/users/hello/${id}/world?q=foo`, {
method: 'POST',
body: JSON.stringify({ hello: 'world' }),
});
Table of contents
Features
next-smoothie offers a range of features to streamline your Next.js App Router experience:
- Elegant decorator syntax (all HTTP methods are available). Custom decorators for varied needs are supported.
- Direct data return from the handler (
Response
orNextResponse
usage isn't required). - Pleasant error handling (no need to use
try..catch
andNextResponse
to return an error to the client). - Service-Controller pattern is supported.
- The library does not interfere with built-in Next.js features including extending of request object.
Overview
Why Next.js is a good choice?
Next.js 13+ with App Router is a great ready-to-go framework that saves a lot of time and effort setting up and maintaining a React project. With Next.js:
- You don't need to manually set up Webpack, Babel, ESLint, TypeScript.
- Hot module reload is enabled by default and always works, so you don't need to find out why it stopped working after a dependency update.
- Server-side rendering is enabled by default.
- Routing and file structure are well-documented, eliminating the need for custom design.
- It doesn't require you to "eject" scripts and configs if you want to modify them.
- It's a widely known and well-used framework, no need to spend time thinking of a choice.
As result both long-term and short-term the development is cheaper, faster and more efficient.
Limitations of Next.js API Routes
The pros mentioned above are about front-end part (routes created with page.tsx
), but the API route handlers provide very specific and very limited way to define API routes. Per every endpoint you're going to create a separate file called route.ts
that exports route handlers that implement an HTTP method corresponding to their name:
export async function GET() {
// ...
return NextResponse.json(data)
}
export async function POST() {
// ...
return NextResponse.json(data)
}
Let's imagine that your app requires to build the following endpoints:
GET /user - get all users
POST /user - create user
GET /user/me - get current user
PUT /user/me - update current user (password, etc)
GET /user/[id] - get specified user by ID
PUT /user/[id] - update a specified user (let's say, name only)
GET /team - get all teams
GET /team/[id] - get a specific team
POST /team/[id]/assign-user - some specialised endpoint that assigns a user to a specific team (whatever that means)
With the built-in Next.js 13+ features your API folder structure is going to look the following:
/api/user/
/route.ts
/me/
/route.ts
/[id]/
/route.ts
/api/team/
/route.ts
/[id]/
/route.ts
/assign-user/
/route.ts
It's hard to manage this file structure (especially if you have complex API), and you may want to apply some creativity to reduce number of files and simplify the structure:
- Move all features from /users folder (
/me
and/[id]
) to/user/route.ts
and use query parameter instead:/user
,/user/?who=me
,/user/?who=[id]
- Do the same trick with the teams:
/team
,/team?id=[id]
,/team?id=[id]&action=assign-user
The file structure now looks like the following:
/api/user/
/route.ts
/api/team/
/route.ts
It looks better (even though it still looks wrong) but the code inside these files make you write too many if
conditions and will definitely make your code less readable. To make this documentation shorter, let me rely on your imagination.
A potential solution: Pairing Next.js with NestJS
Last few years I solved the problem above by combining Next.js and NestJS framework in one project. Next.js was used as a front-end framework and NestJS was used as back-end framework. Unfortunately this solution requires to spend resources on additional code and deployment management:
- Should it be a monorepo or 2 different repositories?
- Monorepo is harder to manage and deploy.
- Two repos are harder to synchronize (if deployed back-end code and front-end code compatible to each other at this moment of time?).
- Both applications require to be run on their own port and we need to deploy them to 2 different servers. Multiply that by the numbers of environments (the most common are: dev, staging, prod) and you'll need to handle too many servers.
It would be nice if we could:
- Use a single NodeJS project run in 1 port;
- Keep the project in one simple repository;
- Use single deployment server;
- Apply NestJS-like syntax to define routes;
- Make the project development and infrastructure cheaper.
The new solution: next-smoothie
Next.js includes Dynamic Routes that enable us to create "catch-all" route handlers for a specific endpoint prefix. The library uses this feature to implement creation of route handlers with much more friendly syntax. The route handlers are going to be exported on one catch-all route file. To achieve that you're going to need to create the following files:
/api/[[...]]/route.ts
/controllers
/UserController.ts
/TeamController.ts
First, /controllers
is a folder that contains our dynamic controller files. The names of the folder and files don't matter so you can name it /routers
for example.
Create your controllers:
// /controllers/UserController.ts
import { get, post, put, prefix } from 'next-smoothie';
@prefix('users')
export default class UserController {
@get()
static getAll() {
return someORM.getAllUsers();
}
@get('me')
static getMe() {
// ...
}
@put('me')
static async updateMe(req: NextRequest) {
const body = await req.json() as { firstName: string; lastName: string; };
// ...
}
@get(':id')
static async getOneUser(req: NextRequest, { id }: { id: string }) {
return someORM.getUserById(id);
}
@put(':id')
static async updateOneUser(req: NextRequest, { id }: { id: string }) {
const body = await req.json() as { firstName: string; lastName: string; };
return someORM.updateUserById(id, body);
}
}
// /controllers/TeamController.ts
import { get, post, prefix } from 'next-smoothie';
@prefix('teams')
export default class TeamController {
@get()
static getAll() {
return someORM.getAllTeams();
}
@get(':id')
static getOneTeam(req: NextRequest, { id }: { id: string }) {
// ...
}
@post(':id/assign-user')
static assignUser() {
// ...
}
}
Finally, create the catch-all route.
// /api/[[...]]/route.ts - this is a real file path where [[...]] is a folder name
import { activateControllers } from 'next-smoothie';
import UserController from '../controllers/UserController';
import TeamController from '../controllers/TeamController';
export const { GET, POST, PUT } = activateControllers([UserController, TeamController]);
That's it. Notice that the methods modified by the decorators defined as static
methods and the classes are never instantiated.
Also it's worthy to mention that @prefix
decorator is just syntax sugar and you're not required to use it.
Custom decorators
You can extend features of the controller by defining a custom decorator that can:
- Run additional request validation, for example to check if user is authorised.
- Catch specific errors.
- Add more properties to the
req
object: current user, parsed and modified request body, etc.
There is typical code from a random project:
// ...
export default class MyController {
// ...
@post()
@authGuard()
@permissionGuard(Permission.CREATE)
@log(Action.CREATE, { model: 'MyModel' })
@handleZodErrors()
static async create(req: GuardedRequest) {
const body = ZodModel.parse(await req.json());
return this.myService.create(body);
}
// ...
}
To create a decorator you can use createDecorator
that's described at the API section with a few examples.
All further examples are going to use Prisma ORM but you can use any ORM you like.
Service-Controller pattern
Optionally, you can improve your controller code by splitting it into service and controller. Service is a place where you make database requests and perform other data manipulation actions. Controller is where we use the decorators, check permissions, and validate incoming data, then call methods of the service. To achieve that, create another simple class (without no parent or decorators) with static methods:
// /controllers/user/UserService.ts
export default class UserService {
static findAllUsers() {
return prisma.user.findMany();
}
}
Then inject the service as another static property to the controller
// /controllers/user/UserController.ts
import UserService from './UserService';
// ...
@prefix('users')
export default class UserController {
private static userService = UserService;
@get()
@authGuard()
static getAllUsers() {
return this.userService.findAllUsers();
}
}
Then initialise the controller as before:
// /api/[[...]]/route.ts
import { activateControllers } from 'next-smoothie';
import UserController from '../controllers/user/UserController';
export const { GET } = activateControllers([UserController]);
Potential file structure with users, posts and comments may look like that:
/controllers/
/user/
/UserService.ts
/UserController.ts
/post/
/PostService.ts
/PostController.ts
/comment/
/CommentService.ts
/CommentController.ts
Services can use other services:
// /controllers/user/UserService.ts
import PostService from '../post/PostService';
export default class UserService {
private static postService = PostService;
static doSomething() {
this.postService.doSomething();
}
}
In case service A is dependent on service B, and service B is dependent on service A you can turn the other service property into a getter:
// /controllers/user/UserService.ts
import PostService from '../post/PostService';
export default class UserService {
private static get postService() { return PostService; };
static doSomething1() {
this.postService.doSomething2();
}
}
// /controllers/user/PostService.ts
import UserService from '../post/UserService';
export default class PostService {
private static get userService() { return UserService; };
static doSomething2() {
this.userService.doSomething1();
}
}
Or you can avoid setting up service as a property at all:
// /controllers/user/UserController.ts
import UserService from './UserService';
// ...
@prefix('users')
export default class UserController {
@get()
@authGuard()
static getAllUsers() {
return UserService.findAllUsers();
}
}
// /controllers/user/UserService.ts
import PostService from '../post/PostService';
export default class UserService {
static doSomething1() {
PostService.doSomething2();
}
}
But it is still recommended to declare services as class properties to keep the classes self-documented.
Return type
Controller method can return an instance of Response
or custom data. Custom data is serialised to JSON and returned with status 200.
@get()
static getSomething() {
// same as NextResponse.json({ hello: 'world' }, { status: 200 })
return { hello: 'world' };
}
- If
Response
instance (that also extendsNextResponse
) orundefined
is returned, passes it to the route handler as is. - If something else is returned, the library asumes that the value is an variable that needs to be serialised into JSON and sent to the client.
Take a look at this example:
import { redirect } from 'next/navigation';
class ExampleService {
@get('a')
static getA() {
return NextResponse.json({ hello: 'world' }, { status: 200 });
}
@get('b')
static getB() {
return new Response(JSON.stringify({ hello: 'world' }), {
status: 200,
headers: {
'Content-Type': 'application/json',
},
});
}
@get('c')
static getC() {
// return nothing (undefined)
}
@get('d')
static getD() {
return { hello: 'world' };
}
}
- The routes A and B respond with result as is because they both return an instance of
Response
. - Route C returns
undefined
as is and causes an error "No response is returned from route handler". - Route D serialises the returned custom data and sends it to the client. The following snippet of code will probably make it clearer:
export default function GET() {
// ...
// A, B, C
if(result instanceof Response || typeof result === 'undefined') {
return result;
}
// D
return NextResponse.json(result);
}
Error handling
You can throw errors directly from the controller method. The library catches thrown exception and returns an object of type ErrorResponseBody
.
// some client-side code
import { type ErrorResponseBody } from 'next-smoothie';
const dataOrError: MyData | ErrorResponseBody = await (await fetch('...')).json();
The shape of this type is the following:
type ErrorResponseBody = {
statusCode: HttpStatus;
message: string;
isError: true;
}
To throw an error you can use HttpException
class together with HttpStatus
enum. You can also throw the errors from the service methods.
import { HttpException, HttpStatus } from 'next-smoothie'
// ...
@get()
static getSomething() {
if(somethingWrong) {
throw new HttpException(HttpStatus.I_AM_A_TEAPOT, "I'm a teapot");
}
// ...
}
// ...
All other exceptions are considered as 500 errors and handled similarly.
// ...
@get()
static getSomething() {
if(somethingWrong) {
throw new Error('Something is wrong');
}
// ...
}
// ...
API
import {
// main API
type ErrorResponseBody,
HttpException,
HttpStatus,
createSegment,
createDecorator,
// global controller members created with createSegment
get, post, put, patch, del, head, options,
prefix,
activateControllers,
} from 'next-smoothie';
createSegment
function, global decorators and handlers
The function createSegment
initialises route handlers for one particular router segment. Using the function directly allows you to isolate some particular route path from other route handlers and provides a chance to refactor your code partially. Let's say you want to override only /users
route handlers by using the library but keep /comments
and /posts
as is.
/api/posts/
/route.ts
/[id]/
/route.ts
/api/comments/
/route.ts
/[id]/
/route.ts
/api/users/[[...]]/
/route.ts
In this example, only the users
dynamic route will utilize the library. With createSegment
you can define local variables that are going to be used for one particular segment.
import { createSegment } from 'next-smoothie';
const { get, post, activateControllers } = createSegment();
class UserController {
@get()
static getAll() {
// ...
}
@post()
static create() {
// ...
}
}
export const { GET, POST } = activateControllers([UserController]);
This is what createSegment
returns:
const {
get, post, put, patch, del, head, options, // HTTP methods
prefix,
activateControllers,
} = createSegment();
(notice that DELETE method decorator is shortned to @del
).
activateControllers
returns all route handlers for all supported HTTP methods and also accepts options with onError
handler that allows to listen to all errors for logging. It is important to remember that it is also called on NEXT_REDIRECT.
export const { GET, POST, PUT, PATCH, DELETE, OPTIONS, HEAD } = activateControllers(controllers, {
onError(error) {
console.log(error);
}
});
As you may already guess, some of the the variables imported from the library are created by createSegment
to keep the code cleaner for the "global" segment instance.
// these vars are initialised within the library by createSegment
import {
get, post, put, patch, del, head, options,
prefix,
activateControllers,
} from 'next-smoothie';
HttpException
class and HttpStatus
enum
HttpException
accepts 2 arguments. The first one is an HTTP code that can be retrieved from HttpStatus
, the other one is error text.
import { HttpException, HttpStatus } from 'next-smoothie';
// ...
throw new HttpException(HttpStatus.BAD_REQUEST, 'Something went wrong');
HttpMethod
enum
HttpMethod
enum has no specific purpose. It is used internally and I thought it might be useful to export it. You can use it with your fetching library for example:
fetch('...', {
method: HttpMethod.POST,
})
createDecorator
function
createDecorator
is a higher-order function that produces a decorator factory (a function that returns a decorator). It accepts a middleware function with the following parameters:
request
, which extendsNextRequest
.next
, a function that should be invoked and its result returned to call subsequent decorators or the route handler.- Additional arguments are passed through to the decorator factory.
import { createDecorator, get } from 'next-smoothie';
const myDecorator = createDecorator((req, next, a: string, b: number) => {
console.log(a, b); // Outputs: "foo", 1
if(isSomething) {
// override route method behavior and return { hello: 'world' } from the endpoint
return { hello: 'world' };
}
return next();
});
class MyController {
@get()
@myDecorator('foo', 1) // Passes 'foo' as 'a', and 1 as 'b'
static get() {
// ...
}
}
authGuard
example
There is the example code that defines authGuard
decorator that does two things:
- Checks if a user is authorised and returns an Unauthorised status if not.
- Adds
currentUser
to the request object.
To extend req
object you can define your custom interface that extends NextRequest
.
// types.ts
import { type NextRequest } from 'next/server'
import { type User } from '@prisma/client';
export default interface GuardedRequest extends NextRequest {
currentUser: User;
}
Then define the authGuard
decorator itself.
// authGuard.ts
import { HttpException, HttpStatus, createDecorator } from 'next-smoothie';
import { NextRequest } from 'next/server';
import checkAuth from './checkAuth';
const authGuard = createDecorator(async (req: GuardedRequest, next) => {
// ... define userId and isAuthorised
// parse access token for example
if (!isAuthorised) {
throw new HttpException(HttpStatus.UNAUTHORIZED, 'Unauthorized');
}
// let's imagine you use Prisma and you want to find a user by userId
const currentUser = await prisma.user.findUnique({ where: { id: userId } });
req.currentUser = currentUser;
return next();
});
export default authGuard;
And finally use the decorator as we did above:
// ...
export default class UserController {
// ...
@get('me')
@authGuard()
static async getMe(req: GuardedRequest) {
return req.currentUser;
}
// ...
}
handleZodErrors
example
You can catch any error in your custom decorator and provide relevant response to the client. At this exmple we're checking if ZodError
is thrown.
import { ZodError } from 'zod';
import { HttpException, HttpStatus, createDecorator } from 'next-smoothie';
const handleZodErrors = createDecorator(async (req, next) => {
try {
return await next();
} catch (e) {
if (e instanceof ZodError) {
throw new HttpException(
HttpStatus.BAD_REQUEST,
e.errors?.map((error) => `${error.code}: ${error.message}`).join('; ') ?? 'Validation error'
);
}
throw e;
}
});
export default handleZodErrors;
If ZodModel.parse
encounters an error and throws a ZodError
the decorator is going to catch it and return corresponding response.
// ...
export default class UserController {
// ...
@post()
@handleZodErrors()
static async create(req: NextRequest) {
const data = ZodModel.parse(await req.json());
}
// ...
}
Enjoy!