@logtape/redaction 2.2.0-dev.679 → 2.2.0-dev.683

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.
@@ -1,3 +1,4 @@
1
+ import { RedactionTraversalOptions } from "./traversal.cjs";
1
2
  import { ConsoleFormatter, TextFormatter } from "@logtape/logtape";
2
3
 
3
4
  //#region src/pattern.d.ts
@@ -51,6 +52,22 @@ declare const JWT_PATTERN: RedactionPattern;
51
52
  * @since 0.10.0
52
53
  */
53
54
  type RedactionPatterns = readonly RedactionPattern[];
55
+ /**
56
+ * Options for pattern-based redaction.
57
+ * @since 2.2.0
58
+ */
59
+ interface PatternRedactionOptions extends RedactionTraversalOptions {
60
+ /**
61
+ * Maximum recursion depth for object and array traversal.
62
+ * @default `20`
63
+ */
64
+ readonly maxDepth?: number;
65
+ /**
66
+ * Maximum number of properties or array elements to process per object.
67
+ * @default `1000`
68
+ */
69
+ readonly maxProperties?: number;
70
+ }
54
71
  /**
55
72
  * Applies data redaction to a {@link TextFormatter}.
56
73
  *
@@ -82,10 +99,11 @@ type RedactionPatterns = readonly RedactionPattern[];
82
99
  * ```
83
100
  * @param formatter The text formatter to apply redaction to.
84
101
  * @param patterns The redaction patterns to apply.
102
+ * @param options Options for bounding recursive traversal of formatter output.
85
103
  * @returns The redacted text formatter.
86
104
  * @since 0.10.0
87
105
  */
88
- declare function redactByPattern(formatter: TextFormatter, patterns: RedactionPatterns): TextFormatter;
106
+ declare function redactByPattern(formatter: TextFormatter, patterns: RedactionPatterns, options?: PatternRedactionOptions): TextFormatter;
89
107
  /**
90
108
  * Applies data redaction to a {@link ConsoleFormatter}.
91
109
  *
@@ -116,11 +134,12 @@ declare function redactByPattern(formatter: TextFormatter, patterns: RedactionPa
116
134
  * ```
117
135
  * @param formatter The console formatter to apply redaction to.
118
136
  * @param patterns The redaction patterns to apply.
137
+ * @param options Options for bounding recursive traversal of formatter output.
119
138
  * @returns The redacted console formatter.
120
139
  * @since 0.10.0
121
140
  */
122
- declare function redactByPattern(formatter: ConsoleFormatter, patterns: RedactionPatterns): ConsoleFormatter;
141
+ declare function redactByPattern(formatter: ConsoleFormatter, patterns: RedactionPatterns, options?: PatternRedactionOptions): ConsoleFormatter;
123
142
  //# sourceMappingURL=pattern.d.ts.map
124
143
  //#endregion
125
- export { CREDIT_CARD_NUMBER_PATTERN, EMAIL_ADDRESS_PATTERN, JWT_PATTERN, KR_RRN_PATTERN, RedactionPattern, RedactionPatterns, US_SSN_PATTERN, redactByPattern };
144
+ export { CREDIT_CARD_NUMBER_PATTERN, EMAIL_ADDRESS_PATTERN, JWT_PATTERN, KR_RRN_PATTERN, PatternRedactionOptions, RedactionPattern, RedactionPatterns, US_SSN_PATTERN, redactByPattern };
126
145
  //# sourceMappingURL=pattern.d.cts.map
@@ -1 +1 @@
1
- {"version":3,"file":"pattern.d.cts","names":[],"sources":["../src/pattern.ts"],"sourcesContent":[],"mappings":";;;;;;AAWA;AAsBA;AAUA;AASa,UAzCI,gBAAA,CAyCY;EAUhB;AASb;AASA;AAyDA;EAA+B,SAAA,OAAA,EAzHX,MAyHW;EAAA;;;AAGf;AAmChB;EAA+B,SAAA,WAAA,EAAA,MAAA,GAAA,CAAA,CAAA,KAAA,EAAA,MAAA,EAAA,GAAA,IAAA,EAAA,SAAA,GAAA,EAAA,EAAA,GAAA,MAAA,CAAA;;;;AAGZ;;cAjJN,uBAAuB;;;;;cAUvB,4BAA4B;;;;;cAS5B,gBAAgB;;;;;;cAUhB,gBAAgB;;;;;cAShB,aAAa;;;;;KASd,iBAAA,YAA6B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBAyDzB,eAAA,YACH,yBACD,oBACT;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBAmCa,eAAA,YACH,4BACD,oBACT"}
1
+ {"version":3,"file":"pattern.d.cts","names":[],"sources":["../src/pattern.ts"],"sourcesContent":[],"mappings":";;;;;;;AAuBA;AAsBA;AAUA;AASa,UAzCI,gBAAA,CAyCY;EAUhB;AASb;AASA;AAMA;EAsEgB,SAAA,OAAA,EA5II,MA4IW;EAAA;;;;;EAIf,SAAA,WAAA,EAAA,MAAA,GAAA,CAAA,CAAA,KAAA,EAAA,MAAA,EAAA,GAAA,IAAA,EAAA,SAAA,GAAA,EAAA,EAAA,GAAA,MAAA,CAAA;AAoChB;;;;;AAIG,cAvKU,qBAuKV,EAvKiC,gBAuKjC;AAAgB;;;;cA7JN,4BAA4B;;;;;cAS5B,gBAAgB;;;;;;cAUhB,gBAAgB;;;;;cAShB,aAAa;;;;;KASd,iBAAA,YAA6B;;;;;UAMxB,uBAAA,SAAgC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBAsEjC,eAAA,YACH,yBACD,6BACA,0BACT;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBAoCa,eAAA,YACH,4BACD,6BACA,0BACT"}
package/dist/pattern.d.ts CHANGED
@@ -1,3 +1,4 @@
1
+ import { RedactionTraversalOptions } from "./traversal.js";
1
2
  import { ConsoleFormatter, TextFormatter } from "@logtape/logtape";
2
3
 
3
4
  //#region src/pattern.d.ts
@@ -51,6 +52,22 @@ declare const JWT_PATTERN: RedactionPattern;
51
52
  * @since 0.10.0
52
53
  */
53
54
  type RedactionPatterns = readonly RedactionPattern[];
