@mandujs/core 0.12.1 → 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 (177) hide show
  1. package/README.ko.md +304 -304
  2. package/README.md +653 -653
  3. package/package.json +8 -8
  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/healing.ts +2 -0
  106. package/src/guard/index.ts +2 -0
  107. package/src/guard/negotiation.ts +430 -4
  108. package/src/guard/presets/atomic.ts +70 -70
  109. package/src/guard/presets/clean.ts +77 -77
  110. package/src/guard/presets/cqrs.test.ts +175 -0
  111. package/src/guard/presets/cqrs.ts +107 -0
  112. package/src/guard/presets/fsd.ts +79 -79
  113. package/src/guard/presets/hexagonal.ts +68 -68
  114. package/src/guard/presets/index.ts +291 -288
  115. package/src/guard/reporter.ts +445 -445
  116. package/src/guard/rules.ts +12 -12
  117. package/src/guard/statistics.ts +578 -578
  118. package/src/guard/suggestions.ts +358 -352
  119. package/src/guard/types.ts +348 -347
  120. package/src/guard/validator.ts +834 -834
  121. package/src/guard/watcher.ts +404 -404
  122. package/src/index.ts +1 -0
  123. package/src/intent/index.ts +310 -310
  124. package/src/island/index.ts +304 -304
  125. package/src/logging/index.ts +22 -22
  126. package/src/logging/transports.ts +365 -365
  127. package/src/paths.test.ts +47 -0
  128. package/src/paths.ts +47 -0
  129. package/src/plugins/index.ts +38 -38
  130. package/src/plugins/registry.ts +377 -377
  131. package/src/plugins/types.ts +363 -363
  132. package/src/report/build.ts +1 -1
  133. package/src/report/index.ts +1 -1
  134. package/src/router/fs-patterns.ts +387 -387
  135. package/src/router/fs-routes.ts +344 -401
  136. package/src/router/fs-scanner.ts +497 -497
  137. package/src/router/fs-types.ts +270 -278
  138. package/src/router/index.ts +81 -81
  139. package/src/runtime/boundary.tsx +232 -232
  140. package/src/runtime/compose.ts +222 -222
  141. package/src/runtime/lifecycle.ts +381 -381
  142. package/src/runtime/logger.test.ts +345 -345
  143. package/src/runtime/logger.ts +677 -677
  144. package/src/runtime/router.test.ts +476 -476
  145. package/src/runtime/router.ts +105 -105
  146. package/src/runtime/security.ts +155 -155
  147. package/src/runtime/server.ts +24 -24
  148. package/src/runtime/session-key.ts +328 -328
  149. package/src/runtime/ssr.ts +367 -367
  150. package/src/runtime/streaming-ssr.ts +1245 -1245
  151. package/src/runtime/trace.ts +144 -144
  152. package/src/seo/index.ts +214 -214
  153. package/src/seo/integration/ssr.ts +307 -307
  154. package/src/seo/render/basic.ts +427 -427
  155. package/src/seo/render/index.ts +143 -143
  156. package/src/seo/render/jsonld.ts +539 -539
  157. package/src/seo/render/opengraph.ts +191 -191
  158. package/src/seo/render/robots.ts +116 -116
  159. package/src/seo/render/sitemap.ts +137 -137
  160. package/src/seo/render/twitter.ts +126 -126
  161. package/src/seo/resolve/index.ts +353 -353
  162. package/src/seo/resolve/opengraph.ts +143 -143
  163. package/src/seo/resolve/robots.ts +73 -73
  164. package/src/seo/resolve/title.ts +94 -94
  165. package/src/seo/resolve/twitter.ts +73 -73
  166. package/src/seo/resolve/url.ts +97 -97
  167. package/src/seo/routes/index.ts +290 -290
  168. package/src/seo/types.ts +575 -575
  169. package/src/slot/validator.ts +39 -39
  170. package/src/spec/index.ts +3 -3
  171. package/src/spec/load.ts +76 -76
  172. package/src/spec/lock.ts +56 -56
  173. package/src/utils/bun.ts +8 -8
  174. package/src/utils/lru-cache.ts +75 -75
  175. package/src/utils/safe-io.ts +188 -188
  176. package/src/utils/string-safe.ts +298 -298
  177. 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';