Middleware
Middleware is a function which is called before the route handler. Middleware functions have access to the request and response objects, and the next()
middleware function in the application’s request-response cycle. The next middleware function is commonly denoted by a variable named next
.
data:image/s3,"s3://crabby-images/4bf9e/4bf9edba4fcccb432b4216405d2dca256f2a8c6f" alt=""
Nest middleware are, by default, equivalent to express middleware. The following description from the official express documentation describes the capabilities of middleware:
Middleware functions can perform the following tasks:
- execute any code.
- make changes to the request and the response objects.
- end the request-response cycle.
- call the next middleware function in the stack.
- if the current middleware function does not end the request-response cycle, it must call
next()
to pass control to the next middleware function. Otherwise, the request will be left hanging.
You implement custom Nest middleware in either a function, or in a class with an @Injectable()
decorator. The class should implement the NestMiddleware
interface, while the function does not have any special requirements. Let's start by implementing a simple middleware feature using the class method.
WarningExpress
andfastify
handle middleware differently and provide different method signatures, read more here.
import { Injectable, NestMiddleware } from '@nestjs/common';
import { Request, Response, NextFunction } from 'express';
@Injectable()
export class LoggerMiddleware implements NestMiddleware {
use(req: Request, res: Response, next: NextFunction) {
console.log('Request...');
next();
}
}
import { Injectable } from '@nestjs/common';
@Injectable()
export class LoggerMiddleware {
use(req, res, next) {
console.log('Request...');
next();
}
}
Dependency injection#
Nest middleware fully supports Dependency Injection. Just as with providers and controllers, they are able to inject dependencies that are available within the same module. As usual, this is done through the constructor
.
Applying middleware#
There is no place for middleware in the @Module()
decorator. Instead, we set them up using the configure()
method of the module class. Modules that include middleware have to implement the NestModule
interface. Let's set up the LoggerMiddleware
at the AppModule
level.
import { Module, NestModule, MiddlewareConsumer } from '@nestjs/common';
import { LoggerMiddleware } from './common/middleware/logger.middleware';
import { CatsModule } from './cats/cats.module';
@Module({
imports: [CatsModule],
})
export class AppModule implements NestModule {
configure(consumer: MiddlewareConsumer) {
consumer
.apply(LoggerMiddleware)
.forRoutes('cats');
}
}
import { Module } from '@nestjs/common';
import { LoggerMiddleware } from './common/middleware/logger.middleware';
import { CatsModule } from './cats/cats.module';
@Module({
imports: [CatsModule],
})
export class AppModule {
configure(consumer) {
consumer
.apply(LoggerMiddleware)
.forRoutes('cats');
}
}
In the above example we have set up the LoggerMiddleware
for the /cats
route handlers that were previously defined inside the CatsController
. We may also further restrict a middleware to a particular request method by passing an object containing the route path
and request method
to the forRoutes()
method when configuring the middleware. In the example below, notice that we import the RequestMethod
enum to reference the desired request method type.
import { Module, NestModule, RequestMethod, MiddlewareConsumer } from '@nestjs/common';
import { LoggerMiddleware } from './common/middleware/logger.middleware';
import { CatsModule } from './cats/cats.module';
@Module({
imports: [CatsModule],
})
export class AppModule implements NestModule {
configure(consumer: MiddlewareConsumer) {
consumer
.apply(LoggerMiddleware)
.forRoutes({ path: 'cats', method: RequestMethod.GET });
}
}
import { Module, RequestMethod } from '@nestjs/common';
import { LoggerMiddleware } from './common/middleware/logger.middleware';
import { CatsModule } from './cats/cats.module';
@Module({
imports: [CatsModule],
})
export class AppModule {
configure(consumer) {
consumer
.apply(LoggerMiddleware)
.forRoutes({ path: 'cats', method: RequestMethod.GET });
}
}
Hint Theconfigure()
method can be made asynchronous usingasync/await
(e.g., you canawait
completion of an asynchronous operation inside theconfigure()
method body).
Warning When using theexpress
adapter, the NestJS app will registerjson
andurlencoded
from the packagebody-parser
by default. This means if you want to customize that middleware via theMiddlewareConsumer
, you need to turn off the global middleware by setting thebodyParser
flag tofalse
when creating the application withNestFactory.create()
.
Route wildcards#
Pattern-based routes are also supported in NestJS middleware. For example, the named wildcard (*splat
) can be used as a wildcard to match any combination of characters in a route. In the following example, the middleware will be executed for any route that starts with abcd/
, regardless of the number of characters that follow.
forRoutes({
path: 'abcd/*splat',
method: RequestMethod.ALL,
});
Hintsplat
is simply the name of the wildcard parameter and has no special meaning. You can name it anything you like, for example,*wildcard
.
The 'abcd/*'
route path will match abcd/1
, abcd/123
, abcd/abc
, and so on. The hyphen ( -
) and the dot (.
) are interpreted literally by string-based paths. However, abcd/
with no additional characters will not match the route. For this, you need to wrap the wildcard in braces to make it optional:
forRoutes({
path: 'abcd/{*splat}',
method: RequestMethod.ALL,
});
Middleware consumer#
The MiddlewareConsumer
is a helper class. It provides several built-in methods to manage middleware. All of them can be simply chained in the fluent style. The forRoutes()
method can take a single string, multiple strings, a RouteInfo
object, a controller class and even multiple controller classes. In most cases you'll probably just pass a list of controllers separated by commas. Below is an example with a single controller:
import { Module, NestModule, MiddlewareConsumer } from '@nestjs/common';
import { LoggerMiddleware } from './common/middleware/logger.middleware';
import { CatsModule } from './cats/cats.module';
import { CatsController } from './cats/cats.controller';
@Module({
imports: [CatsModule],
})
export class AppModule implements NestModule {
configure(consumer: MiddlewareConsumer) {
consumer
.apply(LoggerMiddleware)
.forRoutes(CatsController);
}
}
import { Module } from '@nestjs/common';
import { LoggerMiddleware } from './common/middleware/logger.middleware';
import { CatsModule } from './cats/cats.module';
import { CatsController } from './cats/cats.controller';
@Module({
imports: [CatsModule],
})
export class AppModule {
configure(consumer) {
consumer
.apply(LoggerMiddleware)
.forRoutes(CatsController);
}
}
Hint The apply()
method may either take a single middleware, or multiple arguments to specify multiple middlewares.
Excluding routes#
At times, we may want to exclude certain routes from having middleware applied. This can be easily achieved using the exclude()
method. The exclude()
method accepts a single string, multiple strings, or a RouteInfo
object to identify the routes to be excluded.
Here's an example of how to use it:
consumer
.apply(LoggerMiddleware)
.exclude(
{ path: 'cats', method: RequestMethod.GET },
{ path: 'cats', method: RequestMethod.POST },
'cats/{*splat}',
)
.forRoutes(CatsController);
Hint The exclude()
method supports wildcard parameters using the path-to-regexp package.
With the example above, LoggerMiddleware
will be bound to all routes defined inside CatsController
except the three passed to the exclude()
method.
This approach provides flexibility in applying or excluding middleware based on specific routes or route patterns.
Functional middleware#
The LoggerMiddleware
class we've been using is quite simple. It has no members, no additional methods, and no dependencies. Why can't we just define it in a simple function instead of a class? In fact, we can. This type of middleware is called functional middleware. Let's transform the logger middleware from class-based into functional middleware to illustrate the difference:
import { Request, Response, NextFunction } from 'express';
export function logger(req: Request, res: Response, next: NextFunction) {
console.log(`Request...`);
next();
};
export function logger(req, res, next) {
console.log(`Request...`);
next();
};
And use it within the AppModule
:
consumer
.apply(logger)
.forRoutes(CatsController);
Hint Consider using the simpler functional middleware alternative any time your middleware doesn't need any dependencies.
Multiple middleware#
As mentioned above, in order to bind multiple middleware that are executed sequentially, simply provide a comma separated list inside the apply()
method:
consumer.apply(cors(), helmet(), logger).forRoutes(CatsController);
Global middleware#
If we want to bind middleware to every registered route at once, we can use the use()
method that is supplied by the INestApplication
instance:
const app = await NestFactory.create(AppModule);
app.use(logger);
await app.listen(process.env.PORT ?? 3000);
Hint Accessing the DI container in a global middleware is not possible. You can use a functional middleware instead when usingapp.use()
. Alternatively, you can use a class middleware and consume it with.forRoutes('*')
within theAppModule
(or any other module).