55
+ /**
56
+ * Options for pattern-based redaction.
57
+ * @since 2.2.0
58
+ */
59
+ interface PatternRedactionOptions extends RedactionTraversalOptions {
60
+ /**
61
+ * Maximum recursion depth for object and array traversal.
62
+ * @default `20`
63
+ */
64
+ readonly maxDepth?: number;
65
+ /**
66
+ * Maximum number of properties or array elements to process per object.
67
+ * @default `1000`
68
+ */
69
+ readonly maxProperties?: number;
70
+ }
54
71
  /**
55
72
  * Applies data redaction to a {@link TextFormatter}.
56
73
  *
@@ -82,10 +99,11 @@ type RedactionPatterns = readonly RedactionPattern[];
82
99
  * ```
83
100
  * @param formatter The text formatter to apply redaction to.
84
101
  * @param patterns The redaction patterns to apply.
102
+ * @param options Options for bounding recursive traversal of formatter output.
85
103
  * @returns The redacted text formatter.
86
104
  * @since 0.10.0
87
105
  */
88
- declare function redactByPattern(formatter: TextFormatter, patterns: RedactionPatterns): TextFormatter;
106
+ declare function redactByPattern(formatter: TextFormatter, patterns: RedactionPatterns, options?: PatternRedactionOptions): TextFormatter;
89
107
  /**
90
108
  * Applies data redaction to a {@link ConsoleFormatter}.
91
109
  *
@@ -116,11 +134,12 @@ declare function redactByPattern(formatter: TextFormatter, patterns: RedactionPa
116
134
  * ```
117
135
  * @param formatter The console formatter to apply redaction to.
118
136
  * @param patterns The redaction patterns to apply.
137
+ * @param options Options for bounding recursive traversal of formatter output.
119
138
  * @returns The redacted console formatter.
120
139
  * @since 0.10.0
121
140
  */
122
- declare function redactByPattern(formatter: ConsoleFormatter, patterns: RedactionPatterns): ConsoleFormatter;
141
+ declare function redactByPattern(formatter: ConsoleFormatter, patterns: RedactionPatterns, options?: PatternRedactionOptions): ConsoleFormatter;
123
142
  //# sourceMappingURL=pattern.d.ts.map
124
143
  //#endregion
125
- export { CREDIT_CARD_NUMBER_PATTERN, EMAIL_ADDRESS_PATTERN, JWT_PATTERN, KR_RRN_PATTERN, RedactionPattern, RedactionPatterns, US_SSN_PATTERN, redactByPattern };
144
+ export { CREDIT_CARD_NUMBER_PATTERN, EMAIL_ADDRESS_PATTERN, JWT_PATTERN, KR_RRN_PATTERN, PatternRedactionOptions, RedactionPattern, RedactionPatterns, US_SSN_PATTERN, redactByPattern };
126
145
  //# sourceMappingURL=pattern.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"pattern.d.ts","names":[],"sources":["../src/pattern.ts"],"sourcesContent":[],"mappings":";;;;;;AAWA;AAsBA;AAUA;AASa,UAzCI,gBAAA,CAyCY;EAUhB;AASb;AASA;AAyDA;EAA+B,SAAA,OAAA,EAzHX,MAyHW;EAAA;;;AAGf;AAmChB;EAA+B,SAAA,WAAA,EAAA,MAAA,GAAA,CAAA,CAAA,KAAA,EAAA,MAAA,EAAA,GAAA,IAAA,EAAA,SAAA,GAAA,EAAA,EAAA,GAAA,MAAA,CAAA;;;;AAGZ;;cAjJN,uBAAuB;;;;;cAUvB,4BAA4B;;;;;cAS5B,gBAAgB;;;;;;cAUhB,gBAAgB;;;;;cAShB,aAAa;;;;;KASd,iBAAA,YAA6B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBAyDzB,eAAA,YACH,yBACD,oBACT;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBAmCa,eAAA,YACH,4BACD,oBACT"}
1
+ {"version":3,"file":"pattern.d.ts","names":[],"sources":["../src/pattern.ts"],"sourcesContent":[],"mappings":";;;;;;;AAuBA;AAsBA;AAUA;AASa,UAzCI,gBAAA,CAyCY;EAUhB;AASb;AASA;AAMA;EAsEgB,SAAA,OAAA,EA5II,MA4IW;EAAA;;;;;EAIf,SAAA,WAAA,EAAA,MAAA,GAAA,CAAA,CAAA,KAAA,EAAA,MAAA,EAAA,GAAA,IAAA,EAAA,SAAA,GAAA,EAAA,EAAA,GAAA,MAAA,CAAA;AAoChB;;;;;AAIG,cAvKU,qBAuKV,EAvKiC,gBAuKjC;AAAgB;;;;cA7JN,4BAA4B;;;;;cAS5B,gBAAgB;;;;;;cAUhB,gBAAgB;;;;;cAShB,aAAa;;;;;KASd,iBAAA,YAA6B;;;;;UAMxB,uBAAA,SAAgC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBAsEjC,eAAA,YACH,yBACD,6BACA,0BACT;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBAoCa,eAAA,YACH,4BACD,6BACA,0BACT"}
package/dist/pattern.js CHANGED
@@ -1,4 +1,9 @@
1
+ import { createRedactionTraversalContext, redactionTruncatedValue } from "./traversal.js";
2
+ import { getLogger } from "@logtape/logtape";
3
+
1
4
  //#region src/pattern.ts
