@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.
- package/README.ko.md +304 -304
- package/README.md +653 -653
- package/package.json +1 -1
- package/src/brain/architecture/analyzer.ts +28 -26
- package/src/brain/doctor/analyzer.ts +1 -1
- package/src/bundler/build.ts +91 -91
- package/src/bundler/css.ts +302 -302
- package/src/bundler/dev.ts +0 -1
- package/src/change/history.ts +3 -3
- package/src/change/snapshot.ts +10 -9
- package/src/change/transaction.ts +2 -2
- package/src/client/Link.tsx +227 -227
- package/src/client/globals.ts +44 -44
- package/src/client/hooks.ts +267 -267
- package/src/client/index.ts +5 -5
- package/src/client/island.ts +8 -8
- package/src/client/router.ts +435 -435
- package/src/client/runtime.ts +23 -23
- package/src/client/serialize.ts +404 -404
- package/src/client/window-state.ts +101 -101
- package/src/config/mandu.ts +94 -96
- package/src/config/validate.ts +213 -215
- package/src/config/watcher.ts +311 -311
- package/src/constants.ts +40 -40
- package/src/content/content-layer.ts +314 -314
- package/src/content/content.test.ts +433 -433
- package/src/content/data-store.ts +245 -245
- package/src/content/digest.ts +133 -133
- package/src/content/index.ts +164 -164
- package/src/content/loader-context.ts +172 -172
- package/src/content/loaders/api.ts +216 -216
- package/src/content/loaders/file.ts +169 -169
- package/src/content/loaders/glob.ts +252 -252
- package/src/content/loaders/index.ts +34 -34
- package/src/content/loaders/types.ts +137 -137
- package/src/content/meta-store.ts +209 -209
- package/src/content/types.ts +282 -282
- package/src/content/watcher.ts +135 -135
- package/src/contract/client-safe.test.ts +42 -42
- package/src/contract/client-safe.ts +114 -114
- package/src/contract/client.ts +16 -16
- package/src/contract/define.ts +459 -459
- package/src/contract/handler.ts +10 -10
- package/src/contract/normalize.test.ts +276 -276
- package/src/contract/normalize.ts +404 -404
- package/src/contract/registry.test.ts +206 -206
- package/src/contract/registry.ts +568 -568
- package/src/contract/schema.ts +48 -48
- package/src/contract/types.ts +58 -58
- package/src/contract/validator.ts +32 -32
- package/src/devtools/ai/context-builder.ts +375 -375
- package/src/devtools/ai/index.ts +25 -25
- package/src/devtools/ai/mcp-connector.ts +465 -465
- package/src/devtools/client/catchers/error-catcher.ts +327 -327
- package/src/devtools/client/catchers/index.ts +18 -18
- package/src/devtools/client/catchers/network-proxy.ts +363 -363
- package/src/devtools/client/components/index.ts +39 -39
- package/src/devtools/client/components/kitchen-root.tsx +362 -362
- package/src/devtools/client/components/mandu-character.tsx +241 -241
- package/src/devtools/client/components/overlay.tsx +368 -368
- package/src/devtools/client/components/panel/errors-panel.tsx +259 -259
- package/src/devtools/client/components/panel/guard-panel.tsx +244 -244
- package/src/devtools/client/components/panel/index.ts +32 -32
- package/src/devtools/client/components/panel/islands-panel.tsx +304 -304
- package/src/devtools/client/components/panel/network-panel.tsx +292 -292
- package/src/devtools/client/components/panel/panel-container.tsx +259 -259
- package/src/devtools/client/filters/context-filters.ts +282 -282
- package/src/devtools/client/filters/index.ts +16 -16
- package/src/devtools/client/index.ts +63 -63
- package/src/devtools/client/persistence.ts +335 -335
- package/src/devtools/client/state-manager.ts +478 -478
- package/src/devtools/design-tokens.ts +263 -263
- package/src/devtools/hook/create-hook.ts +207 -207
- package/src/devtools/hook/index.ts +13 -13
- package/src/devtools/index.ts +439 -439
- package/src/devtools/init.ts +266 -266
- package/src/devtools/protocol.ts +237 -237
- package/src/devtools/server/index.ts +17 -17
- package/src/devtools/server/source-context.ts +444 -444
- package/src/devtools/types.ts +319 -319
- package/src/devtools/worker/index.ts +25 -25
- package/src/devtools/worker/redaction-worker.ts +222 -222
- package/src/devtools/worker/worker-manager.ts +409 -409
- package/src/error/classifier.ts +2 -2
- package/src/error/domains.ts +265 -265
- package/src/error/formatter.ts +32 -32
- package/src/error/result.ts +46 -46
- package/src/error/stack-analyzer.ts +5 -0
- package/src/error/types.ts +6 -6
- package/src/errors/extractor.ts +409 -409
- package/src/errors/index.ts +19 -19
- package/src/filling/auth.ts +308 -308
- package/src/filling/context.ts +569 -569
- package/src/filling/deps.ts +238 -238
- package/src/generator/contract-glue.ts +2 -1
- package/src/generator/generate.ts +12 -10
- package/src/generator/index.ts +3 -3
- package/src/generator/templates.ts +80 -79
- package/src/guard/analyzer.ts +360 -360
- package/src/guard/ast-analyzer.ts +806 -806
- package/src/guard/auto-correct.ts +1 -1
- package/src/guard/check.ts +128 -128
- package/src/guard/contract-guard.ts +9 -9
- package/src/guard/file-type.test.ts +24 -24
- package/src/guard/presets/atomic.ts +70 -70
- package/src/guard/presets/clean.ts +77 -77
- package/src/guard/presets/cqrs.test.ts +35 -14
- package/src/guard/presets/fsd.ts +79 -79
- package/src/guard/presets/hexagonal.ts +68 -68
- package/src/guard/presets/index.ts +291 -291
- package/src/guard/reporter.ts +445 -445
- package/src/guard/rules.ts +12 -12
- package/src/guard/statistics.ts +578 -578
- package/src/guard/suggestions.ts +358 -358
- package/src/guard/types.ts +348 -348
- package/src/guard/validator.ts +834 -834
- package/src/guard/watcher.ts +404 -404
- package/src/index.ts +1 -0
- package/src/intent/index.ts +310 -310
- package/src/island/index.ts +304 -304
- package/src/logging/index.ts +22 -22
- package/src/logging/transports.ts +365 -365
- package/src/paths.test.ts +47 -0
- package/src/paths.ts +47 -0
- package/src/plugins/index.ts +38 -38
- package/src/plugins/registry.ts +377 -377
- package/src/plugins/types.ts +363 -363
- package/src/report/build.ts +1 -1
- package/src/report/index.ts +1 -1
- package/src/router/fs-patterns.ts +387 -387
- package/src/router/fs-routes.ts +344 -401
- package/src/router/fs-scanner.ts +497 -497
- package/src/router/fs-types.ts +270 -278
- package/src/router/index.ts +81 -81
- package/src/runtime/boundary.tsx +232 -232
- package/src/runtime/compose.ts +222 -222
- package/src/runtime/lifecycle.ts +381 -381
- package/src/runtime/logger.test.ts +345 -345
- package/src/runtime/logger.ts +677 -677
- package/src/runtime/router.test.ts +476 -476
- package/src/runtime/router.ts +105 -105
- package/src/runtime/security.ts +155 -155
- package/src/runtime/server.ts +24 -24
- package/src/runtime/session-key.ts +328 -328
- package/src/runtime/ssr.ts +367 -367
- package/src/runtime/streaming-ssr.ts +1245 -1245
- package/src/runtime/trace.ts +144 -144
- package/src/seo/index.ts +214 -214
- package/src/seo/integration/ssr.ts +307 -307
- package/src/seo/render/basic.ts +427 -427
- package/src/seo/render/index.ts +143 -143
- package/src/seo/render/jsonld.ts +539 -539
- package/src/seo/render/opengraph.ts +191 -191
- package/src/seo/render/robots.ts +116 -116
- package/src/seo/render/sitemap.ts +137 -137
- package/src/seo/render/twitter.ts +126 -126
- package/src/seo/resolve/index.ts +353 -353
- package/src/seo/resolve/opengraph.ts +143 -143
- package/src/seo/resolve/robots.ts +73 -73
- package/src/seo/resolve/title.ts +94 -94
- package/src/seo/resolve/twitter.ts +73 -73
- package/src/seo/resolve/url.ts +97 -97
- package/src/seo/routes/index.ts +290 -290
- package/src/seo/types.ts +575 -575
- package/src/slot/validator.ts +39 -39
- package/src/spec/index.ts +3 -3
- package/src/spec/load.ts +76 -76
- package/src/spec/lock.ts +56 -56
- package/src/utils/bun.ts +8 -8
- package/src/utils/lru-cache.ts +75 -75
- package/src/utils/safe-io.ts +188 -188
- package/src/utils/string-safe.ts +298 -298
- package/src/watcher/rules.ts +5 -5
package/src/error/classifier.ts
CHANGED
|
@@ -95,7 +95,7 @@ export class ErrorClassifier {
|
|
|
95
95
|
case "spec":
|
|
96
96
|
errorType = "SPEC_ERROR";
|
|
97
97
|
code = ErrorCode.SPEC_VALIDATION_ERROR;
|
|
98
|
-
fixFile = blameFrame?.file || "
|
|
98
|
+
fixFile = blameFrame?.file || ".mandu/routes.manifest.json";
|
|
99
99
|
suggestion = "Spec 파일의 JSON 구문 또는 스키마를 확인하세요";
|
|
100
100
|
break;
|
|
101
101
|
|
|
@@ -258,7 +258,7 @@ export class ErrorClassifier {
|
|
|
258
258
|
export function createSpecError(
|
|
259
259
|
code: ErrorCode,
|
|
260
260
|
message: string,
|
|
261
|
-
file: string = "
|
|
261
|
+
file: string = ".mandu/routes.manifest.json",
|
|
262
262
|
suggestion?: string
|
|
263
263
|
): ManduError {
|
|
264
264
|
return {
|
package/src/error/domains.ts
CHANGED
|
@@ -1,265 +1,265 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* 도메인별 에러 클래스
|
|
3
|
-
*
|
|
4
|
-
* ManduError 인터페이스 기반의 실제 Error 클래스들.
|
|
5
|
-
* try-catch에서 instanceof 체크 가능.
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
import { ErrorCode, ERROR_MESSAGES, ERROR_SUMMARIES } from "./types";
|
|
9
|
-
import type { ManduError, RouteContext, ErrorType } from "./types";
|
|
10
|
-
|
|
11
|
-
/**
|
|
12
|
-
* Mandu 에러 베이스 클래스
|
|
13
|
-
*/
|
|
14
|
-
export abstract class ManduBaseError extends Error implements ManduError {
|
|
15
|
-
abstract readonly errorType: ErrorType;
|
|
16
|
-
readonly code: ErrorCode | string;
|
|
17
|
-
readonly httpStatus?: number;
|
|
18
|
-
readonly summary: string;
|
|
19
|
-
readonly fix: { file: string; suggestion: string; line?: number };
|
|
20
|
-
readonly route?: RouteContext;
|
|
21
|
-
readonly timestamp: string;
|
|
22
|
-
|
|
23
|
-
constructor(
|
|
24
|
-
code: ErrorCode | string,
|
|
25
|
-
message: string,
|
|
26
|
-
fix: { file: string; suggestion: string; line?: number },
|
|
27
|
-
options?: {
|
|
28
|
-
httpStatus?: number;
|
|
29
|
-
route?: RouteContext;
|
|
30
|
-
cause?: unknown;
|
|
31
|
-
}
|
|
32
|
-
) {
|
|
33
|
-
super(message, { cause: options?.cause });
|
|
34
|
-
this.name = this.constructor.name;
|
|
35
|
-
this.code = code;
|
|
36
|
-
this.httpStatus = options?.httpStatus;
|
|
37
|
-
this.summary =
|
|
38
|
-
typeof code === "string" && code in ERROR_SUMMARIES
|
|
39
|
-
? ERROR_SUMMARIES[code as ErrorCode]
|
|
40
|
-
: message;
|
|
41
|
-
this.fix = fix;
|
|
42
|
-
this.route = options?.route;
|
|
43
|
-
this.timestamp = new Date().toISOString();
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
/**
|
|
47
|
-
* ManduError 인터페이스로 변환
|
|
48
|
-
*/
|
|
49
|
-
toManduError(): ManduError {
|
|
50
|
-
return {
|
|
51
|
-
errorType: this.errorType,
|
|
52
|
-
code: this.code,
|
|
53
|
-
httpStatus: this.httpStatus,
|
|
54
|
-
message: this.message,
|
|
55
|
-
summary: this.summary,
|
|
56
|
-
fix: this.fix,
|
|
57
|
-
route: this.route,
|
|
58
|
-
timestamp: this.timestamp,
|
|
59
|
-
};
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
// ============================================================
|
|
64
|
-
// 파일 시스템 에러
|
|
65
|
-
// ============================================================
|
|
66
|
-
|
|
67
|
-
/**
|
|
68
|
-
* 파일 읽기/쓰기 에러
|
|
69
|
-
*/
|
|
70
|
-
export class FileError extends ManduBaseError {
|
|
71
|
-
readonly errorType = "LOGIC_ERROR" as const;
|
|
72
|
-
readonly filePath: string;
|
|
73
|
-
readonly operation: "read" | "write" | "access" | "stat";
|
|
74
|
-
|
|
75
|
-
constructor(
|
|
76
|
-
filePath: string,
|
|
77
|
-
operation: "read" | "write" | "access" | "stat",
|
|
78
|
-
cause?: unknown
|
|
79
|
-
) {
|
|
80
|
-
const message = `파일 ${operation} 실패: ${filePath}`;
|
|
81
|
-
super(
|
|
82
|
-
ErrorCode.SLOT_IMPORT_ERROR,
|
|
83
|
-
message,
|
|
84
|
-
{
|
|
85
|
-
file: filePath,
|
|
86
|
-
suggestion: `파일이 존재하고 읽기 권한이 있는지 확인하세요`,
|
|
87
|
-
},
|
|
88
|
-
{ cause }
|
|
89
|
-
);
|
|
90
|
-
this.filePath = filePath;
|
|
91
|
-
this.operation = operation;
|
|
92
|
-
}
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
/**
|
|
96
|
-
* 디렉토리 읽기 에러
|
|
97
|
-
*/
|
|
98
|
-
export class DirectoryError extends ManduBaseError {
|
|
99
|
-
readonly errorType = "LOGIC_ERROR" as const;
|
|
100
|
-
readonly dirPath: string;
|
|
101
|
-
|
|
102
|
-
constructor(dirPath: string, cause?: unknown) {
|
|
103
|
-
super(
|
|
104
|
-
ErrorCode.SLOT_NOT_FOUND,
|
|
105
|
-
`디렉토리 읽기 실패: ${dirPath}`,
|
|
106
|
-
{
|
|
107
|
-
file: dirPath,
|
|
108
|
-
suggestion: `디렉토리가 존재하고 접근 가능한지 확인하세요`,
|
|
109
|
-
},
|
|
110
|
-
{ cause }
|
|
111
|
-
);
|
|
112
|
-
this.dirPath = dirPath;
|
|
113
|
-
}
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
// ============================================================
|
|
117
|
-
// Guard 에러
|
|
118
|
-
// ============================================================
|
|
119
|
-
|
|
120
|
-
/**
|
|
121
|
-
* Guard 아키텍처 검사 에러
|
|
122
|
-
*/
|
|
123
|
-
export class GuardError extends ManduBaseError {
|
|
124
|
-
readonly errorType = "LOGIC_ERROR" as const;
|
|
125
|
-
readonly ruleId: string;
|
|
126
|
-
|
|
127
|
-
constructor(
|
|
128
|
-
ruleId: string,
|
|
129
|
-
message: string,
|
|
130
|
-
file: string,
|
|
131
|
-
options?: {
|
|
132
|
-
line?: number;
|
|
133
|
-
suggestion?: string;
|
|
134
|
-
cause?: unknown;
|
|
135
|
-
}
|
|
136
|
-
) {
|
|
137
|
-
super(
|
|
138
|
-
ErrorCode.SLOT_VALIDATION_ERROR,
|
|
139
|
-
message,
|
|
140
|
-
{
|
|
141
|
-
file,
|
|
142
|
-
suggestion: options?.suggestion || "아키텍처 규칙을 확인하세요",
|
|
143
|
-
line: options?.line,
|
|
144
|
-
},
|
|
145
|
-
{ cause: options?.cause }
|
|
146
|
-
);
|
|
147
|
-
this.ruleId = ruleId;
|
|
148
|
-
}
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
// ============================================================
|
|
152
|
-
// Router 에러
|
|
153
|
-
// ============================================================
|
|
154
|
-
|
|
155
|
-
/**
|
|
156
|
-
* 라우터 에러
|
|
157
|
-
*/
|
|
158
|
-
export class RouterError extends ManduBaseError {
|
|
159
|
-
readonly errorType = "FRAMEWORK_BUG" as const;
|
|
160
|
-
|
|
161
|
-
constructor(
|
|
162
|
-
message: string,
|
|
163
|
-
file: string,
|
|
164
|
-
options?: {
|
|
165
|
-
route?: RouteContext;
|
|
166
|
-
cause?: unknown;
|
|
167
|
-
}
|
|
168
|
-
) {
|
|
169
|
-
super(
|
|
170
|
-
ErrorCode.FRAMEWORK_ROUTER_ERROR,
|
|
171
|
-
message,
|
|
172
|
-
{
|
|
173
|
-
file,
|
|
174
|
-
suggestion: "라우트 설정을 확인하세요",
|
|
175
|
-
},
|
|
176
|
-
{ httpStatus: 500, ...options }
|
|
177
|
-
);
|
|
178
|
-
}
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
// ============================================================
|
|
182
|
-
// SSR 에러
|
|
183
|
-
// ============================================================
|
|
184
|
-
|
|
185
|
-
/**
|
|
186
|
-
* SSR 렌더링 에러
|
|
187
|
-
*/
|
|
188
|
-
export class SSRError extends ManduBaseError {
|
|
189
|
-
readonly errorType = "FRAMEWORK_BUG" as const;
|
|
190
|
-
|
|
191
|
-
constructor(
|
|
192
|
-
message: string,
|
|
193
|
-
route: RouteContext,
|
|
194
|
-
cause?: unknown
|
|
195
|
-
) {
|
|
196
|
-
super(
|
|
197
|
-
ErrorCode.FRAMEWORK_SSR_ERROR,
|
|
198
|
-
message,
|
|
199
|
-
{
|
|
200
|
-
file: `app/${route.id}/page.tsx`,
|
|
201
|
-
suggestion: "페이지 컴포넌트에서 렌더링 오류가 발생했습니다",
|
|
202
|
-
},
|
|
203
|
-
{ httpStatus: 500, route, cause }
|
|
204
|
-
);
|
|
205
|
-
}
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
// ============================================================
|
|
209
|
-
// Contract 에러
|
|
210
|
-
// ============================================================
|
|
211
|
-
|
|
212
|
-
/**
|
|
213
|
-
* API 계약 위반 에러
|
|
214
|
-
*/
|
|
215
|
-
export class ContractError extends ManduBaseError {
|
|
216
|
-
readonly errorType = "LOGIC_ERROR" as const;
|
|
217
|
-
|
|
218
|
-
constructor(
|
|
219
|
-
message: string,
|
|
220
|
-
contractFile: string,
|
|
221
|
-
options?: {
|
|
222
|
-
route?: RouteContext;
|
|
223
|
-
cause?: unknown;
|
|
224
|
-
}
|
|
225
|
-
) {
|
|
226
|
-
super(
|
|
227
|
-
ErrorCode.SLOT_VALIDATION_ERROR,
|
|
228
|
-
message,
|
|
229
|
-
{
|
|
230
|
-
file: contractFile,
|
|
231
|
-
suggestion: "API 계약과 실제 구현이 일치하는지 확인하세요",
|
|
232
|
-
},
|
|
233
|
-
{ httpStatus: 400, ...options }
|
|
234
|
-
);
|
|
235
|
-
}
|
|
236
|
-
}
|
|
237
|
-
|
|
238
|
-
// ============================================================
|
|
239
|
-
// Security 에러
|
|
240
|
-
// ============================================================
|
|
241
|
-
|
|
242
|
-
/**
|
|
243
|
-
* 보안 관련 에러
|
|
244
|
-
*/
|
|
245
|
-
export class SecurityError extends ManduBaseError {
|
|
246
|
-
readonly errorType = "LOGIC_ERROR" as const;
|
|
247
|
-
readonly securityType: "path_traversal" | "injection" | "unauthorized" | "import_violation";
|
|
248
|
-
|
|
249
|
-
constructor(
|
|
250
|
-
securityType: "path_traversal" | "injection" | "unauthorized" | "import_violation",
|
|
251
|
-
message: string,
|
|
252
|
-
file?: string
|
|
253
|
-
) {
|
|
254
|
-
super(
|
|
255
|
-
ErrorCode.SLOT_HANDLER_ERROR,
|
|
256
|
-
message,
|
|
257
|
-
{
|
|
258
|
-
file: file || "unknown",
|
|
259
|
-
suggestion: "보안 정책을 위반하는 요청입니다",
|
|
260
|
-
},
|
|
261
|
-
{ httpStatus: 403 }
|
|
262
|
-
);
|
|
263
|
-
this.securityType = securityType;
|
|
264
|
-
}
|
|
265
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* 도메인별 에러 클래스
|
|
3
|
+
*
|
|
4
|
+
* ManduError 인터페이스 기반의 실제 Error 클래스들.
|
|
5
|
+
* try-catch에서 instanceof 체크 가능.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { ErrorCode, ERROR_MESSAGES, ERROR_SUMMARIES } from "./types";
|
|
9
|
+
import type { ManduError, RouteContext, ErrorType } from "./types";
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Mandu 에러 베이스 클래스
|
|
13
|
+
*/
|
|
14
|
+
export abstract class ManduBaseError extends Error implements ManduError {
|
|
15
|
+
abstract readonly errorType: ErrorType;
|
|
16
|
+
readonly code: ErrorCode | string;
|
|
17
|
+
readonly httpStatus?: number;
|
|
18
|
+
readonly summary: string;
|
|
19
|
+
readonly fix: { file: string; suggestion: string; line?: number };
|
|
20
|
+
readonly route?: RouteContext;
|
|
21
|
+
readonly timestamp: string;
|
|
22
|
+
|
|
23
|
+
constructor(
|
|
24
|
+
code: ErrorCode | string,
|
|
25
|
+
message: string,
|
|
26
|
+
fix: { file: string; suggestion: string; line?: number },
|
|
27
|
+
options?: {
|
|
28
|
+
httpStatus?: number;
|
|
29
|
+
route?: RouteContext;
|
|
30
|
+
cause?: unknown;
|
|
31
|
+
}
|
|
32
|
+
) {
|
|
33
|
+
super(message, { cause: options?.cause });
|
|
34
|
+
this.name = this.constructor.name;
|
|
35
|
+
this.code = code;
|
|
36
|
+
this.httpStatus = options?.httpStatus;
|
|
37
|
+
this.summary =
|
|
38
|
+
typeof code === "string" && code in ERROR_SUMMARIES
|
|
39
|
+
? ERROR_SUMMARIES[code as ErrorCode]
|
|
40
|
+
: message;
|
|
41
|
+
this.fix = fix;
|
|
42
|
+
this.route = options?.route;
|
|
43
|
+
this.timestamp = new Date().toISOString();
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* ManduError 인터페이스로 변환
|
|
48
|
+
*/
|
|
49
|
+
toManduError(): ManduError {
|
|
50
|
+
return {
|
|
51
|
+
errorType: this.errorType,
|
|
52
|
+
code: this.code,
|
|
53
|
+
httpStatus: this.httpStatus,
|
|
54
|
+
message: this.message,
|
|
55
|
+
summary: this.summary,
|
|
56
|
+
fix: this.fix,
|
|
57
|
+
route: this.route,
|
|
58
|
+
timestamp: this.timestamp,
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// ============================================================
|
|
64
|
+
// 파일 시스템 에러
|
|
65
|
+
// ============================================================
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* 파일 읽기/쓰기 에러
|
|
69
|
+
*/
|
|
70
|
+
export class FileError extends ManduBaseError {
|
|
71
|
+
readonly errorType = "LOGIC_ERROR" as const;
|
|
72
|
+
readonly filePath: string;
|
|
73
|
+
readonly operation: "read" | "write" | "access" | "stat";
|
|
74
|
+
|
|
75
|
+
constructor(
|
|
76
|
+
filePath: string,
|
|
77
|
+
operation: "read" | "write" | "access" | "stat",
|
|
78
|
+
cause?: unknown
|
|
79
|
+
) {
|
|
80
|
+
const message = `파일 ${operation} 실패: ${filePath}`;
|
|
81
|
+
super(
|
|
82
|
+
ErrorCode.SLOT_IMPORT_ERROR,
|
|
83
|
+
message,
|
|
84
|
+
{
|
|
85
|
+
file: filePath,
|
|
86
|
+
suggestion: `파일이 존재하고 읽기 권한이 있는지 확인하세요`,
|
|
87
|
+
},
|
|
88
|
+
{ cause }
|
|
89
|
+
);
|
|
90
|
+
this.filePath = filePath;
|
|
91
|
+
this.operation = operation;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* 디렉토리 읽기 에러
|
|
97
|
+
*/
|
|
98
|
+
export class DirectoryError extends ManduBaseError {
|
|
99
|
+
readonly errorType = "LOGIC_ERROR" as const;
|
|
100
|
+
readonly dirPath: string;
|
|
101
|
+
|
|
102
|
+
constructor(dirPath: string, cause?: unknown) {
|
|
103
|
+
super(
|
|
104
|
+
ErrorCode.SLOT_NOT_FOUND,
|
|
105
|
+
`디렉토리 읽기 실패: ${dirPath}`,
|
|
106
|
+
{
|
|
107
|
+
file: dirPath,
|
|
108
|
+
suggestion: `디렉토리가 존재하고 접근 가능한지 확인하세요`,
|
|
109
|
+
},
|
|
110
|
+
{ cause }
|
|
111
|
+
);
|
|
112
|
+
this.dirPath = dirPath;
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
// ============================================================
|
|
117
|
+
// Guard 에러
|
|
118
|
+
// ============================================================
|
|
119
|
+
|
|
120
|
+
/**
|
|
121
|
+
* Guard 아키텍처 검사 에러
|
|
122
|
+
*/
|
|
123
|
+
export class GuardError extends ManduBaseError {
|
|
124
|
+
readonly errorType = "LOGIC_ERROR" as const;
|
|
125
|
+
readonly ruleId: string;
|
|
126
|
+
|
|
127
|
+
constructor(
|
|
128
|
+
ruleId: string,
|
|
129
|
+
message: string,
|
|
130
|
+
file: string,
|
|
131
|
+
options?: {
|
|
132
|
+
line?: number;
|
|
133
|
+
suggestion?: string;
|
|
134
|
+
cause?: unknown;
|
|
135
|
+
}
|
|
136
|
+
) {
|
|
137
|
+
super(
|
|
138
|
+
ErrorCode.SLOT_VALIDATION_ERROR,
|
|
139
|
+
message,
|
|
140
|
+
{
|
|
141
|
+
file,
|
|
142
|
+
suggestion: options?.suggestion || "아키텍처 규칙을 확인하세요",
|
|
143
|
+
line: options?.line,
|
|
144
|
+
},
|
|
145
|
+
{ cause: options?.cause }
|
|
146
|
+
);
|
|
147
|
+
this.ruleId = ruleId;
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
// ============================================================
|
|
152
|
+
// Router 에러
|
|
153
|
+
// ============================================================
|
|
154
|
+
|
|
155
|
+
/**
|
|
156
|
+
* 라우터 에러
|
|
157
|
+
*/
|
|
158
|
+
export class RouterError extends ManduBaseError {
|
|
159
|
+
readonly errorType = "FRAMEWORK_BUG" as const;
|
|
160
|
+
|
|
161
|
+
constructor(
|
|
162
|
+
message: string,
|
|
163
|
+
file: string,
|
|
164
|
+
options?: {
|
|
165
|
+
route?: RouteContext;
|
|
166
|
+
cause?: unknown;
|
|
167
|
+
}
|
|
168
|
+
) {
|
|
169
|
+
super(
|
|
170
|
+
ErrorCode.FRAMEWORK_ROUTER_ERROR,
|
|
171
|
+
message,
|
|
172
|
+
{
|
|
173
|
+
file,
|
|
174
|
+
suggestion: "라우트 설정을 확인하세요",
|
|
175
|
+
},
|
|
176
|
+
{ httpStatus: 500, ...options }
|
|
177
|
+
);
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
// ============================================================
|
|
182
|
+
// SSR 에러
|
|
183
|
+
// ============================================================
|
|
184
|
+
|
|
185
|
+
/**
|
|
186
|
+
* SSR 렌더링 에러
|
|
187
|
+
*/
|
|
188
|
+
export class SSRError extends ManduBaseError {
|
|
189
|
+
readonly errorType = "FRAMEWORK_BUG" as const;
|
|
190
|
+
|
|
191
|
+
constructor(
|
|
192
|
+
message: string,
|
|
193
|
+
route: RouteContext,
|
|
194
|
+
cause?: unknown
|
|
195
|
+
) {
|
|
196
|
+
super(
|
|
197
|
+
ErrorCode.FRAMEWORK_SSR_ERROR,
|
|
198
|
+
message,
|
|
199
|
+
{
|
|
200
|
+
file: `app/${route.id}/page.tsx`,
|
|
201
|
+
suggestion: "페이지 컴포넌트에서 렌더링 오류가 발생했습니다",
|
|
202
|
+
},
|
|
203
|
+
{ httpStatus: 500, route, cause }
|
|
204
|
+
);
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
// ============================================================
|
|
209
|
+
// Contract 에러
|
|
210
|
+
// ============================================================
|
|
211
|
+
|
|
212
|
+
/**
|
|
213
|
+
* API 계약 위반 에러
|
|
214
|
+
*/
|
|
215
|
+
export class ContractError extends ManduBaseError {
|
|
216
|
+
readonly errorType = "LOGIC_ERROR" as const;
|
|
217
|
+
|
|
218
|
+
constructor(
|
|
219
|
+
message: string,
|
|
220
|
+
contractFile: string,
|
|
221
|
+
options?: {
|
|
222
|
+
route?: RouteContext;
|
|
223
|
+
cause?: unknown;
|
|
224
|
+
}
|
|
225
|
+
) {
|
|
226
|
+
super(
|
|
227
|
+
ErrorCode.SLOT_VALIDATION_ERROR,
|
|
228
|
+
message,
|
|
229
|
+
{
|
|
230
|
+
file: contractFile,
|
|
231
|
+
suggestion: "API 계약과 실제 구현이 일치하는지 확인하세요",
|
|
232
|
+
},
|
|
233
|
+
{ httpStatus: 400, ...options }
|
|
234
|
+
);
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
// ============================================================
|
|
239
|
+
// Security 에러
|
|
240
|
+
// ============================================================
|
|
241
|
+
|
|
242
|
+
/**
|
|
243
|
+
* 보안 관련 에러
|
|
244
|
+
*/
|
|
245
|
+
export class SecurityError extends ManduBaseError {
|
|
246
|
+
readonly errorType = "LOGIC_ERROR" as const;
|
|
247
|
+
readonly securityType: "path_traversal" | "injection" | "unauthorized" | "import_violation";
|
|
248
|
+
|
|
249
|
+
constructor(
|
|
250
|
+
securityType: "path_traversal" | "injection" | "unauthorized" | "import_violation",
|
|
251
|
+
message: string,
|
|
252
|
+
file?: string
|
|
253
|
+
) {
|
|
254
|
+
super(
|
|
255
|
+
ErrorCode.SLOT_HANDLER_ERROR,
|
|
256
|
+
message,
|
|
257
|
+
{
|
|
258
|
+
file: file || "unknown",
|
|
259
|
+
suggestion: "보안 정책을 위반하는 요청입니다",
|
|
260
|
+
},
|
|
261
|
+
{ httpStatus: 403 }
|
|
262
|
+
);
|
|
263
|
+
this.securityType = securityType;
|
|
264
|
+
}
|
|
265
|
+
}
|
package/src/error/formatter.ts
CHANGED
|
@@ -105,14 +105,14 @@ export function createNotFoundResponse(
|
|
|
105
105
|
pathname: string,
|
|
106
106
|
routeContext?: RouteContext
|
|
107
107
|
): ManduError {
|
|
108
|
-
return {
|
|
109
|
-
errorType: "SPEC_ERROR",
|
|
110
|
-
code: ErrorCode.SPEC_ROUTE_NOT_FOUND,
|
|
111
|
-
httpStatus: 404,
|
|
112
|
-
message: `Route not found: ${pathname}`,
|
|
113
|
-
summary: "라우트 없음 - spec 파일에 추가 필요",
|
|
114
|
-
fix: {
|
|
115
|
-
file: "
|
|
108
|
+
return {
|
|
109
|
+
errorType: "SPEC_ERROR",
|
|
110
|
+
code: ErrorCode.SPEC_ROUTE_NOT_FOUND,
|
|
111
|
+
httpStatus: 404,
|
|
112
|
+
message: `Route not found: ${pathname}`,
|
|
113
|
+
summary: "라우트 없음 - spec 파일에 추가 필요",
|
|
114
|
+
fix: {
|
|
115
|
+
file: ".mandu/routes.manifest.json",
|
|
116
116
|
suggestion: `'${pathname}' 패턴의 라우트를 추가하세요`,
|
|
117
117
|
},
|
|
118
118
|
route: routeContext,
|
|
@@ -127,14 +127,14 @@ export function createHandlerNotFoundResponse(
|
|
|
127
127
|
routeId: string,
|
|
128
128
|
pattern: string
|
|
129
129
|
): ManduError {
|
|
130
|
-
return {
|
|
131
|
-
errorType: "FRAMEWORK_BUG",
|
|
132
|
-
code: ErrorCode.FRAMEWORK_ROUTER_ERROR,
|
|
133
|
-
httpStatus: 500,
|
|
134
|
-
message: `Handler not registered for route: ${routeId}`,
|
|
135
|
-
summary: "핸들러 미등록 - generate 재실행 필요",
|
|
136
|
-
fix: {
|
|
137
|
-
file:
|
|
130
|
+
return {
|
|
131
|
+
errorType: "FRAMEWORK_BUG",
|
|
132
|
+
code: ErrorCode.FRAMEWORK_ROUTER_ERROR,
|
|
133
|
+
httpStatus: 500,
|
|
134
|
+
message: `Handler not registered for route: ${routeId}`,
|
|
135
|
+
summary: "핸들러 미등록 - generate 재실행 필요",
|
|
136
|
+
fix: {
|
|
137
|
+
file: `.mandu/generated/server/routes/${routeId}.route.ts`,
|
|
138
138
|
suggestion: "bunx mandu generate를 실행하세요",
|
|
139
139
|
},
|
|
140
140
|
route: {
|
|
@@ -153,14 +153,14 @@ export function createPageLoadErrorResponse(
|
|
|
153
153
|
pattern: string,
|
|
154
154
|
originalError?: Error
|
|
155
155
|
): ManduError {
|
|
156
|
-
const error: ManduError = {
|
|
157
|
-
errorType: "LOGIC_ERROR",
|
|
158
|
-
code: ErrorCode.SLOT_IMPORT_ERROR,
|
|
159
|
-
httpStatus: 500,
|
|
160
|
-
message: originalError?.message || `Failed to load page module for route: ${routeId}`,
|
|
161
|
-
summary: `페이지 모듈 로드 실패 - ${routeId}.route.tsx 확인 필요`,
|
|
162
|
-
fix: {
|
|
163
|
-
file:
|
|
156
|
+
const error: ManduError = {
|
|
157
|
+
errorType: "LOGIC_ERROR",
|
|
158
|
+
code: ErrorCode.SLOT_IMPORT_ERROR,
|
|
159
|
+
httpStatus: 500,
|
|
160
|
+
message: originalError?.message || `Failed to load page module for route: ${routeId}`,
|
|
161
|
+
summary: `페이지 모듈 로드 실패 - ${routeId}.route.tsx 확인 필요`,
|
|
162
|
+
fix: {
|
|
163
|
+
file: `.mandu/generated/web/routes/${routeId}.route.tsx`,
|
|
164
164
|
suggestion: "import 경로와 컴포넌트 export를 확인하세요",
|
|
165
165
|
},
|
|
166
166
|
route: {
|
|
@@ -189,14 +189,14 @@ export function createSSRErrorResponse(
|
|
|
189
189
|
pattern: string,
|
|
190
190
|
originalError?: Error
|
|
191
191
|
): ManduError {
|
|
192
|
-
const error: ManduError = {
|
|
193
|
-
errorType: "FRAMEWORK_BUG",
|
|
194
|
-
code: ErrorCode.FRAMEWORK_SSR_ERROR,
|
|
195
|
-
httpStatus: 500,
|
|
196
|
-
message: originalError?.message || `SSR rendering failed for route: ${routeId}`,
|
|
197
|
-
summary: `SSR 렌더링 실패 - 컴포넌트 확인 필요`,
|
|
198
|
-
fix: {
|
|
199
|
-
file:
|
|
192
|
+
const error: ManduError = {
|
|
193
|
+
errorType: "FRAMEWORK_BUG",
|
|
194
|
+
code: ErrorCode.FRAMEWORK_SSR_ERROR,
|
|
195
|
+
httpStatus: 500,
|
|
196
|
+
message: originalError?.message || `SSR rendering failed for route: ${routeId}`,
|
|
197
|
+
summary: `SSR 렌더링 실패 - 컴포넌트 확인 필요`,
|
|
198
|
+
fix: {
|
|
199
|
+
file: `.mandu/generated/web/routes/${routeId}.route.tsx`,
|
|
200
200
|
suggestion: "React 컴포넌트가 서버에서 렌더링 가능한지 확인하세요 (브라우저 전용 API 사용 금지)",
|
|
201
201
|
},
|
|
202
202
|
route: {
|