@mandujs/core 0.12.2 → 0.13.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.
Files changed (173) hide show
  1. package/README.ko.md +304 -304
  2. package/README.md +653 -653
  3. package/package.json +1 -1
  4. package/src/brain/architecture/analyzer.ts +28 -26
  5. package/src/brain/doctor/analyzer.ts +1 -1
  6. package/src/bundler/build.ts +91 -91
  7. package/src/bundler/css.ts +302 -302
  8. package/src/bundler/dev.ts +0 -1
  9. package/src/change/history.ts +3 -3
  10. package/src/change/snapshot.ts +10 -9
  11. package/src/change/transaction.ts +2 -2
  12. package/src/client/Link.tsx +227 -227
  13. package/src/client/globals.ts +44 -44
  14. package/src/client/hooks.ts +267 -267
  15. package/src/client/index.ts +5 -5
  16. package/src/client/island.ts +8 -8
  17. package/src/client/router.ts +435 -435
  18. package/src/client/runtime.ts +23 -23
  19. package/src/client/serialize.ts +404 -404
  20. package/src/client/window-state.ts +101 -101
  21. package/src/config/mandu.ts +94 -96
  22. package/src/config/validate.ts +213 -215
  23. package/src/config/watcher.ts +311 -311
  24. package/src/constants.ts +40 -40
  25. package/src/content/content-layer.ts +314 -314
  26. package/src/content/content.test.ts +433 -433
  27. package/src/content/data-store.ts +245 -245
  28. package/src/content/digest.ts +133 -133
  29. package/src/content/index.ts +164 -164
  30. package/src/content/loader-context.ts +172 -172
  31. package/src/content/loaders/api.ts +216 -216
  32. package/src/content/loaders/file.ts +169 -169
  33. package/src/content/loaders/glob.ts +252 -252
  34. package/src/content/loaders/index.ts +34 -34
  35. package/src/content/loaders/types.ts +137 -137
  36. package/src/content/meta-store.ts +209 -209
  37. package/src/content/types.ts +282 -282
  38. package/src/content/watcher.ts +135 -135
  39. package/src/contract/client-safe.test.ts +42 -42
  40. package/src/contract/client-safe.ts +114 -114
  41. package/src/contract/client.ts +16 -16
  42. package/src/contract/define.ts +459 -459
  43. package/src/contract/handler.ts +10 -10
  44. package/src/contract/normalize.test.ts +276 -276
  45. package/src/contract/normalize.ts +404 -404
  46. package/src/contract/registry.test.ts +206 -206
  47. package/src/contract/registry.ts +568 -568
  48. package/src/contract/schema.ts +48 -48
  49. package/src/contract/types.ts +58 -58
  50. package/src/contract/validator.ts +32 -32
  51. package/src/devtools/ai/context-builder.ts +375 -375
  52. package/src/devtools/ai/index.ts +25 -25
  53. package/src/devtools/ai/mcp-connector.ts +465 -465
  54. package/src/devtools/client/catchers/error-catcher.ts +327 -327
  55. package/src/devtools/client/catchers/index.ts +18 -18
  56. package/src/devtools/client/catchers/network-proxy.ts +363 -363
  57. package/src/devtools/client/components/index.ts +39 -39
  58. package/src/devtools/client/components/kitchen-root.tsx +362 -362
  59. package/src/devtools/client/components/mandu-character.tsx +241 -241
  60. package/src/devtools/client/components/overlay.tsx +368 -368
  61. package/src/devtools/client/components/panel/errors-panel.tsx +259 -259
  62. package/src/devtools/client/components/panel/guard-panel.tsx +244 -244
  63. package/src/devtools/client/components/panel/index.ts +32 -32
  64. package/src/devtools/client/components/panel/islands-panel.tsx +304 -304
  65. package/src/devtools/client/components/panel/network-panel.tsx +292 -292
  66. package/src/devtools/client/components/panel/panel-container.tsx +259 -259
  67. package/src/devtools/client/filters/context-filters.ts +282 -282
  68. package/src/devtools/client/filters/index.ts +16 -16
  69. package/src/devtools/client/index.ts +63 -63
  70. package/src/devtools/client/persistence.ts +335 -335
  71. package/src/devtools/client/state-manager.ts +478 -478
  72. package/src/devtools/design-tokens.ts +263 -263
  73. package/src/devtools/hook/create-hook.ts +207 -207
  74. package/src/devtools/hook/index.ts +13 -13
  75. package/src/devtools/index.ts +439 -439
  76. package/src/devtools/init.ts +266 -266
  77. package/src/devtools/protocol.ts +237 -237
  78. package/src/devtools/server/index.ts +17 -17
  79. package/src/devtools/server/source-context.ts +444 -444
  80. package/src/devtools/types.ts +319 -319
  81. package/src/devtools/worker/index.ts +25 -25
  82. package/src/devtools/worker/redaction-worker.ts +222 -222
  83. package/src/devtools/worker/worker-manager.ts +409 -409
  84. package/src/error/classifier.ts +2 -2
  85. package/src/error/domains.ts +265 -265
  86. package/src/error/formatter.ts +32 -32
  87. package/src/error/result.ts +46 -46
  88. package/src/error/stack-analyzer.ts +5 -0
  89. package/src/error/types.ts +6 -6
  90. package/src/errors/extractor.ts +409 -409
  91. package/src/errors/index.ts +19 -19
  92. package/src/filling/auth.ts +308 -308
  93. package/src/filling/context.ts +569 -569
  94. package/src/filling/deps.ts +238 -238
  95. package/src/generator/contract-glue.ts +2 -1
  96. package/src/generator/generate.ts +12 -10
  97. package/src/generator/index.ts +3 -3
  98. package/src/generator/templates.ts +80 -79
  99. package/src/guard/analyzer.ts +360 -360
  100. package/src/guard/ast-analyzer.ts +806 -806
  101. package/src/guard/auto-correct.ts +1 -1
  102. package/src/guard/check.ts +128 -128
  103. package/src/guard/contract-guard.ts +9 -9
  104. package/src/guard/file-type.test.ts +24 -24
  105. package/src/guard/presets/atomic.ts +70 -70
  106. package/src/guard/presets/clean.ts +77 -77
  107. package/src/guard/presets/cqrs.test.ts +35 -14
  108. package/src/guard/presets/fsd.ts +79 -79
  109. package/src/guard/presets/hexagonal.ts +68 -68
  110. package/src/guard/presets/index.ts +291 -291
  111. package/src/guard/reporter.ts +445 -445
  112. package/src/guard/rules.ts +12 -12
  113. package/src/guard/statistics.ts +578 -578
  114. package/src/guard/suggestions.ts +358 -358
  115. package/src/guard/types.ts +348 -348
  116. package/src/guard/validator.ts +834 -834
  117. package/src/guard/watcher.ts +404 -404
  118. package/src/index.ts +1 -0
  119. package/src/intent/index.ts +310 -310
  120. package/src/island/index.ts +304 -304
  121. package/src/logging/index.ts +22 -22
  122. package/src/logging/transports.ts +365 -365
  123. package/src/paths.test.ts +47 -0
  124. package/src/paths.ts +47 -0
  125. package/src/plugins/index.ts +38 -38
  126. package/src/plugins/registry.ts +377 -377
  127. package/src/plugins/types.ts +363 -363
  128. package/src/report/build.ts +1 -1
  129. package/src/report/index.ts +1 -1
  130. package/src/router/fs-patterns.ts +387 -387
  131. package/src/router/fs-routes.ts +344 -401
  132. package/src/router/fs-scanner.ts +497 -497
  133. package/src/router/fs-types.ts +270 -278
  134. package/src/router/index.ts +81 -81
  135. package/src/runtime/boundary.tsx +232 -232
  136. package/src/runtime/compose.ts +222 -222
  137. package/src/runtime/lifecycle.ts +381 -381
  138. package/src/runtime/logger.test.ts +345 -345
  139. package/src/runtime/logger.ts +677 -677
  140. package/src/runtime/router.test.ts +476 -476
  141. package/src/runtime/router.ts +105 -105
  142. package/src/runtime/security.ts +155 -155
  143. package/src/runtime/server.ts +24 -24
  144. package/src/runtime/session-key.ts +328 -328
  145. package/src/runtime/ssr.ts +367 -367
  146. package/src/runtime/streaming-ssr.ts +1245 -1245
  147. package/src/runtime/trace.ts +144 -144
  148. package/src/seo/index.ts +214 -214
  149. package/src/seo/integration/ssr.ts +307 -307
  150. package/src/seo/render/basic.ts +427 -427
  151. package/src/seo/render/index.ts +143 -143
  152. package/src/seo/render/jsonld.ts +539 -539
  153. package/src/seo/render/opengraph.ts +191 -191
  154. package/src/seo/render/robots.ts +116 -116
  155. package/src/seo/render/sitemap.ts +137 -137
  156. package/src/seo/render/twitter.ts +126 -126
  157. package/src/seo/resolve/index.ts +353 -353
  158. package/src/seo/resolve/opengraph.ts +143 -143
  159. package/src/seo/resolve/robots.ts +73 -73
  160. package/src/seo/resolve/title.ts +94 -94
  161. package/src/seo/resolve/twitter.ts +73 -73
  162. package/src/seo/resolve/url.ts +97 -97
  163. package/src/seo/routes/index.ts +290 -290
  164. package/src/seo/types.ts +575 -575
  165. package/src/slot/validator.ts +39 -39
  166. package/src/spec/index.ts +3 -3
  167. package/src/spec/load.ts +76 -76
  168. package/src/spec/lock.ts +56 -56
  169. package/src/utils/bun.ts +8 -8
  170. package/src/utils/lru-cache.ts +75 -75
  171. package/src/utils/safe-io.ts +188 -188
  172. package/src/utils/string-safe.ts +298 -298
  173. package/src/watcher/rules.ts +5 -5