5
+ const metaLogger = getLogger(["logtape", "meta"]);
6
+ let reportingRedactionLimit = false;
2
7
  /**
3
8
  * A redaction pattern for email addresses.
4
9
  * @since 0.10.0
@@ -49,28 +54,46 @@ const JWT_PATTERN = {
49
54
  function isBuiltInObject(value) {
50
55
  return value instanceof Error || value instanceof Date || value instanceof RegExp || value instanceof Map || value instanceof Set || value instanceof WeakMap || value instanceof WeakSet || value instanceof Promise || value instanceof ArrayBuffer || typeof SharedArrayBuffer !== "undefined" && value instanceof SharedArrayBuffer || ArrayBuffer.isView(value);
51
56
  }
52
- function redactByPattern(formatter, patterns) {
57
+ function redactByPattern(formatter, patterns, options = {}) {
53
58
  for (const { pattern } of patterns) if (!pattern.global) throw new TypeError(`Pattern ${pattern} does not have the global flag set.`);
54
59
  function replaceString(str) {
55
60
  for (const p of patterns) str = typeof p.replacement === "string" ? str.replaceAll(p.pattern, p.replacement) : str.replaceAll(p.pattern, p.replacement);
56
61
  return str;
57
62
  }
58
- function replaceObject(object, visited) {
63
+ function replaceObject(object, context, depth) {
59
64
  if (typeof object === "object" && object !== null) {
60
- if (visited.has(object)) return visited.get(object);
65
+ if (context.visited.has(object)) return context.visited.get(object);
61
66
  }
62
67
  if (typeof object === "string") return replaceString(object);
63
68
  if (Array.isArray(object)) {
64
69
  const copy = [];
65
- visited.set(object, copy);
66
- object.forEach((item) => copy.push(replaceObject(item, visited)));
70
+ const length = Math.min(object.length, context.limits.maxProperties);
71
+ copy.length = length;
72
+ context.visited.set(object, copy);
73
+ if (object.length > context.limits.maxProperties) reportLimitOnce(context, "maxProperties");
74
+ for (let i = 0; i < length; i++) {
75
+ if (!(i in object)) continue;
76
+ const item = object[i];
77
+ if (shouldTraverseValue(item) && depth + 1 > context.limits.maxDepth) {
78
+ reportLimitOnce(context, "maxDepth");
79
+ copy[i] = redactionTruncatedValue;
80
+ } else copy[i] = replaceObject(item, context, depth + 1);
81
+ }
67
82
  return copy;
68
83
  }
69
84
  if (typeof object === "object" && object !== null) {
70
85
  if (isBuiltInObject(object)) return object;
71
86
  const redacted = {};
72
- visited.set(object, redacted);
73
- for (const key in object) if (Object.prototype.hasOwnProperty.call(object, key)) redacted[key] = replaceObject(object[key], visited);
87
+ context.visited.set(object, redacted);
88
+ const keys = Object.keys(object);
89
+ if (keys.length > context.limits.maxProperties) reportLimitOnce(context, "maxProperties");
90
+ for (const key of keys.slice(0, context.limits.maxProperties)) {
91
+ const value = object[key];
92
+ if (shouldTraverseValue(value) && depth + 1 > context.limits.maxDepth) {
93
+ reportLimitOnce(context, "maxDepth");
94
+ redacted[key] = redactionTruncatedValue;
95
+ } else redacted[key] = replaceObject(value, context, depth + 1);
96
+ }
74
97
  return redacted;
75
98
  }
76
99
  return object;
@@ -78,9 +101,33 @@ function redactByPattern(formatter, patterns) {
78
101
  return (record) => {
79
102
  const output = formatter(record);
80
103
  if (typeof output === "string") return replaceString(output);
81
- return output.map((obj) => replaceObject(obj, /* @__PURE__ */ new Map()));
104
+ const context = createPatternRedactionContext(options);
105
+ if (output.length > context.limits.maxProperties) reportLimitOnce(context, "maxProperties");
106
+ return output.slice(0, context.limits.maxProperties).map((obj) => replaceObject(obj, context, 0));
82
107
  };
83
108
  }
109
+ function reportRedactionLimitExceeded(limit, limits) {
110
+ if (reportingRedactionLimit || typeof metaLogger.warn !== "function") return;
111
+ try {
112
+ reportingRedactionLimit = true;
113
+ metaLogger.warn("Redaction traversal exceeded {limit}; replacing or omitting remaining data to keep logging bounded.", {
114
+ limit,
115
+ ...limits
116
+ });
117
+ } catch {} finally {
118
+ reportingRedactionLimit = false;
119
+ }
120
+ }
121
+ function shouldTraverseValue(value) {
122
+ return typeof value === "object" && value !== null && !isBuiltInObject(value);
123
+ }
124
+ function createPatternRedactionContext(options) {
125
+ return createRedactionTraversalContext(options, reportRedactionLimitExceeded);
126
+ }
127
+ function reportLimitOnce(context, limit) {
128
+ if (context.exceededLimits.has(limit)) return;
129
+ context.reportLimitExceeded(limit);
130
+ }
84
131
 
85
132
  //#endregion
86
133
  export { CREDIT_CARD_NUMBER_PATTERN, EMAIL_ADDRESS_PATTERN, JWT_PATTERN, KR_RRN_PATTERN, US_SSN_PATTERN, redactByPattern };
