Skip to content

Commit 3d65af2

Browse files
committed
fix(core): Improve middleware route exclusion for dynamic endpoints
fixes nestjs#13593
1 parent 67e30e2 commit 3d65af2

File tree

6 files changed

+470
-18
lines changed

6 files changed

+470
-18
lines changed

packages/core/middleware/builder.ts

+12-1
Original file line numberDiff line numberDiff line change
@@ -11,9 +11,10 @@ import {
1111
} from '@nestjs/common/interfaces/middleware';
1212
import { stripEndSlash } from '@nestjs/common/utils/shared.utils';
1313
import { iterate } from 'iterare';
14+
import { isRouteIncluded } from '../router/utils';
1415
import { RouteInfoPathExtractor } from './route-info-path-extractor';
1516
import { RoutesMapper } from './routes-mapper';
16-
import { filterMiddleware } from './utils';
17+
import { filterMiddleware, mapToExcludeRoute } from './utils';
1718

1819
export class MiddlewareBuilder implements MiddlewareConsumer {
1920
private readonly middlewareCollection = new Set<MiddlewareConfiguration>();
@@ -84,9 +85,19 @@ export class MiddlewareBuilder implements MiddlewareConsumer {
8485

8586
const flattedRoutes = this.getRoutesFlatList(routes);
8687
const forRoutes = this.removeOverlappedRoutes(flattedRoutes);
88+
89+
const includedRoutes = flattedRoutes.filter(route =>
90+
isRouteIncluded(
91+
mapToExcludeRoute(this.excludedRoutes),
92+
route.path,
93+
route.method,
94+
),
95+
);
96+
8797
const configuration = {
8898
middleware: filterMiddleware(
8999
this.middleware,
100+
includedRoutes,
90101
this.excludedRoutes,
91102
this.builder.getHttpAdapter(),
92103
),

packages/core/middleware/utils.ts

+25-8
Original file line numberDiff line numberDiff line change
@@ -8,9 +8,11 @@ import {
88
import { iterate } from 'iterare';
99
import * as pathToRegexp from 'path-to-regexp';
1010
import { uid } from 'uid';
11-
import { ExcludeRouteMetadata } from '../router/interfaces/exclude-route-metadata.interface';
11+
import {
12+
ExcludeRouteMetadata,
13+
RouteMetadata,
14+
} from '../router/interfaces/exclude-route-metadata.interface';
1215
import { isRouteExcluded } from '../router/utils';
13-
1416
export const mapToExcludeRoute = (
1517
routes: (string | RouteInfo)[],
1618
): ExcludeRouteMetadata[] => {
@@ -32,20 +34,28 @@ export const mapToExcludeRoute = (
3234

3335
export const filterMiddleware = <T extends Function | Type<any> = any>(
3436
middleware: T[],
35-
routes: RouteInfo[],
37+
includedRoutes: RouteInfo[],
38+
excludedRoutes: RouteInfo[],
3639
httpAdapter: HttpServer,
3740
) => {
38-
const excludedRoutes = mapToExcludeRoute(routes);
3941
return iterate([])
4042
.concat(middleware)
4143
.filter(isFunction)
42-
.map((item: T) => mapToClass(item, excludedRoutes, httpAdapter))
44+
.map((item: T) =>
45+
mapToClass(
46+
item,
47+
mapToExcludeRoute(includedRoutes),
48+
mapToExcludeRoute(excludedRoutes),
49+
httpAdapter,
50+
),
51+
)
4352
.toArray();
4453
};
4554

4655
export const mapToClass = <T extends Function | Type<any>>(
4756
middleware: T,
48-
excludedRoutes: ExcludeRouteMetadata[],
57+
includedRoutes: RouteMetadata[],
58+
excludedRoutes: RouteMetadata[],
4959
httpAdapter: HttpServer,
5060
) => {
5161
if (isMiddlewareClass(middleware)) {
@@ -57,6 +67,7 @@ export const mapToClass = <T extends Function | Type<any>>(
5767
const [req, _, next] = params as [Record<string, any>, any, Function];
5868
const isExcluded = isMiddlewareRouteExcluded(
5969
req,
70+
includedRoutes,
6071
excludedRoutes,
6172
httpAdapter,
6273
);
@@ -74,6 +85,7 @@ export const mapToClass = <T extends Function | Type<any>>(
7485
const [req, _, next] = params as [Record<string, any>, any, Function];
7586
const isExcluded = isMiddlewareRouteExcluded(
7687
req,
88+
includedRoutes,
7789
excludedRoutes,
7890
httpAdapter,
7991
);
@@ -106,7 +118,8 @@ export function assignToken(metatype: Type<any>, token = uid(21)): Type<any> {
106118

107119
export function isMiddlewareRouteExcluded(
108120
req: Record<string, any>,
109-
excludedRoutes: ExcludeRouteMetadata[],
121+
includedRoutes: RouteMetadata[],
122+
excludedRoutes: RouteMetadata[],
110123
httpAdapter: HttpServer,
111124
): boolean {
112125
if (excludedRoutes.length <= 0) {
@@ -120,5 +133,9 @@ export function isMiddlewareRouteExcluded(
120133
? originalUrl.slice(0, queryParamsIndex)
121134
: originalUrl;
122135

123-
return isRouteExcluded(excludedRoutes, pathname, RequestMethod[reqMethod]);
136+
if (includedRoutes.length > 0) {
137+
return !isRouteExcluded(includedRoutes, pathname, RequestMethod[reqMethod]);
138+
} else {
139+
return isRouteExcluded(excludedRoutes, pathname, RequestMethod[reqMethod]);
140+
}
124141
}

packages/core/router/interfaces/exclude-route-metadata.interface.ts

+2
Original file line numberDiff line numberDiff line change
@@ -16,3 +16,5 @@ export interface ExcludeRouteMetadata {
1616
*/
1717
requestMethod: RequestMethod;
1818
}
19+
20+
export type RouteMetadata = ExcludeRouteMetadata;

packages/core/router/utils/exclude-route.util.ts

+16
Original file line numberDiff line numberDiff line change
@@ -21,3 +21,19 @@ export function isRouteExcluded(
2121
return false;
2222
});
2323
}
24+
25+
export function isRouteIncluded(
26+
excludedRoutes: ExcludeRouteMetadata[],
27+
path: string,
28+
requestMethod?: RequestMethod,
29+
) {
30+
return excludedRoutes.some(route => {
31+
if (
32+
isRequestMethodAll(route.requestMethod) ||
33+
route.requestMethod === requestMethod
34+
) {
35+
return route.pathRegex.exec(addLeadingSlash(path)) && route.path !== path;
36+
}
37+
return false;
38+
});
39+
}

0 commit comments

Comments
 (0)