UNPKG

axios-error-redact

Version:

Library to redact sensitive information from Axios errors

183 lines 19.1 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __exportStar = (this && this.__exportStar) || function(m, exports) { for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p); }; Object.defineProperty(exports, "__esModule", { value: true }); exports.AxiosErrorRedactor = exports.redactedKeyword = void 0; exports.createErrorInterceptor = createErrorInterceptor; exports.isHttpErrorResponse = isHttpErrorResponse; __exportStar(require("./types"), exports); exports.redactedKeyword = '<REDACTED>'; const queryParamsRegex = /(?<=\?|#)\S+/ig; const pathParamsRegex = /(\?|#)\S+/ig; /** * construct the full url * @param base base url * @param path sub path * @param queryPath query path if exists * @returns full url */ function joinURL(base, path, queryPath = '') { if (!base) { return `${path}${queryPath}`; } const joint = base.endsWith('/') || path.startsWith('/') ? '' : '/'; return `${base}${joint}${path}${queryPath}`; } /** * extracts query path parameters * @param input full url * @returns query path parameters if found, otherwise empty string */ function extractQueryPath(input) { var _a; if (!input) { return ''; } const match = (_a = input.match(pathParamsRegex)) === null || _a === void 0 ? void 0 : _a.pop(); return match || ''; } /** * tries to json parse the input * @param input any input * @returns parsed data if possible, otherwise undefined */ function parseData(input) { try { return JSON.parse(input); } catch { return; } } /** * recursively redacts sensitive data from the object * @param data data to redact * @param flag whether to perform redaction * @returns redacted data */ function redactData(data, flag) { if (!data) { return data; } if (typeof data === 'object') { if (Array.isArray(data)) { return data.map(value => redactData(value, flag)); } return Object.fromEntries(Object.entries(data).map(([key, value]) => [key, redactData(value, flag)])); } const parsedData = parseData(data); if (parsedData && typeof parsedData === 'object') { return redactData(parsedData, flag); } return flag ? exports.redactedKeyword : data; } /** * This class is used to redact sensitive data from Axios error objects. */ class AxiosErrorRedactor { constructor(options) { var _a, _b, _c; this.redactQueryData = (_a = options === null || options === void 0 ? void 0 : options.redactQueryDataEnabled) !== null && _a !== void 0 ? _a : true; this.redactRequestData = (_b = options === null || options === void 0 ? void 0 : options.redactRequestDataEnabled) !== null && _b !== void 0 ? _b : true; this.redactResponseData = (_c = options === null || options === void 0 ? void 0 : options.redactResponseDataEnabled) !== null && _c !== void 0 ? _c : true; } /** * Disables redaction of the request data * @returns the instance of the redactor */ skipRequestData() { this.redactRequestData = false; return this; } /** * Disables redaction of the response data * @returns the instance of the redactor */ skipResponseData() { this.redactResponseData = false; return this; } /** * Disables redaction of the query data * @returns the instance of the redactor */ skipQueryData() { this.redactQueryData = false; return this; } /** * redacts query string from the url * @param url raw url * @returns redacted query string */ redactUrlQueryParams(url) { if (!url) { return ''; } return this.redactQueryData ? url.replace(queryParamsRegex, exports.redactedKeyword) : url; } /** * Redacts sensitive data from the Axios rejection error * @param error any of errors that can be thrown by axios * @returns HttpErrorResponse in case of axios error, otherwise passthrough the error */ redactError(error) { var _a, _b, _c, _d, _e, _f, _g, _h; if (!error || !error.isAxiosError) { return error; } const baseURL = this.redactUrlQueryParams((_a = error.config) === null || _a === void 0 ? void 0 : _a.baseURL); const path = this.redactUrlQueryParams((_b = error.config) === null || _b === void 0 ? void 0 : _b.url); const queryPath = extractQueryPath(path) ? '' : extractQueryPath((_c = error.request) === null || _c === void 0 ? void 0 : _c.path); const fullURL = this.redactUrlQueryParams(joinURL(baseURL, path, queryPath)); return { isErrorRedactedResponse: true, fullURL, message: error.message, response: { statusCode: (_d = error.response) === null || _d === void 0 ? void 0 : _d.status, statusMessage: ((_e = error.response) === null || _e === void 0 ? void 0 : _e.statusText) || '', data: redactData((_f = error.response) === null || _f === void 0 ? void 0 : _f.data, this.redactResponseData), }, request: { baseURL, path, method: ((_g = error.config) === null || _g === void 0 ? void 0 : _g.method) || '', data: redactData((_h = error.config) === null || _h === void 0 ? void 0 : _h.data, this.redactRequestData), }, }; } } exports.AxiosErrorRedactor = AxiosErrorRedactor; /** * Simple factory function to create an error interceptor for axios * @returns error interceptor for axios */ function createErrorInterceptor() { const redactor = new AxiosErrorRedactor(); return function (error) { return Promise.reject(redactor.redactError(error)); }; } /** * predicate to check if the input is an HttpErrorResponse * @param input any input * @returns whether the input is an HttpErrorResponse */ function isHttpErrorResponse(input) { return typeof input === 'object' && Boolean(input === null || input === void 0 ? void 0 : input.isErrorRedactedResponse); } //# 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) {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) {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) {return data;}\n\n\n  if (typeof data === 'object') {\n    if (Array.isArray(data)) {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') {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) {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) {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"]}