Compose middleware
In your middleware.js
you might potentially want to do more interesting things than just setting security headers. However, Next.js doesn't offer an idiomatic abstraction for composing middlewares yet.
If you like to use this package for composing middleware only, there's a
bundle @komw/next-safe-middleware/dist/compose
for just that.
chain
This package provides a minimal and simple abstraction for composing/chaining multiple middlewares within the middleware.js
file. For this, it provides a function chain
, that accepts an array of type ChainableMiddleware
.
The ChainableMiddleware
interface is compatible with the Next.js spec (what needs to be exported from middleware.js
) and provides additional handles for composition of multiple middlewares with chain
:
type NextMiddlewareResult = NextResponse | Response | null | undefined;
// this is the Middleware spec of Next.jstype NextMiddleware = ( request: NextRequest, event: NextFetchEvent) => NextMiddlewareResult | Promise<NextMiddlewareResult>;
// this is the extended spec this package usestype ChainableMiddleware = ( request: NextRequest, event: NextFetchEvent ctx?: MiddlewareChainContext) => NextMiddlewareResult | void | Promise<NextMiddlewareResult | void>;
request
: https://vercel.com/docs/concepts/functions/edge-functions#nextrequest
event
: https://vercel.com/docs/concepts/functions/edge-functions#nextfetchevent
ctx
: a context object of type MiddlewareChainContext
:
type MiddlewareChainContext<K, V> = { res: { readonly get: () => NextMiddlewareResult; readonly set: (res: NextMiddlewareResult, override?: boolean) => void; }; cache: { readonly get: (key: K) => V | null | undefined; readonly set: (key: K, value: V) => void; }; finalize: { readonly addCallback: (finalizer: ChainFinalizer<K, V>) => void; readonly terminatedByResponse: () => NextMiddlewareResult; };};
res
: accessors for a continued response. ChainableMiddleware
's' can access and modify it (adding/changing headers, cookies etc.) or set it for middlewares to their right to pick up upon.
cache
: accessors for an object cache for the lifetime of chain
.
finalize
: handle for ChainableMiddleware
's to register callbacks to be executed before chain
terminates.
When executed in chain
, the return value of ChainableMiddleware
is interpreted as follows:
- a
ChainableMiddleware
returns a response:chain
gets terminated. Middlewares to its right in the chain don't execute - a
ChainableMiddleware
returns nothing:chain
continues with the middleware to its right - a
ChainableMiddleware
passes a response withctx.res.set
and returns nothing:chain
continues with the middleware to its right, which can retrieve this response withctx.res.get
.
Only match a subset of requests with chainMatch
To execute a middleware chain only on a subset of requests, there's a variant chainMatch
.
It consumes a matcher function of type NextRequestPredicate
as its argument and returns a matched variant of chain
:
type NextRequestPredicate = (req: NextRequest) => boolean;type ChainMatcher = NextRequestPredicate;
The resulting middleware chain will only execute if the matcher function evaluates to true
for a request. Other than that, it behaves like regular chain
.
const isPageRequest: NextRequestPredicate = (req) => ...
const chainableMiddlewares: ChainableMiddleware[] = [...]
const chainOnPageRequests = chainMatch(isPageRequest)
const middleware = chainOnPageRequests(...chainableMiddlewares)
Stable root-level middleware of Next.js
12.2 matches all requests by default,
including assets in public
and files under /_next
. Read the middleware
upgrade guide
before you upgrade an existing setup with Beta middleware.
The abstractions around chain
are work in progress and an experiment to see
what works well for composition and they are necessary to ship CSP and
security middleware as modular and reusable pieces. As soon as something
simliar becomes part of Next.js, this package will use them and deprecate the
custom ones.
Example: Security middleware with Geo-blocking
// middleware.jsimport { chain, chainMatch, isPageRequest, csp, strictDynamic,} from "@komw/next-safe-middleware";
/** @type {import('@komw/next-safe-middleware').ChainableMiddleware} */const geoBlockMiddleware = (req) => { const BLOCKED_COUNTRY = "GB"; const country = req.geo.country || "US";
if (country === BLOCKED_COUNTRY) { const response = new Response("Blocked for legal reasons", { status: 451 }); // returning response terminates the chain return response; }};
const securityMiddleware = [csp(), strictDynamic()];
/** * geoBlockMiddleware will be invoked on all requests * from `BLOCKED_COUNTRY` and then block the request * and terminate chain by returning a response with status 451 * * securityMiddleware will only run on requests * that didn't get geo-blocked and only on requests for pages */export default chain( geoBlockMiddleware, chainMatch(isPageRequest)(...securityMiddleware));
Configurable middleware
Usage
Every middleware of this package supports configuration with an (async) initializer function:
type ConfigInitializerParams = { req: NextRequest; evt: NextFetchEvent; ctx: MiddlewareChainContext; // from next/server userAgent: UserAgent;};
type ConfigInitalizer<Config> = ( params: ConfigInitializerParams) => Config | Promise<Config>;
type MiddlewareConfig<Config> = Config | ConfigInitalizer<Config>;
For example, you can use this capability to select different CSP configurations for different user agents:
// middleware.jsimport { chainMatch, isPageRequest, csp, strictDynamic,} from "@komw/next-safe-middleware";
// CSP in always in report-only mode for Firefox// and by env var for other browsersconst cspMiddleware = csp(async ({ userAgent }) => { const browserName = userAgent.browser.name || ""; const reportOnly = !!process.env.CSP_REPORT_ONLY || browserName.includes("Firefox"); return { reportOnly, };});
const securityMiddleware = [cspMiddleware, strictDynamic()];
export default chainMatch(isPageRequest)(...securityMiddleware);
Builder
TBD