Skip to content

Commit 3efc19f

Browse files
authored
Merge pull request #866 from g-ongenae/feature/add-option-to-mask-headers
[HTTP-EXCEPTION-FILTER] Add option to mask headers
2 parents 4cf0af5 + ca3f18e commit 3efc19f

File tree

2 files changed

+142
-4
lines changed

2 files changed

+142
-4
lines changed

packages/http-exception-filter/src/http-exception-filter.ts

+105-3
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,70 @@
11
import { ArgumentsHost, Catch, ExceptionFilter, HttpException, HttpStatus, Logger } from '@nestjs/common';
22
import { HttpArgumentsHost } from '@nestjs/common/interfaces';
3-
import { Response } from 'express';
3+
import { Request, Response } from 'express';
44
import { get } from 'lodash';
55
import { getCode, getErrorMessage } from './error.utils';
66

7+
/**
8+
* Option to mask headers
9+
*/
10+
export type MaskHeaders = Record<string, boolean | ((headerValue: string | string[]) => unknown)>;
11+
12+
/**
13+
* HttpExceptionFilter options
14+
*/
15+
export interface HttpExceptionFilterOptions {
16+
/**
17+
* Disable the masking of headers
18+
* @default false
19+
*/
20+
disableMasking?: boolean;
21+
22+
/**
23+
* Placeholder to use when masking a header
24+
* @default '****';
25+
*/
26+
maskingPlaceholder?: string;
27+
28+
/**
29+
* Mask configuration
30+
*/
31+
mask?: {
32+
/**
33+
* The headers to mask with their mask configuration
34+
* - `true` to replace the header value with the `maskingPlaceholder`
35+
* - a function to replace the header value with the result of the function
36+
* @example
37+
* ```ts
38+
* mask: {
39+
* requestHeader: {
40+
* // log authorization type only
41+
* 'authorization': (headerValue: string) => headerValue.split(' ')[0],
42+
* 'x-api-key': true,
43+
* }
44+
* }
45+
* ```
46+
*/
47+
requestHeader?: MaskHeaders;
48+
};
49+
}
50+
751
/**
852
* Catch and format thrown exception in NestJS application based on Express
953
*/
1054
@Catch()
1155
export class HttpExceptionFilter implements ExceptionFilter {
1256
private readonly logger: Logger = new Logger(HttpExceptionFilter.name);
1357

58+
private readonly disableMasking: boolean;
59+
private readonly maskingPlaceholder: string;
60+
private readonly mask: HttpExceptionFilterOptions['mask'];
61+
62+
constructor(options?: HttpExceptionFilterOptions) {
63+
this.disableMasking = options?.disableMasking ?? false;
64+
this.maskingPlaceholder = options?.maskingPlaceholder ?? '****';
65+
this.mask = options?.mask ?? {};
66+
}
67+
1468
/**
1569
* Catch and format thrown exception
1670
*/
@@ -50,15 +104,15 @@ export class HttpExceptionFilter implements ExceptionFilter {
50104
this.logger.error(
51105
{
52106
message: `${status} [${request.method} ${request.url}] has thrown a critical error`,
53-
headers: request.headers,
107+
headers: this.maskHeaders(request.headers),
54108
},
55109
exceptionStack,
56110
);
57111
} else if (status >= HttpStatus.BAD_REQUEST) {
58112
this.logger.warn({
59113
message: `${status} [${request.method} ${request.url}] has thrown an HTTP client error`,
60114
exceptionStack,
61-
headers: request.headers,
115+
headers: this.maskHeaders(request.headers),
62116
});
63117
}
64118
response.status(status).send({
@@ -67,4 +121,52 @@ export class HttpExceptionFilter implements ExceptionFilter {
67121
status,
68122
});
69123
}
124+
125+
/**
126+
* Mask the given headers
127+
* @param headers the headers to mask
128+
* @returns the masked headers
129+
*/
130+
private maskHeaders(headers: Request['headers']): Record<string, unknown> {
131+
if (this.disableMasking || this.mask?.requestHeader === undefined) {
132+
return headers;
133+
}
134+
135+
return Object.keys(headers).reduce<Record<string, unknown>>(
136+
(maskedHeaders: Record<string, unknown>, headerKey: string): Record<string, unknown> => {
137+
const headerValue = headers[headerKey];
138+
const mask = this.mask?.requestHeader?.[headerKey];
139+
140+
if (headerValue === undefined) {
141+
return maskedHeaders;
142+
}
143+
144+
if (mask === true) {
145+
return {
146+
...maskedHeaders,
147+
[headerKey]: this.maskingPlaceholder,
148+
};
149+
}
150+
151+
if (typeof mask === 'function') {
152+
try {
153+
return {
154+
...maskedHeaders,
155+
[headerKey]: mask(headerValue),
156+
};
157+
} catch (error) {
158+
this.logger.warn(`HttpFilterOptions - Masking error for header ${headerKey}`, { error, mask, headerKey });
159+
160+
return {
161+
...maskedHeaders,
162+
[headerKey]: this.maskingPlaceholder,
163+
};
164+
}
165+
}
166+
167+
return maskedHeaders;
168+
},
169+
headers,
170+
);
171+
}
70172
}

packages/http-exception-filter/test/http-exception-filter.test.ts

+37-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,17 @@ describe('Http Exception Filter', () => {
1515
app = moduleRef.createNestApplication();
1616
app.useLogger(Logger);
1717
app.useGlobalPipes(new ValidationPipe());
18-
app.useGlobalFilters(new HttpExceptionFilter());
18+
app.useGlobalFilters(
19+
new HttpExceptionFilter({
20+
mask: {
21+
requestHeader: {
22+
authorization: (headerValue: string | string[]) =>
23+
typeof headerValue === 'string' ? headerValue.split(' ')[0] : undefined,
24+
'x-api-key': true,
25+
},
26+
},
27+
}),
28+
);
1929

2030
await app.init();
2131
});
@@ -156,4 +166,30 @@ describe('Http Exception Filter', () => {
156166
status: 413,
157167
});
158168
});
169+
170+
it('should mask the headers', async () => {
171+
const warnSpy: jest.SpyInstance = jest.spyOn(Logger.prototype, 'warn');
172+
const url: string = `/cats/notfound`;
173+
174+
const { body: resBody } = await request(app.getHttpServer())
175+
.get(url)
176+
.set('Authorization', 'Bearer 123456')
177+
.set('X-API-Key', '123456')
178+
.expect(HttpStatus.NOT_FOUND);
179+
180+
expect(resBody).toEqual({
181+
code: 'UNKNOWN_ENTITY',
182+
message: 'Id notfound could not be found',
183+
status: 404,
184+
});
185+
186+
expect(warnSpy).toHaveBeenCalledWith({
187+
message: `404 [GET ${url}] has thrown an HTTP client error`,
188+
exceptionStack: expect.any(String),
189+
headers: expect.objectContaining({
190+
authorization: 'Bearer',
191+
'x-api-key': '****',
192+
}),
193+
});
194+
});
159195
});

0 commit comments

Comments
 (0)