@@ -1,282 +1,282 @@
1
- /**
2
- * Mandu Kitchen DevTools - Context Filters
3
- * @version 1.0.3
4
- *
5
- * 마스킹 파이프라인 - PII/시크릿 정보 필터링
6
- */
7
-
8
- import type { RedactPattern } from '../../types';
9
-
10
- // ============================================================================
11
- // Built-in Patterns
12
- // ============================================================================
13
-
14
- /**
15
- * 기본 제공 시크릿 패턴
16
- */
17
- const BUILT_IN_SECRET_PATTERNS: RegExp[] = [
18
- // JWT tokens
19
- /eyJ[A-Za-z0-9-_]+\.eyJ[A-Za-z0-9-_]+\.[A-Za-z0-9-_.+/]*/g,
20
-
21
- // AWS keys
22
- /AKIA[0-9A-Z]{16}/g,
23
- /[A-Za-z0-9/+=]{40}/g, // AWS secret (when near access key)
24
-
25
- // API keys (generic patterns)
26
- /api[_-]?key["\s:=]+["']?[A-Za-z0-9-_]{20,}["']?/gi,
27
- /secret[_-]?key["\s:=]+["']?[A-Za-z0-9-_]{20,}["']?/gi,
28
-
29
- // Private keys
30
- /-----BEGIN [A-Z ]+ PRIVATE KEY-----[\s\S]*?-----END [A-Z ]+ PRIVATE KEY-----/g,
31
-
32
- // Bearer tokens
33
- /Bearer\s+[A-Za-z0-9-_.]+/gi,
34
-
35
- // Basic auth
36
- /Basic\s+[A-Za-z0-9+/=]+/gi,
37
- ];
38
-
39
- /**
40
- * PII 패턴 (이메일, 전화번호, IP 등)
41
- */
42
- const PII_PATTERNS: RegExp[] = [
43
- // Email
44
- /[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/g,
45
-
46
- // Phone numbers (다양한 형식)
47
- /\b\d{3}[-.]?\d{3,4}[-.]?\d{4}\b/g,
48
- /\+\d{1,3}[-.\s]?\(?\d{1,4}\)?[-.\s]?\d{1,4}[-.\s]?\d{1,9}/g,
49
-
50
- // IPv4
51
- /\b\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}\b/g,
52
-
53
- // IPv6
54
- /([0-9a-fA-F]{1,4}:){7,7}[0-9a-fA-F]{1,4}|([0-9a-fA-F]{1,4}:){1,7}:|([0-9a-fA-F]{1,4}:){1,6}:[0-9a-fA-F]{1,4}/g,
55
-
56
- // Credit card (기본 형식만)
57
- /\b\d{4}[-\s]?\d{4}[-\s]?\d{4}[-\s]?\d{4}\b/g,
58
-
59
- // SSN (US)
60
- /\b\d{3}-\d{2}-\d{4}\b/g,
61
- ];
62
-
63
- // ============================================================================
64
- // Filter Functions
65
- // ============================================================================
66
-
67
- /**
68
- * Phase 1: 주석 제거
69
- */
70
- export function removeComments(code: string): string {
71
- // Single-line comments
72
- let result = code.replace(/\/\/.*$/gm, '');
73
-
74
- // Multi-line comments
75
- result = result.replace(/\/\*[\s\S]*?\*\//g, '');
76
-
77
- // HTML comments
78
- result = result.replace(/<!--[\s\S]*?-->/g, '');
79
-
80
- return result;
81
- }
82
-
83
- /**
84
- * Phase 1b: 문자열 처리
85
- *
86
- * @param mode
87
- * - 'smart': PII/시크릿 패턴만 마스킹 (권장)
88
- * - 'strip': 모든 문자열 제거
89
- */
90
- export function handleStrings(code: string, mode: 'smart' | 'strip'): string {
91
- if (mode === 'strip') {
92
- // 모든 문자열 리터럴 제거
93
- return code
94
- .replace(/"(?:[^"\\]|\\.)*"/g, '"[STRING]"')
95
- .replace(/'(?:[^'\\]|\\.)*'/g, "'[STRING]'")
96
- .replace(/`(?:[^`\\]|\\.)*`/g, '`[STRING]`');
97
- }
98
-
99
- // Smart mode: PII/시크릿만 마스킹
100
- let result = code;
101
-
102
- // 시크릿 패턴 마스킹
103
- for (const pattern of BUILT_IN_SECRET_PATTERNS) {
104
- result = result.replace(pattern, '[SECRET]');
105
- }
106
-
107
- // PII 패턴 마스킹
108
- for (const pattern of PII_PATTERNS) {
109
- result = result.replace(pattern, '[PII]');
110
- }
111
-
112
- return result;
113
- }
114
-
115
- /**
116
- * Phase 2: 기본 보안 마스킹 (항상 적용, 비활성화 불가)
117
- */
118
- export function redactBuiltInSecrets(text: string): string {
119
- let result = text;
120
-
121
- for (const pattern of BUILT_IN_SECRET_PATTERNS) {
122
- // Reset regex state for global patterns
123
- pattern.lastIndex = 0;
124
- result = result.replace(pattern, '[REDACTED]');
125
- }
126
-
127
- return result;
128
- }
129
-
130
- /**
131
- * Phase 3: 사용자 정의 패턴 적용 (옵트인)
132
- */
133
- export function redactCustomPatterns(
134
- text: string,
135
- patterns: RedactPattern[]
136
- ): string {
137
- let result = text;
138
-
139
- for (const patternDef of patterns) {
140
- try {
141
- const regex = new RegExp(patternDef.source, patternDef.flags ?? 'gi');
142
- const replacement = patternDef.replacement ?? '[REDACTED]';
143
- result = result.replace(regex, replacement);
144
- } catch (e) {
145
- // 잘못된 정규식은 무시
146
- console.warn(
147
- `[Mandu Kitchen] Invalid redact pattern: ${patternDef.source}`,
148
- e
149
- );
150
- }
151
- }
152
-
153
- return result;
154
- }
155
-
156
- /**
157
- * Phase 4: 용량 제한 (항상 마지막)
158
- */
159
- export function truncate(text: string, maxBytes: number): string {
160
- if (maxBytes <= 0) return text;
161
-
162
- // UTF-8 바이트 길이 계산
163
- const encoder = new TextEncoder();
164
- const encoded = encoder.encode(text);
165
-
166
- if (encoded.length <= maxBytes) {
167
- return text;
168
- }
169
-
170
- // 바이트 단위로 자르고 디코딩
171
- const truncated = encoded.slice(0, maxBytes);
172
- const decoder = new TextDecoder('utf-8', { fatal: false });
173
- let result = decoder.decode(truncated);
174
-
175
- // 잘린 멀티바이트 문자 처리 (마지막 불완전한 문자 제거)
176
- if (result.endsWith('\ufffd')) {
177
- result = result.slice(0, -1);
178
- }
179
-
180
- return result + '... [TRUNCATED]';
181
- }
182
-
183
- // ============================================================================
184
- // Context Filters Pipeline
185
- // ============================================================================
186
-
187
- export interface FilterOptions {
188
- /** 문자열 처리 모드 */
189
- stringMode?: 'smart' | 'strip';
190
- /** 사용자 정의 패턴 */
191
- customPatterns?: RedactPattern[];
192
- /** 최대 바이트 */
193
- maxBytes?: number;
194
- /** 주석 제거 여부 (기본: true) */
195
- removeComments?: boolean;
196
- }
197
-
198
- /**
199
- * 전체 필터 파이프라인 실행
200
- */
201
- export function applyContextFilters(
202
- text: string,
203
- options: FilterOptions = {}
204
- ): string {
205
- const {
206
- stringMode = 'smart',
207
- customPatterns = [],
208
- maxBytes = 50_000, // 50KB default
209
- removeComments: shouldRemoveComments = true,
210
- } = options;
211
-
212
- let result = text;
213
-
214
- // Phase 1: 주석 제거
215
- if (shouldRemoveComments) {
216
- result = removeComments(result);
217
- }
218
-
219
- // Phase 1b: 문자열 처리
220
- result = handleStrings(result, stringMode);
221
-
222
- // Phase 2: 기본 보안 마스킹 (항상 적용)
223
- result = redactBuiltInSecrets(result);
224
-
225
- // Phase 3: 사용자 정의 패턴
226
- if (customPatterns.length > 0) {
227
- result = redactCustomPatterns(result, customPatterns);
228
- }
229
-
230
- // Phase 4: 용량 제한 (항상 마지막)
231
- result = truncate(result, maxBytes);
232
-
233
- return result;
234
- }
235
-
236
- // ============================================================================
237
- // Stack Trace Sanitizer
238
- // ============================================================================
239
-
240
- /**
241
- * 스택 트레이스에서 민감 정보 제거
242
- */
243
- export function sanitizeStackTrace(stack: string | undefined): string | undefined {
244
- if (!stack) return undefined;
245
-
246
- let result = stack;
247
-
248
- // 파일 경로에서 사용자명 제거
249
- result = result.replace(/\/Users\/[^/]+\//g, '/Users/[USER]/');
250
- result = result.replace(/\\Users\\[^\\]+\\/g, '\\Users\\[USER]\\');
251
- result = result.replace(/\/home\/[^/]+\//g, '/home/[USER]/');
252
-
253
- // 쿼리스트링 파라미터 마스킹
254
- result = result.replace(/\?[^\s)]+/g, '?[PARAMS]');
255
-
256
- // 기본 시크릿 마스킹
257
- result = redactBuiltInSecrets(result);
258
-
259
- return result;
260
- }
261
-
262
- // ============================================================================
263
- // Error Message Sanitizer
264
- // ============================================================================
265
-
266
- /**
267
- * 에러 메시지에서 민감 정보 제거
268
- */
269
- export function sanitizeErrorMessage(message: string): string {
270
- let result = message;
271
-
272
- // PII 마스킹
273
- for (const pattern of PII_PATTERNS) {
274
- pattern.lastIndex = 0;
275
- result = result.replace(pattern, '[PII]');
276
- }
277
-
278
- // 시크릿 마스킹
279
- result = redactBuiltInSecrets(result);
280
-
281
- return result;
282
- }
1
+ /**
2
+ * Mandu Kitchen DevTools - Context Filters
3
+ * @version 1.0.3
4
+ *
5
+ * 마스킹 파이프라인 - PII/시크릿 정보 필터링
6
+ */
7
+
8
+ import type { RedactPattern } from '../../types';
9
+
10
+ // ============================================================================
11
+ // Built-in Patterns
12
+ // ============================================================================
13
+
14
+ /**
15
+ * 기본 제공 시크릿 패턴
16
+ */
17
+ const BUILT_IN_SECRET_PATTERNS: RegExp[] = [
18
+ // JWT tokens
19
+ /eyJ[A-Za-z0-9-_]+\.eyJ[A-Za-z0-9-_]+\.[A-Za-z0-9-_.+/]*/g,
20
+
21
+ // AWS keys
22
+ /AKIA[0-9A-Z]{16}/g,
23
+ /[A-Za-z0-9/+=]{40}/g, // AWS secret (when near access key)
24
+
25
+ // API keys (generic patterns)
26
+ /api[_-]?key["\s:=]+["']?[A-Za-z0-9-_]{20,}["']?/gi,
27
+ /secret[_-]?key["\s:=]+["']?[A-Za-z0-9-_]{20,}["']?/gi,
28
+
29
+ // Private keys
30
+ /-----BEGIN [A-Z ]+ PRIVATE KEY-----[\s\S]*?-----END [A-Z ]+ PRIVATE KEY-----/g,
31
+
32
+ // Bearer tokens
33
+ /Bearer\s+[A-Za-z0-9-_.]+/gi,
34
+
35
+ // Basic auth
36
+ /Basic\s+[A-Za-z0-9+/=]+/gi,
37
+ ];
38
+
39
+ /**
40
+ * PII 패턴 (이메일, 전화번호, IP 등)
41
+ */
42
+ const PII_PATTERNS: RegExp[] = [
43
+ // Email
44
+ /[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/g,
45
+
46
+ // Phone numbers (다양한 형식)
47
+ /\b\d{3}[-.]?\d{3,4}[-.]?\d{4}\b/g,
48
+ /\+\d{1,3}[-.\s]?\(?\d{1,4}\)?[-.\s]?\d{1,4}[-.\s]?\d{1,9}/g,
49
+
50
+ // IPv4
51
+ /\b\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}\b/g,
52
+
53
+ // IPv6
54
+ /([0-9a-fA-F]{1,4}:){7,7}[0-9a-fA-F]{1,4}|([0-9a-fA-F]{1,4}:){1,7}:|([0-9a-fA-F]{1,4}:){1,6}:[0-9a-fA-F]{1,4}/g,
55
+
56
+ // Credit card (기본 형식만)
57
+ /\b\d{4}[-\s]?\d{4}[-\s]?\d{4}[-\s]?\d{4}\b/g,
58
+
59
+ // SSN (US)
60
+ /\b\d{3}-\d{2}-\d{4}\b/g,
61
+ ];
62
+
63
+ // ============================================================================
64
+ // Filter Functions
65
+ // ============================================================================
66
+
67
+ /**
68
+ * Phase 1: 주석 제거
69
+ */
70
+ export function removeComments(code: string): string {
71
+ // Single-line comments
72
+ let result = code.replace(/\/\/.*$/gm, '');
73
+
74
+ // Multi-line comments
75
+ result = result.replace(/\/\*[\s\S]*?\*\//g, '');
76
+
77
+ // HTML comments
78
+ result = result.replace(/<!--[\s\S]*?-->/g, '');
79
+
80
+ return result;
81
+ }
82
+
83
+ /**
84
+ * Phase 1b: 문자열 처리
85
+ *
86
+ * @param mode
87
+ * - 'smart': PII/시크릿 패턴만 마스킹 (권장)
88
+ * - 'strip': 모든 문자열 제거
89
+ */
90
+ export function handleStrings(code: string, mode: 'smart' | 'strip'): string {
91
+ if (mode === 'strip') {
92
+ // 모든 문자열 리터럴 제거
93
+ return code
94
+ .replace(/"(?:[^"\\]|\\.)*"/g, '"[STRING]"')
95
+ .replace(/'(?:[^'\\]|\\.)*'/g, "'[STRING]'")
96
+ .replace(/`(?:[^`\\]|\\.)*`/g, '`[STRING]`');
97
+ }
98
+
99
+ // Smart mode: PII/시크릿만 마스킹
100
+ let result = code;
101
+
102
+ // 시크릿 패턴 마스킹
103
+ for (const pattern of BUILT_IN_SECRET_PATTERNS) {
104
+ result = result.replace(pattern, '[SECRET]');
105
+ }
106
+
107
+ // PII 패턴 마스킹
108
+ for (const pattern of PII_PATTERNS) {
109
+ result = result.replace(pattern, '[PII]');
110
+ }
111
+
112
+ return result;
113
+ }
114
+
115
+ /**
116
+ * Phase 2: 기본 보안 마스킹 (항상 적용, 비활성화 불가)
117
+ */
118
+ export function redactBuiltInSecrets(text: string): string {
119
+ let result = text;
120
+
121
+ for (const pattern of BUILT_IN_SECRET_PATTERNS) {
122
+ // Reset regex state for global patterns
123
+ pattern.lastIndex = 0;
124
+ result = result.replace(pattern, '[REDACTED]');
125
+ }
126
+
127
+ return result;
128
+ }
129
+
130
+ /**
131
+ * Phase 3: 사용자 정의 패턴 적용 (옵트인)
132
+ */
133
+ export function redactCustomPatterns(
134
+ text: string,
135
+ patterns: RedactPattern[]
136
+ ): string {
137
+ let result = text;
138
+
139
+ for (const patternDef of patterns) {
140
+ try {
141
+ const regex = new RegExp(patternDef.source, patternDef.flags ?? 'gi');
142
+ const replacement = patternDef.replacement ?? '[REDACTED]';
143
+ result = result.replace(regex, replacement);
144
+ } catch (e) {
145
+ // 잘못된 정규식은 무시
146
+ console.warn(
147
+ `[Mandu Kitchen] Invalid redact pattern: ${patternDef.source}`,
148
+ e
149
+ );
150
+ }
151
+ }
152
+
153
+ return result;
154
+ }
155
+
156
+ /**
157
+ * Phase 4: 용량 제한 (항상 마지막)
158
+ */
159
+ export function truncate(text: string, maxBytes: number): string {
160
+ if (maxBytes <= 0) return text;
161
+
162
+ // UTF-8 바이트 길이 계산
163
+ const encoder = new TextEncoder();
164
+ const encoded = encoder.encode(text);
165
+
166
+ if (encoded.length <= maxBytes) {
167
+ return text;
168
+ }
169
+
170
+ // 바이트 단위로 자르고 디코딩
171
+ const truncated = encoded.slice(0, maxBytes);
172
+ const decoder = new TextDecoder('utf-8', { fatal: false });
173
+ let result = decoder.decode(truncated);
174
+
175
+ // 잘린 멀티바이트 문자 처리 (마지막 불완전한 문자 제거)
176
+ if (result.endsWith('\ufffd')) {
177
+ result = result.slice(0, -1);
178
+ }
179
+
180
+ return result + '... [TRUNCATED]';
181
+ }
182
+
183
+ // ============================================================================
184
+ // Context Filters Pipeline
185
+ // ============================================================================
186
+
187
+ export interface FilterOptions {
188
+ /** 문자열 처리 모드 */
189
+ stringMode?: 'smart' | 'strip';
190
+ /** 사용자 정의 패턴 */
191
+ customPatterns?: RedactPattern[];
192
+ /** 최대 바이트 */
193
+ maxBytes?: number;
194
+ /** 주석 제거 여부 (기본: true) */
195
+ removeComments?: boolean;
196
+ }
197
+
198
+ /**
199
+ * 전체 필터 파이프라인 실행
200
+ */
201
+ export function applyContextFilters(
202
+ text: string,
203
+ options: FilterOptions = {}
204
+ ): string {
205
+ const {
206
+ stringMode = 'smart',
207
+ customPatterns = [],
208
+ maxBytes = 50_000, // 50KB default
209
+ removeComments: shouldRemoveComments = true,
210
+ } = options;
211
+
212
+ let result = text;
213
+
214
+ // Phase 1: 주석 제거
215
+ if (shouldRemoveComments) {
216
+ result = removeComments(result);
217
+ }
218
+
219
+ // Phase 1b: 문자열 처리
220
+ result = handleStrings(result, stringMode);
221
+
222
+ // Phase 2: 기본 보안 마스킹 (항상 적용)
223
+ result = redactBuiltInSecrets(result);
224
+
225
+ // Phase 3: 사용자 정의 패턴
226
+ if (customPatterns.length > 0) {
227
+ result = redactCustomPatterns(result, customPatterns);
228
+ }
229
+
230
+ // Phase 4: 용량 제한 (항상 마지막)
231
+ result = truncate(result, maxBytes);
232
+
233
+ return result;
234
+ }
235
+
236
+ // ============================================================================
237
+ // Stack Trace Sanitizer
238
+ // ============================================================================
239
+
240
+ /**
241
+ * 스택 트레이스에서 민감 정보 제거
242
+ */
243
+ export function sanitizeStackTrace(stack: string | undefined): string | undefined {
244
+ if (!stack) return undefined;
245
+
246
+ let result = stack;
247
+
248
+ // 파일 경로에서 사용자명 제거
249
+ result = result.replace(/\/Users\/[^/]+\//g, '/Users/[USER]/');
250
+ result = result.replace(/\\Users\\[^\\]+\\/g, '\\Users\\[USER]\\');
251
+ result = result.replace(/\/home\/[^/]+\//g, '/home/[USER]/');
252
+
253
+ // 쿼리스트링 파라미터 마스킹
254
+ result = result.replace(/\?[^\s)]+/g, '?[PARAMS]');
255
+
256
+ // 기본 시크릿 마스킹
257
+ result = redactBuiltInSecrets(result);
258
+
259
+ return result;
260
+ }
261
+
262
+ // ============================================================================
263
+ // Error Message Sanitizer
264
+ // ============================================================================
265
+
266
+ /**
267
+ * 에러 메시지에서 민감 정보 제거
268
+ */
269
+ export function sanitizeErrorMessage(message: string): string {
270
+ let result = message;
271
+
272
+ // PII 마스킹
273
+ for (const pattern of PII_PATTERNS) {
274
+ pattern.lastIndex = 0;
275
+ result = result.replace(pattern, '[PII]');
276
+ }
277
+
278
+ // 시크릿 마스킹
279
+ result = redactBuiltInSecrets(result);
280
+
281
+ return result;
282
+ }
@@ -1,16 +1,16 @@
1
- /**
2
- * Mandu Kitchen DevTools - Filters Module
3
- * @version 1.0.3
4
- */
5
-
6
- export {
7
- removeComments,
8
- handleStrings,
9
- redactBuiltInSecrets,
10
- redactCustomPatterns,
11
- truncate,
12
- applyContextFilters,
13
- sanitizeStackTrace,
14
- sanitizeErrorMessage,
15
- type FilterOptions,
16
- } from './context-filters';
1
+ /**
2
+ * Mandu Kitchen DevTools - Filters Module
3
+ * @version 1.0.3
4
+ */
5
+
6
+ export {
7
+ removeComments,
8
+ handleStrings,
9
+ redactBuiltInSecrets,
10
+ redactCustomPatterns,
11
+ truncate,
12
+ applyContextFilters,
13
+ sanitizeStackTrace,
14
+ sanitizeErrorMessage,
15
+ type FilterOptions,
16
+ } from './context-filters';