1
1
import { ArgumentsHost , Catch , ExceptionFilter , HttpException , HttpStatus , Logger } from '@nestjs/common' ;
2
2
import { HttpArgumentsHost } from '@nestjs/common/interfaces' ;
3
- import { Response } from 'express' ;
3
+ import { Request , Response } from 'express' ;
4
4
import { get } from 'lodash' ;
5
5
import { getCode , getErrorMessage } from './error.utils' ;
6
6
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
+
7
51
/**
8
52
* Catch and format thrown exception in NestJS application based on Express
9
53
*/
10
54
@Catch ( )
11
55
export class HttpExceptionFilter implements ExceptionFilter {
12
56
private readonly logger : Logger = new Logger ( HttpExceptionFilter . name ) ;
13
57
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
+
14
68
/**
15
69
* Catch and format thrown exception
16
70
*/
@@ -50,15 +104,15 @@ export class HttpExceptionFilter implements ExceptionFilter {
50
104
this . logger . error (
51
105
{
52
106
message : `${ status } [${ request . method } ${ request . url } ] has thrown a critical error` ,
53
- headers : request . headers ,
107
+ headers : this . maskHeaders ( request . headers ) ,
54
108
} ,
55
109
exceptionStack ,
56
110
) ;
57
111
} else if ( status >= HttpStatus . BAD_REQUEST ) {
58
112
this . logger . warn ( {
59
113
message : `${ status } [${ request . method } ${ request . url } ] has thrown an HTTP client error` ,
60
114
exceptionStack,
61
- headers : request . headers ,
115
+ headers : this . maskHeaders ( request . headers ) ,
62
116
} ) ;
63
117
}
64
118
response . status ( status ) . send ( {
@@ -67,4 +121,52 @@ export class HttpExceptionFilter implements ExceptionFilter {
67
121
status,
68
122
} ) ;
69
123
}
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
+ }
70
172
}
0 commit comments