axios-error-redact 0.1.410 → 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +43 -13
- package/lib/index.d.ts +42 -20
- package/lib/index.js +86 -7
- package/lib/types.d.ts +69 -0
- package/lib/types.js +3 -0
- package/package.json +2 -2
package/README.md
CHANGED
|
@@ -5,7 +5,7 @@ This can be used as an response interceptor for axios instances, or can be used
|
|
|
5
5
|
|
|
6
6
|
# Compatibility
|
|
7
7
|
|
|
8
|
-
Works with
|
|
8
|
+
Works with
|
|
9
9
|
|
|
10
10
|
- `axios@^0`
|
|
11
11
|
- `axios@^1`
|
|
@@ -24,13 +24,21 @@ The redactor can simply be used in an interceptor to extract non-sensitive data
|
|
|
24
24
|
|
|
25
25
|
```javascript
|
|
26
26
|
import axios from 'axios'
|
|
27
|
-
import {
|
|
27
|
+
import {createErrorInterceptor} from 'axios-error-redact'
|
|
28
28
|
|
|
29
29
|
const instance = axios.create({baseURL: 'http://example.com'})
|
|
30
30
|
|
|
31
|
-
instance.interceptors.response.use(undefined,
|
|
31
|
+
instance.interceptors.response.use(undefined, createErrorInterceptor())
|
|
32
|
+
|
|
33
|
+
try {
|
|
34
|
+
await instance.get()
|
|
35
|
+
} catch(error) {
|
|
36
|
+
// The isHttpErrorResponse helper can be used to ensure the thrown error is a redacted error
|
|
37
|
+
if (isHttpErrorResponse(error)) {
|
|
38
|
+
console.error(error.response.statusMessage, error.message)
|
|
39
|
+
}
|
|
40
|
+
}
|
|
32
41
|
|
|
33
|
-
// instance.get()
|
|
34
42
|
|
|
35
43
|
```
|
|
36
44
|
|
|
@@ -77,7 +85,7 @@ const result = axios.get('http://example.com')
|
|
|
77
85
|
|
|
78
86
|
### Constructor
|
|
79
87
|
|
|
80
|
-
The redactor
|
|
88
|
+
The redactor is initialized with some defaults; in which all of the sensitive data will be redacted (request, response, query)
|
|
81
89
|
|
|
82
90
|
```javascript
|
|
83
91
|
import {AxiosErrorRedactor} from 'axios-error-redact'
|
|
@@ -86,22 +94,22 @@ const redactor = new AxiosErrorRedactor()
|
|
|
86
94
|
|
|
87
95
|
```
|
|
88
96
|
|
|
89
|
-
The constructor also accepts
|
|
97
|
+
The constructor also accepts options to enable or disable these
|
|
90
98
|
|
|
91
99
|
```javascript
|
|
92
100
|
import {AxiosErrorRedactor} from 'axios-error-redact'
|
|
93
101
|
|
|
94
|
-
const redactor = new AxiosErrorRedactor(
|
|
95
|
-
false,
|
|
96
|
-
false,
|
|
97
|
-
false,
|
|
98
|
-
|
|
102
|
+
const redactor = new AxiosErrorRedactor({
|
|
103
|
+
redactRequestDataEnabled: false,
|
|
104
|
+
redactResponseDataEnabled: false,
|
|
105
|
+
redactQueryDataEnabled: false,
|
|
106
|
+
})
|
|
99
107
|
|
|
100
108
|
```
|
|
101
109
|
|
|
102
110
|
### Main Function
|
|
103
111
|
|
|
104
|
-
The main function that can be called on the initiated object is `redactError` which accepts the error as the input and returns the redacted information in
|
|
112
|
+
The main function that can be called on the initiated object is `redactError` which accepts the error as the input and returns the redacted information in an object of type `HttpErrorResponse`
|
|
105
113
|
|
|
106
114
|
```javascript
|
|
107
115
|
import axios from 'axios'
|
|
@@ -121,7 +129,9 @@ There are three functions that can be used and chained after the initiated objec
|
|
|
121
129
|
```javascript
|
|
122
130
|
import {AxiosErrorRedactor} from 'axios-error-redact'
|
|
123
131
|
|
|
124
|
-
const redactor = new AxiosErrorRedactor()
|
|
132
|
+
const redactor = new AxiosErrorRedactor()
|
|
133
|
+
.skipRequestData()
|
|
134
|
+
.skipQueryData()
|
|
125
135
|
|
|
126
136
|
```
|
|
127
137
|
|
|
@@ -131,6 +141,7 @@ The redact library will extract information from axios error and return an objec
|
|
|
131
141
|
|
|
132
142
|
```javascript
|
|
133
143
|
HttpErrorResponse {
|
|
144
|
+
isErrorRedactedResponse: true;
|
|
134
145
|
message: string;
|
|
135
146
|
fullURL: string;
|
|
136
147
|
response: {
|
|
@@ -148,3 +159,22 @@ HttpErrorResponse {
|
|
|
148
159
|
```
|
|
149
160
|
|
|
150
161
|
If the error is not an axios error, then the same error will be returned.
|
|
162
|
+
|
|
163
|
+
### Type guard
|
|
164
|
+
|
|
165
|
+
The `isHttpErrorResponse()` function can be used as a type guard in TypeScript to narrow the error type.
|
|
166
|
+
|
|
167
|
+
This can be useful when multiple error types can be thrown from the try block.
|
|
168
|
+
|
|
169
|
+
Be sure not to use the `isAxiosError()` type guard provided by Axios since all intercepted Axios errors will be transformed into a `HttpErrorResponse`
|
|
170
|
+
|
|
171
|
+
```typescript
|
|
172
|
+
try {
|
|
173
|
+
...
|
|
174
|
+
} catch(error: unknown) {
|
|
175
|
+
if (isHttpErrorResponse(error)) {
|
|
176
|
+
// error is narrowed to type HttpErrorResponse
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
```
|
|
180
|
+
|
package/lib/index.d.ts
CHANGED
|
@@ -1,29 +1,51 @@
|
|
|
1
1
|
import { AxiosError } from 'axios';
|
|
2
|
+
import { AxiosErrorRedactorOptions, HttpErrorResponse } from './types';
|
|
3
|
+
export * from './types';
|
|
2
4
|
export declare const redactedKeyword = "<REDACTED>";
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
response: {
|
|
7
|
-
statusCode?: number;
|
|
8
|
-
statusMessage: string;
|
|
9
|
-
data: any;
|
|
10
|
-
};
|
|
11
|
-
request: {
|
|
12
|
-
baseURL: string;
|
|
13
|
-
path: string;
|
|
14
|
-
method: string;
|
|
15
|
-
data: any;
|
|
16
|
-
};
|
|
17
|
-
}
|
|
5
|
+
/**
|
|
6
|
+
* This class is used to redact sensitive data from Axios error objects.
|
|
7
|
+
*/
|
|
18
8
|
export declare class AxiosErrorRedactor {
|
|
19
|
-
redactRequestData
|
|
20
|
-
redactResponseData
|
|
21
|
-
redactQueryData
|
|
22
|
-
constructor(
|
|
9
|
+
private redactRequestData;
|
|
10
|
+
private redactResponseData;
|
|
11
|
+
private redactQueryData;
|
|
12
|
+
constructor(options?: AxiosErrorRedactorOptions);
|
|
13
|
+
/**
|
|
14
|
+
* Disables redaction of the request data
|
|
15
|
+
* @returns the instance of the redactor
|
|
16
|
+
*/
|
|
23
17
|
skipRequestData(): AxiosErrorRedactor;
|
|
18
|
+
/**
|
|
19
|
+
* Disables redaction of the response data
|
|
20
|
+
* @returns the instance of the redactor
|
|
21
|
+
*/
|
|
24
22
|
skipResponseData(): AxiosErrorRedactor;
|
|
23
|
+
/**
|
|
24
|
+
* Disables redaction of the query data
|
|
25
|
+
* @returns the instance of the redactor
|
|
26
|
+
*/
|
|
25
27
|
skipQueryData(): AxiosErrorRedactor;
|
|
28
|
+
/**
|
|
29
|
+
* redacts query string from the url
|
|
30
|
+
* @param url raw url
|
|
31
|
+
* @returns redacted query string
|
|
32
|
+
*/
|
|
26
33
|
private redactUrlQueryParams;
|
|
34
|
+
/**
|
|
35
|
+
* Redacts sensitive data from the Axios rejection error
|
|
36
|
+
* @param error any of errors that can be thrown by axios
|
|
37
|
+
* @returns HttpErrorResponse in case of axios error, otherwise passthrough the error
|
|
38
|
+
*/
|
|
27
39
|
redactError(error: AxiosError | null | undefined): (HttpErrorResponse | null | undefined | Error);
|
|
28
40
|
}
|
|
29
|
-
|
|
41
|
+
/**
|
|
42
|
+
* Simple factory function to create an error interceptor for axios
|
|
43
|
+
* @returns error interceptor for axios
|
|
44
|
+
*/
|
|
45
|
+
export declare function createErrorInterceptor(): ((error: AxiosError | null | undefined) => Promise<HttpErrorResponse | null | undefined | Error>);
|
|
46
|
+
/**
|
|
47
|
+
* predicate to check if the input is an HttpErrorResponse
|
|
48
|
+
* @param input any input
|
|
49
|
+
* @returns whether the input is an HttpErrorResponse
|
|
50
|
+
*/
|
|
51
|
+
export declare function isHttpErrorResponse(input: any): input is HttpErrorResponse;
|
package/lib/index.js
CHANGED
|
@@ -1,16 +1,44 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
2
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
17
|
exports.AxiosErrorRedactor = exports.redactedKeyword = void 0;
|
|
4
|
-
exports.
|
|
18
|
+
exports.createErrorInterceptor = createErrorInterceptor;
|
|
19
|
+
exports.isHttpErrorResponse = isHttpErrorResponse;
|
|
20
|
+
__exportStar(require("./types"), exports);
|
|
5
21
|
exports.redactedKeyword = '<REDACTED>';
|
|
6
22
|
const queryParamsRegex = /(?<=\?|#)\S+/ig;
|
|
7
23
|
const pathParamsRegex = /(\?|#)\S+/ig;
|
|
24
|
+
/**
|
|
25
|
+
* construct the full url
|
|
26
|
+
* @param base base url
|
|
27
|
+
* @param path sub path
|
|
28
|
+
* @param queryPath query path if exists
|
|
29
|
+
* @returns full url
|
|
30
|
+
*/
|
|
8
31
|
function joinURL(base, path, queryPath = '') {
|
|
9
32
|
if (!base)
|
|
10
33
|
return `${path}${queryPath}`;
|
|
11
34
|
const joint = base.endsWith('/') || path.startsWith('/') ? '' : '/';
|
|
12
35
|
return `${base}${joint}${path}${queryPath}`;
|
|
13
36
|
}
|
|
37
|
+
/**
|
|
38
|
+
* extracts query path parameters
|
|
39
|
+
* @param input full url
|
|
40
|
+
* @returns query path parameters if found, otherwise empty string
|
|
41
|
+
*/
|
|
14
42
|
function extractQueryPath(input) {
|
|
15
43
|
var _a;
|
|
16
44
|
if (!input)
|
|
@@ -18,6 +46,11 @@ function extractQueryPath(input) {
|
|
|
18
46
|
const match = (_a = input.match(pathParamsRegex)) === null || _a === void 0 ? void 0 : _a.pop();
|
|
19
47
|
return match || '';
|
|
20
48
|
}
|
|
49
|
+
/**
|
|
50
|
+
* tries to json parse the input
|
|
51
|
+
* @param input any input
|
|
52
|
+
* @returns parsed data if possible, otherwise undefined
|
|
53
|
+
*/
|
|
21
54
|
function parseData(input) {
|
|
22
55
|
try {
|
|
23
56
|
return JSON.parse(input);
|
|
@@ -26,6 +59,12 @@ function parseData(input) {
|
|
|
26
59
|
return;
|
|
27
60
|
}
|
|
28
61
|
}
|
|
62
|
+
/**
|
|
63
|
+
* recursively redacts sensitive data from the object
|
|
64
|
+
* @param data data to redact
|
|
65
|
+
* @param flag whether to perform redaction
|
|
66
|
+
* @returns redacted data
|
|
67
|
+
*/
|
|
29
68
|
function redactData(data, flag) {
|
|
30
69
|
if (!data)
|
|
31
70
|
return data;
|
|
@@ -39,29 +78,55 @@ function redactData(data, flag) {
|
|
|
39
78
|
return redactData(parsedData, flag);
|
|
40
79
|
return flag ? exports.redactedKeyword : data;
|
|
41
80
|
}
|
|
81
|
+
/**
|
|
82
|
+
* This class is used to redact sensitive data from Axios error objects.
|
|
83
|
+
*/
|
|
42
84
|
class AxiosErrorRedactor {
|
|
43
|
-
constructor(
|
|
44
|
-
|
|
45
|
-
this.
|
|
46
|
-
this.
|
|
85
|
+
constructor(options) {
|
|
86
|
+
var _a, _b, _c;
|
|
87
|
+
this.redactQueryData = (_a = options === null || options === void 0 ? void 0 : options.redactQueryDataEnabled) !== null && _a !== void 0 ? _a : true;
|
|
88
|
+
this.redactRequestData = (_b = options === null || options === void 0 ? void 0 : options.redactRequestDataEnabled) !== null && _b !== void 0 ? _b : true;
|
|
89
|
+
this.redactResponseData = (_c = options === null || options === void 0 ? void 0 : options.redactResponseDataEnabled) !== null && _c !== void 0 ? _c : true;
|
|
47
90
|
}
|
|
91
|
+
/**
|
|
92
|
+
* Disables redaction of the request data
|
|
93
|
+
* @returns the instance of the redactor
|
|
94
|
+
*/
|
|
48
95
|
skipRequestData() {
|
|
49
96
|
this.redactRequestData = false;
|
|
50
97
|
return this;
|
|
51
98
|
}
|
|
99
|
+
/**
|
|
100
|
+
* Disables redaction of the response data
|
|
101
|
+
* @returns the instance of the redactor
|
|
102
|
+
*/
|
|
52
103
|
skipResponseData() {
|
|
53
104
|
this.redactResponseData = false;
|
|
54
105
|
return this;
|
|
55
106
|
}
|
|
107
|
+
/**
|
|
108
|
+
* Disables redaction of the query data
|
|
109
|
+
* @returns the instance of the redactor
|
|
110
|
+
*/
|
|
56
111
|
skipQueryData() {
|
|
57
112
|
this.redactQueryData = false;
|
|
58
113
|
return this;
|
|
59
114
|
}
|
|
115
|
+
/**
|
|
116
|
+
* redacts query string from the url
|
|
117
|
+
* @param url raw url
|
|
118
|
+
* @returns redacted query string
|
|
119
|
+
*/
|
|
60
120
|
redactUrlQueryParams(url) {
|
|
61
121
|
if (!url)
|
|
62
122
|
return '';
|
|
63
123
|
return this.redactQueryData ? url.replace(queryParamsRegex, exports.redactedKeyword) : url;
|
|
64
124
|
}
|
|
125
|
+
/**
|
|
126
|
+
* Redacts sensitive data from the Axios rejection error
|
|
127
|
+
* @param error any of errors that can be thrown by axios
|
|
128
|
+
* @returns HttpErrorResponse in case of axios error, otherwise passthrough the error
|
|
129
|
+
*/
|
|
65
130
|
redactError(error) {
|
|
66
131
|
var _a, _b, _c, _d, _e, _f, _g, _h;
|
|
67
132
|
if (!error || !error.isAxiosError)
|
|
@@ -71,6 +136,7 @@ class AxiosErrorRedactor {
|
|
|
71
136
|
const queryPath = extractQueryPath(path) ? '' : extractQueryPath((_c = error.request) === null || _c === void 0 ? void 0 : _c.path);
|
|
72
137
|
const fullURL = this.redactUrlQueryParams(joinURL(baseURL, path, queryPath));
|
|
73
138
|
return {
|
|
139
|
+
isErrorRedactedResponse: true,
|
|
74
140
|
fullURL,
|
|
75
141
|
message: error.message,
|
|
76
142
|
response: {
|
|
@@ -88,10 +154,23 @@ class AxiosErrorRedactor {
|
|
|
88
154
|
}
|
|
89
155
|
}
|
|
90
156
|
exports.AxiosErrorRedactor = AxiosErrorRedactor;
|
|
91
|
-
|
|
157
|
+
/**
|
|
158
|
+
* Simple factory function to create an error interceptor for axios
|
|
159
|
+
* @returns error interceptor for axios
|
|
160
|
+
*/
|
|
161
|
+
function createErrorInterceptor() {
|
|
92
162
|
const redactor = new AxiosErrorRedactor();
|
|
93
163
|
return function (error) {
|
|
94
164
|
return Promise.reject(redactor.redactError(error));
|
|
95
165
|
};
|
|
96
166
|
}
|
|
97
|
-
|
|
167
|
+
/**
|
|
168
|
+
* predicate to check if the input is an HttpErrorResponse
|
|
169
|
+
* @param input any input
|
|
170
|
+
* @returns whether the input is an HttpErrorResponse
|
|
171
|
+
*/
|
|
172
|
+
function isHttpErrorResponse(input) {
|
|
173
|
+
return typeof input === 'object' &&
|
|
174
|
+
Boolean(input === null || input === void 0 ? void 0 : input.isErrorRedactedResponse);
|
|
175
|
+
}
|
|
176
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import {AxiosError} from 'axios'\nimport {AxiosErrorRedactorOptions, HttpErrorResponse} from './types'\n\nexport * from './types'\n\nexport const redactedKeyword = '<REDACTED>'\n\nconst queryParamsRegex = /(?<=\\?|#)\\S+/ig\nconst pathParamsRegex = /(\\?|#)\\S+/ig\n\n/**\n * construct the full url\n * @param base base url\n * @param path sub path\n * @param queryPath query path if exists\n * @returns full url\n */\nfunction joinURL(base: string, path: string, queryPath = '') {\n  if (!base)\n    return `${path}${queryPath}`\n\n  const joint = base.endsWith('/') || path.startsWith('/') ? '' : '/'\n  return `${base}${joint}${path}${queryPath}`\n}\n\n/**\n * extracts query path parameters\n * @param input full url\n * @returns query path parameters if found, otherwise empty string\n */\nfunction extractQueryPath(input: string | undefined): string {\n  if (!input)\n    return ''\n\n  const match = input.match(pathParamsRegex)?.pop()\n  return match || ''\n}\n\n/**\n * tries to json parse the input\n * @param input any input\n * @returns parsed data if possible, otherwise undefined\n */\nfunction parseData(input: unknown): any {\n  try {\n    return JSON.parse(input as string)\n  } catch {\n    return\n  }\n}\n\n/**\n * recursively redacts sensitive data from the object\n * @param data data to redact\n * @param flag whether to perform redaction\n * @returns redacted data\n */\nfunction redactData(data: unknown, flag: boolean): unknown {\n  if (!data)\n    return data\n\n\n  if (typeof data === 'object') {\n    if (Array.isArray(data))\n      return data.map(value => redactData(value, flag))\n\n\n    return Object.fromEntries(Object.entries(data).map(([key, value])=> [key, redactData(value, flag)]))\n  }\n\n  const parsedData = parseData(data)\n\n  if (parsedData && typeof parsedData === 'object')\n    return redactData(parsedData, flag)\n\n\n  return flag ? redactedKeyword : data\n}\n\n/**\n * This class is used to redact sensitive data from Axios error objects.\n */\nexport class AxiosErrorRedactor {\n  private redactRequestData: boolean\n  private redactResponseData: boolean\n  private redactQueryData: boolean\n\n  constructor(options?: AxiosErrorRedactorOptions) {\n    this.redactQueryData = options?.redactQueryDataEnabled ?? true\n    this.redactRequestData = options?.redactRequestDataEnabled ?? true\n    this.redactResponseData = options?.redactResponseDataEnabled ?? true\n  }\n\n  /**\n   * Disables redaction of the request data\n   * @returns the instance of the redactor\n   */\n  skipRequestData(): AxiosErrorRedactor {\n    this.redactRequestData = false\n    return this\n  }\n\n  /**\n   * Disables redaction of the response data\n   * @returns the instance of the redactor\n   */\n  skipResponseData(): AxiosErrorRedactor {\n    this.redactResponseData = false\n    return this\n  }\n\n  /**\n   * Disables redaction of the query data\n   * @returns the instance of the redactor\n   */\n  skipQueryData(): AxiosErrorRedactor {\n    this.redactQueryData = false\n    return this\n  }\n\n  /**\n   * redacts query string from the url\n   * @param url raw url\n   * @returns redacted query string\n   */\n  private redactUrlQueryParams(url: string | undefined): string {\n    if (!url)\n      return ''\n\n    return this.redactQueryData ? url.replace(queryParamsRegex, redactedKeyword) : url\n  }\n\n  /**\n   * Redacts sensitive data from the Axios rejection error\n   * @param error any of errors that can be thrown by axios\n   * @returns HttpErrorResponse in case of axios error, otherwise passthrough the error\n   */\n  redactError(error: AxiosError | null | undefined): (HttpErrorResponse | null | undefined | Error) {\n    if (!error || !error.isAxiosError)\n      return error\n\n    const baseURL = this.redactUrlQueryParams(error.config?.baseURL)\n    const path = this.redactUrlQueryParams(error.config?.url)\n    const queryPath = extractQueryPath(path) ? '' : extractQueryPath(error.request?.path)\n    const fullURL = this.redactUrlQueryParams(joinURL(baseURL, path, queryPath))\n\n    return {\n      isErrorRedactedResponse: true,\n      fullURL,\n      message: error.message,\n      response: {\n        statusCode: error.response?.status,\n        statusMessage: error.response?.statusText || '',\n        data: redactData(error.response?.data, this.redactResponseData),\n      },\n      request: {\n        baseURL,\n        path,\n        method: error.config?.method || '',\n        data: redactData(error.config?.data, this.redactRequestData),\n      },\n    }\n  }\n}\n\n/**\n * Simple factory function to create an error interceptor for axios\n * @returns error interceptor for axios\n */\nexport function createErrorInterceptor(): ((error: AxiosError | null | undefined)=> Promise<HttpErrorResponse | null | undefined | Error>) {\n  const redactor = new AxiosErrorRedactor()\n\n  return function (error: AxiosError | null | undefined): Promise<HttpErrorResponse | null | undefined | Error> {\n    return Promise.reject(redactor.redactError(error))\n  }\n}\n\n/**\n * predicate to check if the input is an HttpErrorResponse\n * @param input any input\n * @returns whether the input is an HttpErrorResponse\n */\nexport function isHttpErrorResponse(input: any): input is HttpErrorResponse {\n  return typeof input === 'object' &&\n    Boolean(input?.isErrorRedactedResponse)\n}\n"]}
|
package/lib/types.d.ts
ADDED
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
export interface AxiosErrorRedactorOptions {
|
|
2
|
+
/**
|
|
3
|
+
* whether to redact request data
|
|
4
|
+
* @default true
|
|
5
|
+
*/
|
|
6
|
+
readonly redactRequestDataEnabled?: boolean;
|
|
7
|
+
/**
|
|
8
|
+
* whether to redact response data
|
|
9
|
+
* @default true
|
|
10
|
+
*/
|
|
11
|
+
readonly redactResponseDataEnabled?: boolean;
|
|
12
|
+
/**
|
|
13
|
+
* whether to redact query data
|
|
14
|
+
* @default true
|
|
15
|
+
*/
|
|
16
|
+
readonly redactQueryDataEnabled?: boolean;
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Redacted response of Axios error
|
|
20
|
+
*/
|
|
21
|
+
export interface HttpErrorResponse {
|
|
22
|
+
/**
|
|
23
|
+
* whether the response is redacted
|
|
24
|
+
*/
|
|
25
|
+
readonly isErrorRedactedResponse: true;
|
|
26
|
+
/**
|
|
27
|
+
* error message
|
|
28
|
+
*/
|
|
29
|
+
readonly message: string;
|
|
30
|
+
/**
|
|
31
|
+
* full url of the request
|
|
32
|
+
*/
|
|
33
|
+
readonly fullURL: string;
|
|
34
|
+
/**
|
|
35
|
+
* response details
|
|
36
|
+
*/
|
|
37
|
+
readonly response: {
|
|
38
|
+
/**
|
|
39
|
+
* status code of the response
|
|
40
|
+
*/
|
|
41
|
+
readonly statusCode?: number;
|
|
42
|
+
/**
|
|
43
|
+
* status message of the response
|
|
44
|
+
*/
|
|
45
|
+
readonly statusMessage: string;
|
|
46
|
+
/**
|
|
47
|
+
* response data, redacted if set to true
|
|
48
|
+
*/
|
|
49
|
+
readonly data: unknown;
|
|
50
|
+
};
|
|
51
|
+
readonly request: {
|
|
52
|
+
/**
|
|
53
|
+
* base url of the request
|
|
54
|
+
*/
|
|
55
|
+
readonly baseURL: string;
|
|
56
|
+
/**
|
|
57
|
+
* path of the request
|
|
58
|
+
*/
|
|
59
|
+
readonly path: string;
|
|
60
|
+
/**
|
|
61
|
+
* method of the request
|
|
62
|
+
*/
|
|
63
|
+
readonly method: string;
|
|
64
|
+
/**
|
|
65
|
+
* request data, redacted if set to true
|
|
66
|
+
*/
|
|
67
|
+
readonly data: unknown;
|
|
68
|
+
};
|
|
69
|
+
}
|
package/lib/types.js
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgQXhpb3NFcnJvclJlZGFjdG9yT3B0aW9ucyB7XG4gIC8qKlxuICAgKiB3aGV0aGVyIHRvIHJlZGFjdCByZXF1ZXN0IGRhdGFcbiAgICogQGRlZmF1bHQgdHJ1ZVxuICAgKi9cbiAgcmVhZG9ubHkgcmVkYWN0UmVxdWVzdERhdGFFbmFibGVkPzogYm9vbGVhbjtcblxuICAvKipcbiAgICogd2hldGhlciB0byByZWRhY3QgcmVzcG9uc2UgZGF0YVxuICAgKiBAZGVmYXVsdCB0cnVlXG4gICAqL1xuICByZWFkb25seSByZWRhY3RSZXNwb25zZURhdGFFbmFibGVkPzogYm9vbGVhbjtcblxuICAvKipcbiAgICogd2hldGhlciB0byByZWRhY3QgcXVlcnkgZGF0YVxuICAgKiBAZGVmYXVsdCB0cnVlXG4gICAqL1xuICByZWFkb25seSByZWRhY3RRdWVyeURhdGFFbmFibGVkPzogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBSZWRhY3RlZCByZXNwb25zZSBvZiBBeGlvcyBlcnJvclxuICovXG5leHBvcnQgaW50ZXJmYWNlIEh0dHBFcnJvclJlc3BvbnNlIHtcbiAgLyoqXG4gICAqIHdoZXRoZXIgdGhlIHJlc3BvbnNlIGlzIHJlZGFjdGVkXG4gICAqL1xuICByZWFkb25seSBpc0Vycm9yUmVkYWN0ZWRSZXNwb25zZTogdHJ1ZTtcblxuICAvKipcbiAgICogZXJyb3IgbWVzc2FnZVxuICAgKi9cbiAgcmVhZG9ubHkgbWVzc2FnZTogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBmdWxsIHVybCBvZiB0aGUgcmVxdWVzdFxuICAgKi9cbiAgcmVhZG9ubHkgZnVsbFVSTDogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiByZXNwb25zZSBkZXRhaWxzXG4gICAqL1xuICByZWFkb25seSByZXNwb25zZToge1xuXG4gICAgLyoqXG4gICAgICogc3RhdHVzIGNvZGUgb2YgdGhlIHJlc3BvbnNlXG4gICAgICovXG4gICAgcmVhZG9ubHkgc3RhdHVzQ29kZT86IG51bWJlcjtcblxuICAgIC8qKlxuICAgICAqIHN0YXR1cyBtZXNzYWdlIG9mIHRoZSByZXNwb25zZVxuICAgICAqL1xuICAgIHJlYWRvbmx5IHN0YXR1c01lc3NhZ2U6IHN0cmluZztcblxuICAgIC8qKlxuICAgICAqIHJlc3BvbnNlIGRhdGEsIHJlZGFjdGVkIGlmIHNldCB0byB0cnVlXG4gICAgICovXG4gICAgcmVhZG9ubHkgZGF0YTogdW5rbm93bjtcbiAgfTtcbiAgcmVhZG9ubHkgcmVxdWVzdDoge1xuICAgIC8qKlxuICAgICAqIGJhc2UgdXJsIG9mIHRoZSByZXF1ZXN0XG4gICAgICovXG4gICAgcmVhZG9ubHkgYmFzZVVSTDogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogcGF0aCBvZiB0aGUgcmVxdWVzdFxuICAgICAqL1xuICAgIHJlYWRvbmx5IHBhdGg6IHN0cmluZztcblxuICAgIC8qKlxuICAgICAqIG1ldGhvZCBvZiB0aGUgcmVxdWVzdFxuICAgICAqL1xuICAgIHJlYWRvbmx5IG1ldGhvZDogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogcmVxdWVzdCBkYXRhLCByZWRhY3RlZCBpZiBzZXQgdG8gdHJ1ZVxuICAgICAqL1xuICAgIHJlYWRvbmx5IGRhdGE6IHVua25vd247XG4gIH07XG59Il19
|
package/package.json
CHANGED
|
@@ -46,7 +46,7 @@
|
|
|
46
46
|
"eslint-import-resolver-typescript": "^3.6.3",
|
|
47
47
|
"eslint-plugin-import": "^2.30.0",
|
|
48
48
|
"mocha": "^10.7.3",
|
|
49
|
-
"projen": "^0.87.
|
|
49
|
+
"projen": "^0.87.4",
|
|
50
50
|
"ts-eager": "^2.0.2",
|
|
51
51
|
"ts-node": "^10.9.2",
|
|
52
52
|
"typedoc": "^0.26.7",
|
|
@@ -72,7 +72,7 @@
|
|
|
72
72
|
"publishConfig": {
|
|
73
73
|
"access": "public"
|
|
74
74
|
},
|
|
75
|
-
"version": "0.
|
|
75
|
+
"version": "1.0.0",
|
|
76
76
|
"types": "lib/index.d.ts",
|
|
77
77
|
"//": "~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\"."
|
|
78
78
|
}
|