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 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 {getErrorInterceptor} from 'axios-error-redact'
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, getErrorInterceptor())
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 object can be initiated with some defaults; in which all of the sensitive data will be redacted (request, response, query)
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 boolean values to enable or disable these
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, //redactRequestData
96
- false, //redactResponseData
97
- false, //redactQueryData
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 and object of type `HttpErrorResponse`
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().skipRequestData().skipQueryData()
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
- export interface HttpErrorResponse {
4
- message: string;
5
- fullURL: string;
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: boolean;
20
- redactResponseData: boolean;
21
- redactQueryData: boolean;
22
- constructor(redactRequestData?: boolean, redactResponseData?: boolean, redactQueryData?: boolean);
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
- export declare function getErrorInterceptor(): ((error: AxiosError | null | undefined) => Promise<HttpErrorResponse | null | undefined | Error>);
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.getErrorInterceptor = getErrorInterceptor;
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(redactRequestData = true, redactResponseData = true, redactQueryData = true) {
44
- this.redactQueryData = redactQueryData;
45
- this.redactRequestData = redactRequestData;
46
- this.redactResponseData = redactResponseData;
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
- function getErrorInterceptor() {
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
- //# 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'\n\nexport const redactedKeyword = '<REDACTED>'\n\nconst queryParamsRegex = /(?<=\\?|#)\\S+/ig\nconst pathParamsRegex = /(\\?|#)\\S+/ig\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\nfunction extractQueryPath(input: string | undefined): string {\n  if (!input)\n    return ''\n\n  const match = input.match(pathParamsRegex)?.pop()\n  return match || ''\n}\n\nfunction parseData(input: string): any {\n  try {\n    return JSON.parse(input)\n  } catch {\n    return\n  }\n}\n\nfunction redactData(data: any, flag: boolean): any {\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\nexport interface HttpErrorResponse {\n  message: string;\n  fullURL: string;\n  response: {\n    statusCode?: number;\n    statusMessage: string;\n    data: any;\n  };\n  request: {\n    baseURL: string;\n    path: string;\n    method: string;\n    data: any;\n  };\n}\n\nexport class AxiosErrorRedactor {\n  redactRequestData: boolean\n  redactResponseData: boolean\n  redactQueryData: boolean\n\n  constructor(redactRequestData = true, redactResponseData = true, redactQueryData = true) {\n    this.redactQueryData = redactQueryData\n    this.redactRequestData = redactRequestData\n    this.redactResponseData = redactResponseData\n  }\n\n  skipRequestData(): AxiosErrorRedactor {\n    this.redactRequestData = false\n    return this\n  }\n\n  skipResponseData(): AxiosErrorRedactor {\n    this.redactResponseData = false\n    return this\n  }\n\n  skipQueryData(): AxiosErrorRedactor {\n    this.redactQueryData = false\n    return this\n  }\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  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      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\nexport function getErrorInterceptor(): ((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}"]}
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.3",
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.1.410",
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
  }