@@ -1 +1 @@
1
- {"version":3,"file":"pattern.js","names":["EMAIL_ADDRESS_PATTERN: RedactionPattern","CREDIT_CARD_NUMBER_PATTERN: RedactionPattern","US_SSN_PATTERN: RedactionPattern","KR_RRN_PATTERN: RedactionPattern","JWT_PATTERN: RedactionPattern","value: object","formatter: TextFormatter | ConsoleFormatter","patterns: RedactionPatterns","str: string","object: unknown","visited: Map<object, object>","copy: unknown[]","redacted: Record<string, unknown>","record: LogRecord"],"sources":["../src/pattern.ts"],"sourcesContent":["import type {\n ConsoleFormatter,\n LogRecord,\n TextFormatter,\n} from \"@logtape/logtape\";\n\n/**\n * A redaction pattern, which is a pair of regular expression and replacement\n * string or function.\n * @since 0.10.0\n */\nexport interface RedactionPattern {\n /**\n * The regular expression to match against. Note that it must have the\n * `g` (global) flag set, otherwise it will throw a `TypeError`.\n */\n readonly pattern: RegExp;\n\n /**\n * The replacement string or function. If the replacement is a function,\n * it will be called with the matched string and any capture groups (the same\n * signature as `String.prototype.replaceAll()`).\n */\n readonly replacement:\n | string\n // deno-lint-ignore no-explicit-any\n | ((match: string, ...rest: readonly any[]) => string);\n}\n\n/**\n * A redaction pattern for email addresses.\n * @since 0.10.0\n */\nexport const EMAIL_ADDRESS_PATTERN: RedactionPattern = {\n pattern:\n /[\\p{L}0-9.!#$%&'*+/=?^_`{|}~-]+@[\\p{L}0-9](?:[\\p{L}0-9-]{0,61}[\\p{L}0-9])?(?:\\.[\\p{L}0-9](?:[\\p{L}0-9-]{0,61}[\\p{L}0-9])?)+/gu,\n replacement: \"REDACTED@EMAIL.ADDRESS\",\n};\n\n/**\n * A redaction pattern for credit card numbers (including American Express).\n * @since 0.10.0\n */\nexport const CREDIT_CARD_NUMBER_PATTERN: RedactionPattern = {\n pattern: /(?:\\d{4}-){2}(?:\\d{4}-\\d{4}|\\d{6})/g,\n replacement: \"XXXX-XXXX-XXXX-XXXX\",\n};\n\n/**\n * A redaction pattern for U.S. Social Security numbers.\n * @since 0.10.0\n */\nexport const US_SSN_PATTERN: RedactionPattern = {\n pattern: /\\d{3}-\\d{2}-\\d{4}/g,\n replacement: \"XXX-XX-XXXX\",\n};\n\n/**\n * A redaction pattern for South Korean resident registration numbers\n * (住民登錄番號).\n * @since 0.10.0\n */\nexport const KR_RRN_PATTERN: RedactionPattern = {\n pattern: /\\d{6}-\\d{7}/g,\n replacement: \"XXXXXX-XXXXXXX\",\n};\n\n/**\n * A redaction pattern for JSON Web Tokens (JWT).\n * @since 0.10.0\n */\nexport const JWT_PATTERN: RedactionPattern = {\n pattern: /eyJ[a-zA-Z0-9_-]*\\.[a-zA-Z0-9_-]*\\.[a-zA-Z0-9_-]*/g,\n replacement: \"[JWT REDACTED]\",\n};\n\n/**\n * A list of {@link RedactionPattern}s.\n * @since 0.10.0\n */\nexport type RedactionPatterns = readonly RedactionPattern[];\n\n/**\n * Checks if a value is a built-in object that should not be recursively\n * processed (e.g., Error, Date, RegExp, Map, Set, etc.).\n * @param value The value to check.\n * @returns `true` if the value is a built-in object, `false` otherwise.\n */\nfunction isBuiltInObject(value: object): boolean {\n return value instanceof Error ||\n value instanceof Date ||\n value instanceof RegExp ||\n value instanceof Map ||\n value instanceof Set ||\n value instanceof WeakMap ||\n value instanceof WeakSet ||\n value instanceof Promise ||\n value instanceof ArrayBuffer ||\n (typeof SharedArrayBuffer !== \"undefined\" &&\n value instanceof SharedArrayBuffer) ||\n ArrayBuffer.isView(value);\n}\n\n/**\n * Applies data redaction to a {@link TextFormatter}.\n *\n * Note that there are some built-in redaction patterns:\n *\n * - {@link CREDIT_CARD_NUMBER_PATTERN}\n * - {@link EMAIL_ADDRESS_PATTERN}\n * - {@link JWT_PATTERN}\n * - {@link KR_RRN_PATTERN}\n * - {@link US_SSN_PATTERN}\n *\n * @example\n * ```ts\n * import { getFileSink } from \"@logtape/file\";\n * import { getAnsiColorFormatter } from \"@logtape/logtape\";\n * import {\n * CREDIT_CARD_NUMBER_PATTERN,\n * EMAIL_ADDRESS_PATTERN,\n * JWT_PATTERN,\n * redactByPattern,\n * } from \"@logtape/redaction\";\n *\n * const formatter = redactByPattern(getAnsiConsoleFormatter(), [\n * CREDIT_CARD_NUMBER_PATTERN,\n * EMAIL_ADDRESS_PATTERN,\n * JWT_PATTERN,\n * ]);\n * const sink = getFileSink(\"my-app.log\", { formatter });\n * ```\n * @param formatter The text formatter to apply redaction to.\n * @param patterns The redaction patterns to apply.\n * @returns The redacted text formatter.\n * @since 0.10.0\n */\nexport function redactByPattern(\n formatter: TextFormatter,\n patterns: RedactionPatterns,\n): TextFormatter;\n\n/**\n * Applies data redaction to a {@link ConsoleFormatter}.\n *\n * Note that there are some built-in redaction patterns:\n *\n * - {@link CREDIT_CARD_NUMBER_PATTERN}\n * - {@link EMAIL_ADDRESS_PATTERN}\n * - {@link JWT_PATTERN}\n * - {@link KR_RRN_PATTERN}\n * - {@link US_SSN_PATTERN}\n *\n * @example\n * ```ts\n * import { defaultConsoleFormatter, getConsoleSink } from \"@logtape/logtape\";\n * import {\n * CREDIT_CARD_NUMBER_PATTERN,\n * EMAIL_ADDRESS_PATTERN,\n * JWT_PATTERN,\n * redactByPattern,\n * } from \"@logtape/redaction\";\n *\n * const formatter = redactByPattern(defaultConsoleFormatter, [\n * CREDIT_CARD_NUMBER_PATTERN,\n * EMAIL_ADDRESS_PATTERN,\n * JWT_PATTERN,\n * ]);\n * const sink = getConsoleSink({ formatter });\n * ```\n * @param formatter The console formatter to apply redaction to.\n * @param patterns The redaction patterns to apply.\n * @returns The redacted console formatter.\n * @since 0.10.0\n */\nexport function redactByPattern(\n formatter: ConsoleFormatter,\n patterns: RedactionPatterns,\n): ConsoleFormatter;\n\nexport function redactByPattern(\n formatter: TextFormatter | ConsoleFormatter,\n patterns: RedactionPatterns,\n): (record: LogRecord) => string | readonly unknown[] {\n for (const { pattern } of patterns) {\n if (!pattern.global) {\n throw new TypeError(\n `Pattern ${pattern} does not have the global flag set.`,\n );\n }\n }\n\n function replaceString(str: string): string {\n for (const p of patterns) {\n // The following ternary operator may seem strange, but it's for\n // making TypeScript happy:\n str = typeof p.replacement === \"string\"\n ? str.replaceAll(p.pattern, p.replacement)\n : str.replaceAll(p.pattern, p.replacement);\n }\n return str;\n }\n\n function replaceObject(\n object: unknown,\n visited: Map<object, object>,\n ): unknown {\n if (typeof object === \"object\" && object !== null) {\n if (visited.has(object)) {\n return visited.get(object)!; // Circular reference detected\n }\n }\n\n if (typeof object === \"string\") return replaceString(object);\n if (Array.isArray(object)) {\n const copy: unknown[] = [];\n visited.set(object, copy);\n object.forEach((item) => copy.push(replaceObject(item, visited)));\n return copy;\n }\n if (typeof object === \"object\" && object !== null) {\n if (isBuiltInObject(object)) {\n return object;\n }\n const redacted: Record<string, unknown> = {};\n visited.set(object, redacted);\n for (const key in object) {\n if (Object.prototype.hasOwnProperty.call(object, key)) {\n redacted[key] = replaceObject(\n (object as Record<string, unknown>)[key],\n visited,\n );\n }\n }\n return redacted;\n }\n return object;\n }\n\n return (record: LogRecord) => {\n const output = formatter(record);\n if (typeof output === \"string\") return replaceString(output);\n return output.map((obj) => replaceObject(obj, new Map()));\n };\n}\n"],"mappings":";;;;;AAiCA,MAAaA,wBAA0C;CACrD,SACE;CACF,aAAa;AACd;;;;;AAMD,MAAaC,6BAA+C;CAC1D,SAAS;CACT,aAAa;AACd;;;;;AAMD,MAAaC,iBAAmC;CAC9C,SAAS;CACT,aAAa;AACd;;;;;;AAOD,MAAaC,iBAAmC;CAC9C,SAAS;CACT,aAAa;AACd;;;;;AAMD,MAAaC,cAAgC;CAC3C,SAAS;CACT,aAAa;AACd;;;;;;;AAcD,SAAS,gBAAgBC,OAAwB;AAC/C,QAAO,iBAAiB,SACtB,iBAAiB,QACjB,iBAAiB,UACjB,iBAAiB,OACjB,iBAAiB,OACjB,iBAAiB,WACjB,iBAAiB,WACjB,iBAAiB,WACjB,iBAAiB,sBACT,sBAAsB,eAC5B,iBAAiB,qBACnB,YAAY,OAAO,MAAM;AAC5B;AA+ED,SAAgB,gBACdC,WACAC,UACoD;AACpD,MAAK,MAAM,EAAE,SAAS,IAAI,SACxB,MAAK,QAAQ,OACX,OAAM,IAAI,WACP,UAAU,QAAQ;CAKzB,SAAS,cAAcC,KAAqB;AAC1C,OAAK,MAAM,KAAK,SAGd,cAAa,EAAE,gBAAgB,WAC3B,IAAI,WAAW,EAAE,SAAS,EAAE,YAAY,GACxC,IAAI,WAAW,EAAE,SAAS,EAAE,YAAY;AAE9C,SAAO;CACR;CAED,SAAS,cACPC,QACAC,SACS;AACT,aAAW,WAAW,YAAY,WAAW,MAC3C;OAAI,QAAQ,IAAI,OAAO,CACrB,QAAO,QAAQ,IAAI,OAAO;EAC3B;AAGH,aAAW,WAAW,SAAU,QAAO,cAAc,OAAO;AAC5D,MAAI,MAAM,QAAQ,OAAO,EAAE;GACzB,MAAMC,OAAkB,CAAE;AAC1B,WAAQ,IAAI,QAAQ,KAAK;AACzB,UAAO,QAAQ,CAAC,SAAS,KAAK,KAAK,cAAc,MAAM,QAAQ,CAAC,CAAC;AACjE,UAAO;EACR;AACD,aAAW,WAAW,YAAY,WAAW,MAAM;AACjD,OAAI,gBAAgB,OAAO,CACzB,QAAO;GAET,MAAMC,WAAoC,CAAE;AAC5C,WAAQ,IAAI,QAAQ,SAAS;AAC7B,QAAK,MAAM,OAAO,OAChB,KAAI,OAAO,UAAU,eAAe,KAAK,QAAQ,IAAI,CACnD,UAAS,OAAO,cACb,OAAmC,MACpC,QACD;AAGL,UAAO;EACR;AACD,SAAO;CACR;AAED,QAAO,CAACC,WAAsB;EAC5B,MAAM,SAAS,UAAU,OAAO;AAChC,aAAW,WAAW,SAAU,QAAO,cAAc,OAAO;AAC5D,SAAO,OAAO,IAAI,CAAC,QAAQ,cAAc,qBAAK,IAAI,MAAM,CAAC;CAC1D;AACF"}
1
+ {"version":3,"file":"pattern.js","names":["EMAIL_ADDRESS_PATTERN: RedactionPattern","CREDIT_CARD_NUMBER_PATTERN: RedactionPattern","US_SSN_PATTERN: RedactionPattern","KR_RRN_PATTERN: RedactionPattern","JWT_PATTERN: RedactionPattern","value: object","formatter: TextFormatter | ConsoleFormatter","patterns: RedactionPatterns","options: PatternRedactionOptions","str: string","object: unknown","context: RedactionTraversalContext","depth: number","copy: unknown[]","redacted: Record<string, unknown>","record: LogRecord","limit: RedactionLimit","limits: RedactionTraversalLimits","value: unknown","options: RedactionTraversalOptions"],"sources":["../src/pattern.ts"],"sourcesContent":["import {\n type ConsoleFormatter,\n getLogger,\n type LogRecord,\n type TextFormatter,\n} from \"@logtape/logtape\";\nimport {\n createRedactionTraversalContext,\n type RedactionLimit,\n type RedactionTraversalContext,\n type RedactionTraversalLimits,\n type RedactionTraversalOptions,\n redactionTruncatedValue,\n} from \"./traversal.ts\";\n\nconst metaLogger = getLogger([\"logtape\", \"meta\"]);\nlet reportingRedactionLimit = false;\n\n/**\n * A redaction pattern, which is a pair of regular expression and replacement\n * string or function.\n * @since 0.10.0\n */\nexport interface RedactionPattern {\n /**\n * The regular expression to match against. Note that it must have the\n * `g` (global) flag set, otherwise it will throw a `TypeError`.\n */\n readonly pattern: RegExp;\n\n /**\n * The replacement string or function. If the replacement is a function,\n * it will be called with the matched string and any capture groups (the same\n * signature as `String.prototype.replaceAll()`).\n */\n readonly replacement:\n | string\n // deno-lint-ignore no-explicit-any\n | ((match: string, ...rest: readonly any[]) => string);\n}\n\n/**\n * A redaction pattern for email addresses.\n * @since 0.10.0\n */\nexport const EMAIL_ADDRESS_PATTERN: RedactionPattern = {\n pattern:\n /[\\p{L}0-9.!#$%&'*+/=?^_`{|}~-]+@[\\p{L}0-9](?:[\\p{L}0-9-]{0,61}[\\p{L}0-9])?(?:\\.[\\p{L}0-9](?:[\\p{L}0-9-]{0,61}[\\p{L}0-9])?)+/gu,\n replacement: \"REDACTED@EMAIL.ADDRESS\",\n};\n\n/**\n * A redaction pattern for credit card numbers (including American Express).\n * @since 0.10.0\n */\nexport const CREDIT_CARD_NUMBER_PATTERN: RedactionPattern = {\n pattern: /(?:\\d{4}-){2}(?:\\d{4}-\\d{4}|\\d{6})/g,\n replacement: \"XXXX-XXXX-XXXX-XXXX\",\n};\n\n/**\n * A redaction pattern for U.S. Social Security numbers.\n * @since 0.10.0\n */\nexport const US_SSN_PATTERN: RedactionPattern = {\n pattern: /\\d{3}-\\d{2}-\\d{4}/g,\n replacement: \"XXX-XX-XXXX\",\n};\n\n/**\n * A redaction pattern for South Korean resident registration numbers\n * (住民登錄番號).\n * @since 0.10.0\n */\nexport const KR_RRN_PATTERN: RedactionPattern = {\n pattern: /\\d{6}-\\d{7}/g,\n replacement: \"XXXXXX-XXXXXXX\",\n};\n\n/**\n * A redaction pattern for JSON Web Tokens (JWT).\n * @since 0.10.0\n */\nexport const JWT_PATTERN: RedactionPattern = {\n pattern: /eyJ[a-zA-Z0-9_-]*\\.[a-zA-Z0-9_-]*\\.[a-zA-Z0-9_-]*/g,\n replacement: \"[JWT REDACTED]\",\n};\n\n/**\n * A list of {@link RedactionPattern}s.\n * @since 0.10.0\n */\nexport type RedactionPatterns = readonly RedactionPattern[];\n\n/**\n * Options for pattern-based redaction.\n * @since 2.2.0\n */\nexport interface PatternRedactionOptions extends RedactionTraversalOptions {\n /**\n * Maximum recursion depth for object and array traversal.\n * @default `20`\n */\n readonly maxDepth?: number;\n\n /**\n * Maximum number of properties or array elements to process per object.\n * @default `1000`\n */\n readonly maxProperties?: number;\n}\n\n/**\n * Checks if a value is a built-in object that should not be recursively\n * processed (e.g., Error, Date, RegExp, Map, Set, etc.).\n * @param value The value to check.\n * @returns `true` if the value is a built-in object, `false` otherwise.\n */\nfunction isBuiltInObject(value: object): boolean {\n return value instanceof Error ||\n value instanceof Date ||\n value instanceof RegExp ||\n value instanceof Map ||\n value instanceof Set ||\n value instanceof WeakMap ||\n value instanceof WeakSet ||\n value instanceof Promise ||\n value instanceof ArrayBuffer ||\n (typeof SharedArrayBuffer !== \"undefined\" &&\n value instanceof SharedArrayBuffer) ||\n ArrayBuffer.isView(value);\n}\n\n/**\n * Applies data redaction to a {@link TextFormatter}.\n *\n * Note that there are some built-in redaction patterns:\n *\n * - {@link CREDIT_CARD_NUMBER_PATTERN}\n * - {@link EMAIL_ADDRESS_PATTERN}\n * - {@link JWT_PATTERN}\n * - {@link KR_RRN_PATTERN}\n * - {@link US_SSN_PATTERN}\n *\n * @example\n * ```ts\n * import { getFileSink } from \"@logtape/file\";\n * import { getAnsiColorFormatter } from \"@logtape/logtape\";\n * import {\n * CREDIT_CARD_NUMBER_PATTERN,\n * EMAIL_ADDRESS_PATTERN,\n * JWT_PATTERN,\n * redactByPattern,\n * } from \"@logtape/redaction\";\n *\n * const formatter = redactByPattern(getAnsiConsoleFormatter(), [\n * CREDIT_CARD_NUMBER_PATTERN,\n * EMAIL_ADDRESS_PATTERN,\n * JWT_PATTERN,\n * ]);\n * const sink = getFileSink(\"my-app.log\", { formatter });\n * ```\n * @param formatter The text formatter to apply redaction to.\n * @param patterns The redaction patterns to apply.\n * @param options Options for bounding recursive traversal of formatter output.\n * @returns The redacted text formatter.\n * @since 0.10.0\n */\nexport function redactByPattern(\n formatter: TextFormatter,\n patterns: RedactionPatterns,\n options?: PatternRedactionOptions,\n): TextFormatter;\n\n/**\n * Applies data redaction to a {@link ConsoleFormatter}.\n *\n * Note that there are some built-in redaction patterns:\n *\n * - {@link CREDIT_CARD_NUMBER_PATTERN}\n * - {@link EMAIL_ADDRESS_PATTERN}\n * - {@link JWT_PATTERN}\n * - {@link KR_RRN_PATTERN}\n * - {@link US_SSN_PATTERN}\n *\n * @example\n * ```ts\n * import { defaultConsoleFormatter, getConsoleSink } from \"@logtape/logtape\";\n * import {\n * CREDIT_CARD_NUMBER_PATTERN,\n * EMAIL_ADDRESS_PATTERN,\n * JWT_PATTERN,\n * redactByPattern,\n * } from \"@logtape/redaction\";\n *\n * const formatter = redactByPattern(defaultConsoleFormatter, [\n * CREDIT_CARD_NUMBER_PATTERN,\n * EMAIL_ADDRESS_PATTERN,\n * JWT_PATTERN,\n * ]);\n * const sink = getConsoleSink({ formatter });\n * ```\n * @param formatter The console formatter to apply redaction to.\n * @param patterns The redaction patterns to apply.\n * @param options Options for bounding recursive traversal of formatter output.\n * @returns The redacted console formatter.\n * @since 0.10.0\n */\nexport function redactByPattern(\n formatter: ConsoleFormatter,\n patterns: RedactionPatterns,\n options?: PatternRedactionOptions,\n): ConsoleFormatter;\n\nexport function redactByPattern(\n formatter: TextFormatter | ConsoleFormatter,\n patterns: RedactionPatterns,\n options: PatternRedactionOptions = {},\n): (record: LogRecord) => string | readonly unknown[] {\n for (const { pattern } of patterns) {\n if (!pattern.global) {\n throw new TypeError(\n `Pattern ${pattern} does not have the global flag set.`,\n );\n }\n }\n\n function replaceString(str: string): string {\n for (const p of patterns) {\n // The following ternary operator may seem strange, but it's for\n // making TypeScript happy:\n str = typeof p.replacement === \"string\"\n ? str.replaceAll(p.pattern, p.replacement)\n : str.replaceAll(p.pattern, p.replacement);\n }\n return str;\n }\n\n function replaceObject(\n object: unknown,\n context: RedactionTraversalContext,\n depth: number,\n ): unknown {\n if (typeof object === \"object\" && object !== null) {\n if (context.visited.has(object)) {\n return context.visited.get(object)!; // Circular reference detected\n }\n }\n\n if (typeof object === \"string\") return replaceString(object);\n if (Array.isArray(object)) {\n const copy: unknown[] = [];\n const length = Math.min(object.length, context.limits.maxProperties);\n copy.length = length;\n context.visited.set(object, copy);\n if (object.length > context.limits.maxProperties) {\n reportLimitOnce(context, \"maxProperties\");\n }\n for (let i = 0; i < length; i++) {\n if (!(i in object)) continue;\n const item = object[i];\n if (\n shouldTraverseValue(item) && depth + 1 > context.limits.maxDepth\n ) {\n reportLimitOnce(context, \"maxDepth\");\n copy[i] = redactionTruncatedValue;\n } else {\n copy[i] = replaceObject(item, context, depth + 1);\n }\n }\n return copy;\n }\n if (typeof object === \"object\" && object !== null) {\n if (isBuiltInObject(object)) {\n return object;\n }\n const redacted: Record<string, unknown> = {};\n context.visited.set(object, redacted);\n const keys = Object.keys(object);\n if (keys.length > context.limits.maxProperties) {\n reportLimitOnce(context, \"maxProperties\");\n }\n for (const key of keys.slice(0, context.limits.maxProperties)) {\n const value = (object as Record<string, unknown>)[key];\n if (\n shouldTraverseValue(value) && depth + 1 > context.limits.maxDepth\n ) {\n reportLimitOnce(context, \"maxDepth\");\n redacted[key] = redactionTruncatedValue;\n } else {\n redacted[key] = replaceObject(value, context, depth + 1);\n }\n }\n return redacted;\n }\n return object;\n }\n\n return (record: LogRecord) => {\n const output = formatter(record);\n if (typeof output === \"string\") return replaceString(output);\n const context = createPatternRedactionContext(options);\n if (output.length > context.limits.maxProperties) {\n reportLimitOnce(context, \"maxProperties\");\n }\n return output\n .slice(0, context.limits.maxProperties)\n .map((obj) => replaceObject(obj, context, 0));\n };\n}\n\nfunction reportRedactionLimitExceeded(\n limit: RedactionLimit,\n limits: RedactionTraversalLimits,\n): void {\n if (reportingRedactionLimit || typeof metaLogger.warn !== \"function\") {\n return;\n }\n try {\n reportingRedactionLimit = true;\n metaLogger.warn(\n \"Redaction traversal exceeded {limit}; replacing or omitting \" +\n \"remaining data to keep logging bounded.\",\n { limit, ...limits },\n );\n } catch {\n // Meta logging failures must not make normal logging fail.\n } finally {\n reportingRedactionLimit = false;\n }\n}\n\nfunction shouldTraverseValue(value: unknown): boolean {\n return (typeof value === \"object\" && value !== null) &&\n !isBuiltInObject(value);\n}\n\nfunction createPatternRedactionContext(\n options: RedactionTraversalOptions,\n): RedactionTraversalContext {\n return createRedactionTraversalContext(\n options,\n reportRedactionLimitExceeded,\n );\n}\n\nfunction reportLimitOnce(\n context: RedactionTraversalContext,\n limit: RedactionLimit,\n): void {\n if (context.exceededLimits.has(limit)) return;\n context.reportLimitExceeded(limit);\n}\n"],"mappings":"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"}
@@ -0,0 +1,30 @@
1
+
2
+ //#region src/traversal.ts
3
+ const redactionTruncatedValue = "[truncated]";
4
+ const defaultMaxDepth = 20;
5
+ const defaultMaxProperties = 1e3;
6
+ function createRedactionTraversalContext(options, reportLimitExceeded, visited = /* @__PURE__ */ new Map()) {
7
+ const limits = {
8
+ maxDepth: normalizeLimit(options.maxDepth, defaultMaxDepth),
9
+ maxProperties: normalizeLimit(options.maxProperties, defaultMaxProperties)
10
+ };
11
+ const exceededLimits = /* @__PURE__ */ new Set();
12
+ return {
13
+ limits,
14
+ visited,
15
+ exceededLimits,
16
+ reportLimitExceeded(limit) {
17
+ exceededLimits.add(limit);
18
+ reportLimitExceeded(limit, limits);
19
+ }
20
+ };
21
+ }
22
+ function normalizeLimit(value, defaultValue) {
23
+ if (value == null) return defaultValue;
24
+ if (!Number.isFinite(value) || value < 0) return defaultValue;
25
+ return Math.floor(value);
26
+ }
27
+
28
+ //#endregion
29
+ exports.createRedactionTraversalContext = createRedactionTraversalContext;
30
+ exports.redactionTruncatedValue = redactionTruncatedValue;
@@ -0,0 +1,20 @@
1
+ //#region src/traversal.d.ts
2
+ /**
3
+ * Options for bounding recursive redaction traversal.
4
+ * @since 2.2.0
5
+ */
6
+ interface RedactionTraversalOptions {
7
+ /**
8
+ * Maximum recursion depth for object and array traversal.
9
+ * @default `20`
10
+ */
11
+ readonly maxDepth?: number;
12
+ /**
13
+ * Maximum number of properties or array elements to process per object.
14
+ * @default `1000`
15
+ */
16
+ readonly maxProperties?: number;
17
+ }
18
+ //#endregion
19
+ export { RedactionTraversalOptions };
20
+ //# sourceMappingURL=traversal.d.cts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"traversal.d.cts","names":[],"sources":["../src/traversal.ts"],"sourcesContent":[],"mappings":";;AAIA;;;UAAiB,yBAAA"}
@@ -0,0 +1,20 @@
1
+ //#region src/traversal.d.ts
2
+ /**
3
+ * Options for bounding recursive redaction traversal.
4
+ * @since 2.2.0
5
+ */
6
+ interface RedactionTraversalOptions {
7
+ /**
8
+ * Maximum recursion depth for object and array traversal.
9
+ * @default `20`
10
+ */
11
+ readonly maxDepth?: number;
12
+ /**
13
+ * Maximum number of properties or array elements to process per object.
14
+ * @default `1000`
15
+ */
16
+ readonly maxProperties?: number;
17
+ }
18
+ //#endregion
19
+ export { RedactionTraversalOptions };
20
+ //# sourceMappingURL=traversal.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"traversal.d.ts","names":[],"sources":["../src/traversal.ts"],"sourcesContent":[],"mappings":";;AAIA;;;UAAiB,yBAAA"}
@@ -0,0 +1,29 @@
1
+ //#region src/traversal.ts
2
+ const redactionTruncatedValue = "[truncated]";
3
+ const defaultMaxDepth = 20;
4
+ const defaultMaxProperties = 1e3;
5
+ function createRedactionTraversalContext(options, reportLimitExceeded, visited = /* @__PURE__ */ new Map()) {
6
+ const limits = {
7
+ maxDepth: normalizeLimit(options.maxDepth, defaultMaxDepth),
8
+ maxProperties: normalizeLimit(options.maxProperties, defaultMaxProperties)
9
+ };
10
+ const exceededLimits = /* @__PURE__ */ new Set();
11
+ return {
12
+ limits,
13
+ visited,
14
+ exceededLimits,
15
+ reportLimitExceeded(limit) {
16
+ exceededLimits.add(limit);
17
+ reportLimitExceeded(limit, limits);
18
+ }
19
+ };
20
+ }
21
+ function normalizeLimit(value, defaultValue) {
22
+ if (value == null) return defaultValue;
23
+ if (!Number.isFinite(value) || value < 0) return defaultValue;
24
+ return Math.floor(value);
25
+ }
26
+
27
+ //#endregion
28
+ export { createRedactionTraversalContext, redactionTruncatedValue };
29
+ //# sourceMappingURL=traversal.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"traversal.js","names":["options: RedactionTraversalOptions","reportLimitExceeded: (\n limit: RedactionLimit,\n limits: RedactionTraversalLimits,\n ) => void","limits: RedactionTraversalLimits","value: number | undefined","defaultValue: number"],"sources":["../src/traversal.ts"],"sourcesContent":["/**\n * Options for bounding recursive redaction traversal.\n * @since 2.2.0\n */\nexport interface RedactionTraversalOptions {\n /**\n * Maximum recursion depth for object and array traversal.\n * @default `20`\n */\n readonly maxDepth?: number;\n\n /**\n * Maximum number of properties or array elements to process per object.\n * @default `1000`\n */\n readonly maxProperties?: number;\n}\n\nexport interface RedactionTraversalLimits {\n readonly maxDepth: number;\n readonly maxProperties: number;\n}\n\nexport type RedactionLimit = keyof RedactionTraversalLimits;\n\nexport interface RedactionTraversalContext {\n readonly limits: RedactionTraversalLimits;\n readonly visited: Map<object, object>;\n readonly exceededLimits: Set<RedactionLimit>;\n reportLimitExceeded(limit: RedactionLimit): void;\n}\n\nexport const redactionTruncatedValue = \"[truncated]\";\n\nconst defaultMaxDepth = 20;\nconst defaultMaxProperties = 1000;\n\nexport function createRedactionTraversalContext(\n options: RedactionTraversalOptions,\n reportLimitExceeded: (\n limit: RedactionLimit,\n limits: RedactionTraversalLimits,\n ) => void,\n visited = new Map<object, object>(),\n): RedactionTraversalContext {\n const limits: RedactionTraversalLimits = {\n maxDepth: normalizeLimit(options.maxDepth, defaultMaxDepth),\n maxProperties: normalizeLimit(options.maxProperties, defaultMaxProperties),\n };\n const exceededLimits = new Set<RedactionLimit>();\n return {\n limits,\n visited,\n exceededLimits,\n reportLimitExceeded(limit) {\n exceededLimits.add(limit);\n reportLimitExceeded(limit, limits);\n },\n };\n}\n\nfunction normalizeLimit(\n value: number | undefined,\n defaultValue: number,\n): number {\n if (value == null) return defaultValue;\n if (!Number.isFinite(value) || value < 0) return defaultValue;\n return Math.floor(value);\n}\n"],"mappings":";AAgCA,MAAa,0BAA0B;AAEvC,MAAM,kBAAkB;AACxB,MAAM,uBAAuB;AAE7B,SAAgB,gCACdA,SACAC,qBAIA,0BAAU,IAAI,OACa;CAC3B,MAAMC,SAAmC;EACvC,UAAU,eAAe,QAAQ,UAAU,gBAAgB;EAC3D,eAAe,eAAe,QAAQ,eAAe,qBAAqB;CAC3E;CACD,MAAM,iCAAiB,IAAI;AAC3B,QAAO;EACL;EACA;EACA;EACA,oBAAoB,OAAO;AACzB,kBAAe,IAAI,MAAM;AACzB,uBAAoB,OAAO,OAAO;EACnC;CACF;AACF;AAED,SAAS,eACPC,OACAC,cACQ;AACR,KAAI,SAAS,KAAM,QAAO;AAC1B,MAAK,OAAO,SAAS,MAAM,IAAI,QAAQ,EAAG,QAAO;AACjD,QAAO,KAAK,MAAM,MAAM;AACzB"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@logtape/redaction",
3
- "version": "2.2.0-dev.679+7d928b73",
3
+ "version": "2.2.0-dev.683+daa1aac5",
4
4
  "description": "Redact sensitive data from log messages",
5
5
  "keywords": [
6
6
  "logging",
@@ -49,7 +49,7 @@
49
49
  "dist/"
50
50
  ],
51
51
  "peerDependencies": {
52
- "@logtape/logtape": "^2.2.0-dev.679+7d928b73"
52
+ "@logtape/logtape": "^2.2.0-dev.683+daa1aac5"
53
53
  },
54
54
  "devDependencies": {
55
55
  "@alinea/suite": "^0.6.3",