@mandujs/core 0.13.0 → 0.13.1
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 +4 -4
- package/README.md +653 -653
- package/package.json +1 -1
- package/src/bundler/build.ts +91 -91
- package/src/bundler/css.ts +302 -302
- 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 +9 -0
- package/src/config/validate.ts +12 -0
- 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/domains.ts +265 -265
- package/src/error/result.ts +46 -46
- 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 +24 -1
- package/src/filling/deps.ts +238 -238
- package/src/filling/index.ts +2 -0
- package/src/filling/sse.test.ts +168 -0
- package/src/filling/sse.ts +162 -0
- package/src/generator/index.ts +3 -3
- package/src/guard/analyzer.ts +360 -360
- package/src/guard/ast-analyzer.ts +806 -806
- 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/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 +6 -1
- 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/plugins/index.ts +38 -38
- package/src/plugins/registry.ts +377 -377
- package/src/plugins/types.ts +363 -363
- package/src/report/index.ts +1 -1
- package/src/router/fs-patterns.ts +387 -387
- package/src/router/fs-scanner.ts +497 -497
- package/src/runtime/boundary.tsx +232 -232
- package/src/runtime/compose.ts +222 -222
- package/src/runtime/escape.ts +44 -0
- 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 +257 -0
- package/src/runtime/session-key.ts +328 -328
- package/src/runtime/ssr.ts +16 -21
- package/src/runtime/streaming-ssr.ts +24 -33
- 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/filling/context.ts
CHANGED
|
@@ -9,7 +9,8 @@ import type { ZodSchema } from "zod";
|
|
|
9
9
|
import type { ContractSchema, ContractMethod } from "../contract/schema";
|
|
10
10
|
import type { InferBody, InferHeaders, InferParams, InferQuery, InferResponse } from "../contract/types";
|
|
11
11
|
import { ContractValidator, type ContractValidatorOptions } from "../contract/validator";
|
|
12
|
-
import { type FillingDeps,
|
|
12
|
+
import { type FillingDeps, globalDeps } from "./deps";
|
|
13
|
+
import { createSSEConnection, type SSEOptions, type SSEConnection } from "./sse";
|
|
13
14
|
|
|
14
15
|
type ContractInput<
|
|
15
16
|
TContract extends ContractSchema,
|
|
@@ -531,6 +532,28 @@ export class ManduContext {
|
|
|
531
532
|
return this.withCookies(response);
|
|
532
533
|
}
|
|
533
534
|
|
|
535
|
+
/**
|
|
536
|
+
* Create a Server-Sent Events (SSE) response.
|
|
537
|
+
*
|
|
538
|
+
* @example
|
|
539
|
+
* return ctx.sse((sse) => {
|
|
540
|
+
* sse.event("ready", { ok: true });
|
|
541
|
+
* const stop = sse.heartbeat(15000);
|
|
542
|
+
* sse.onClose(() => stop());
|
|
543
|
+
* });
|
|
544
|
+
*/
|
|
545
|
+
sse(setup?: (connection: SSEConnection) => void | Promise<void>, options: SSEOptions = {}): Response {
|
|
546
|
+
const connection = createSSEConnection(this.request.signal, options);
|
|
547
|
+
|
|
548
|
+
if (setup) {
|
|
549
|
+
Promise.resolve(setup(connection)).catch(() => {
|
|
550
|
+
void connection.close();
|
|
551
|
+
});
|
|
552
|
+
}
|
|
553
|
+
|
|
554
|
+
return this.withCookies(connection.response);
|
|
555
|
+
}
|
|
556
|
+
|
|
534
557
|
// ============================================
|
|
535
558
|
// 🥟 상태 저장 (Lifecycle → Handler 전달)
|
|
536
559
|
// ============================================
|
package/src/filling/deps.ts
CHANGED
|
@@ -1,238 +1,238 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* DNA-002: Dependency Injection Pattern
|
|
3
|
-
*
|
|
4
|
-
* Filling 핸들러의 의존성을 명시적으로 주입
|
|
5
|
-
* - 테스트 시 목킹 용이
|
|
6
|
-
* - 의존성 역전 원칙 (DIP) 준수
|
|
7
|
-
* - 외부 서비스와의 결합도 감소
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
/**
|
|
11
|
-
* 데이터베이스 의존성 인터페이스
|
|
12
|
-
*/
|
|
13
|
-
export interface DbDeps {
|
|
14
|
-
/**
|
|
15
|
-
* SQL 쿼리 실행
|
|
16
|
-
*/
|
|
17
|
-
query: <T>(sql: string, params?: unknown[]) => Promise<T>;
|
|
18
|
-
|
|
19
|
-
/**
|
|
20
|
-
* 트랜잭션 실행
|
|
21
|
-
*/
|
|
22
|
-
transaction: <T>(fn: () => Promise<T>) => Promise<T>;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
/**
|
|
26
|
-
* 캐시 의존성 인터페이스
|
|
27
|
-
*/
|
|
28
|
-
export interface CacheDeps {
|
|
29
|
-
/**
|
|
30
|
-
* 캐시에서 값 조회
|
|
31
|
-
*/
|
|
32
|
-
get: <T>(key: string) => Promise<T | null>;
|
|
33
|
-
|
|
34
|
-
/**
|
|
35
|
-
* 캐시에 값 저장
|
|
36
|
-
* @param ttl - Time to live (초 단위)
|
|
37
|
-
*/
|
|
38
|
-
set: <T>(key: string, value: T, ttl?: number) => Promise<void>;
|
|
39
|
-
|
|
40
|
-
/**
|
|
41
|
-
* 캐시에서 값 삭제
|
|
42
|
-
*/
|
|
43
|
-
delete: (key: string) => Promise<void>;
|
|
44
|
-
|
|
45
|
-
/**
|
|
46
|
-
* 패턴에 맞는 키 삭제
|
|
47
|
-
*/
|
|
48
|
-
deletePattern?: (pattern: string) => Promise<void>;
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
/**
|
|
52
|
-
* 로거 의존성 인터페이스
|
|
53
|
-
*/
|
|
54
|
-
export interface LoggerDeps {
|
|
55
|
-
debug: (msg: string, data?: unknown) => void;
|
|
56
|
-
info: (msg: string, data?: unknown) => void;
|
|
57
|
-
warn: (msg: string, data?: unknown) => void;
|
|
58
|
-
error: (msg: string, data?: unknown) => void;
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
/**
|
|
62
|
-
* 이벤트 버스 의존성 인터페이스
|
|
63
|
-
*/
|
|
64
|
-
export interface EventBusDeps {
|
|
65
|
-
/**
|
|
66
|
-
* 이벤트 발행
|
|
67
|
-
*/
|
|
68
|
-
emit: (event: string, payload: unknown) => void;
|
|
69
|
-
|
|
70
|
-
/**
|
|
71
|
-
* 이벤트 구독
|
|
72
|
-
*/
|
|
73
|
-
on: (event: string, handler: (payload: unknown) => void) => () => void;
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
/**
|
|
77
|
-
* Filling 핸들러 의존성 타입
|
|
78
|
-
*
|
|
79
|
-
* 모든 필드는 선택적 → 필요한 것만 주입
|
|
80
|
-
*/
|
|
81
|
-
export interface FillingDeps {
|
|
82
|
-
/**
|
|
83
|
-
* 데이터베이스 접근
|
|
84
|
-
*/
|
|
85
|
-
db?: DbDeps;
|
|
86
|
-
|
|
87
|
-
/**
|
|
88
|
-
* 캐시 접근
|
|
89
|
-
*/
|
|
90
|
-
cache?: CacheDeps;
|
|
91
|
-
|
|
92
|
-
/**
|
|
93
|
-
* HTTP 클라이언트
|
|
94
|
-
*/
|
|
95
|
-
fetch?: typeof fetch;
|
|
96
|
-
|
|
97
|
-
/**
|
|
98
|
-
* 로거
|
|
99
|
-
*/
|
|
100
|
-
logger?: LoggerDeps;
|
|
101
|
-
|
|
102
|
-
/**
|
|
103
|
-
* 이벤트 버스
|
|
104
|
-
*/
|
|
105
|
-
events?: EventBusDeps;
|
|
106
|
-
|
|
107
|
-
/**
|
|
108
|
-
* 현재 시간 (테스트용)
|
|
109
|
-
*/
|
|
110
|
-
now?: () => Date;
|
|
111
|
-
|
|
112
|
-
/**
|
|
113
|
-
* UUID 생성 (테스트용)
|
|
114
|
-
*/
|
|
115
|
-
uuid?: () => string;
|
|
116
|
-
|
|
117
|
-
/**
|
|
118
|
-
* 커스텀 의존성
|
|
119
|
-
*/
|
|
120
|
-
[key: string]: unknown;
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
/**
|
|
124
|
-
* 기본 의존성 생성
|
|
125
|
-
*
|
|
126
|
-
* @example
|
|
127
|
-
* ```ts
|
|
128
|
-
* const deps = createDefaultDeps();
|
|
129
|
-
* console.log(deps.now()); // 현재 시간
|
|
130
|
-
* ```
|
|
131
|
-
*/
|
|
132
|
-
export function createDefaultDeps(): FillingDeps {
|
|
133
|
-
return {
|
|
134
|
-
fetch: globalThis.fetch,
|
|
135
|
-
logger: {
|
|
136
|
-
debug: (msg, data) => console.debug(`[DEBUG] ${msg}`, data ?? ""),
|
|
137
|
-
info: (msg, data) => console.info(`[INFO] ${msg}`, data ?? ""),
|
|
138
|
-
warn: (msg, data) => console.warn(`[WARN] ${msg}`, data ?? ""),
|
|
139
|
-
error: (msg, data) => console.error(`[ERROR] ${msg}`, data ?? ""),
|
|
140
|
-
},
|
|
141
|
-
now: () => new Date(),
|
|
142
|
-
uuid: () => crypto.randomUUID(),
|
|
143
|
-
};
|
|
144
|
-
}
|
|
145
|
-
|
|
146
|
-
/**
|
|
147
|
-
* 테스트용 목 의존성 생성 헬퍼
|
|
148
|
-
*
|
|
149
|
-
* @example
|
|
150
|
-
* ```ts
|
|
151
|
-
* const mockDeps = createMockDeps({
|
|
152
|
-
* db: {
|
|
153
|
-
* query: vi.fn().mockResolvedValue([{ id: 1, name: "Test" }]),
|
|
154
|
-
* transaction: vi.fn(fn => fn()),
|
|
155
|
-
* },
|
|
156
|
-
* now: () => new Date("2025-01-01"),
|
|
157
|
-
* });
|
|
158
|
-
* ```
|
|
159
|
-
*/
|
|
160
|
-
export function createMockDeps(overrides: Partial<FillingDeps> = {}): FillingDeps {
|
|
161
|
-
const noop = () => {};
|
|
162
|
-
const asyncNoop = async () => {};
|
|
163
|
-
|
|
164
|
-
return {
|
|
165
|
-
db: {
|
|
166
|
-
query: async () => [] as any,
|
|
167
|
-
transaction: async (fn) => fn(),
|
|
168
|
-
},
|
|
169
|
-
cache: {
|
|
170
|
-
get: async () => null,
|
|
171
|
-
set: asyncNoop,
|
|
172
|
-
delete: asyncNoop,
|
|
173
|
-
},
|
|
174
|
-
fetch: async () => new Response(),
|
|
175
|
-
logger: {
|
|
176
|
-
debug: noop,
|
|
177
|
-
info: noop,
|
|
178
|
-
warn: noop,
|
|
179
|
-
error: noop,
|
|
180
|
-
},
|
|
181
|
-
events: {
|
|
182
|
-
emit: noop,
|
|
183
|
-
on: () => noop,
|
|
184
|
-
},
|
|
185
|
-
now: () => new Date("2025-01-01T00:00:00Z"),
|
|
186
|
-
uuid: () => "00000000-0000-0000-0000-000000000000",
|
|
187
|
-
...overrides,
|
|
188
|
-
};
|
|
189
|
-
}
|
|
190
|
-
|
|
191
|
-
/**
|
|
192
|
-
* 의존성 병합 (기본값 + 커스텀)
|
|
193
|
-
*/
|
|
194
|
-
export function mergeDeps(
|
|
195
|
-
base: FillingDeps,
|
|
196
|
-
overrides: Partial<FillingDeps>
|
|
197
|
-
): FillingDeps {
|
|
198
|
-
return { ...base, ...overrides };
|
|
199
|
-
}
|
|
200
|
-
|
|
201
|
-
/**
|
|
202
|
-
* 의존성 컨테이너 (싱글톤 관리)
|
|
203
|
-
*/
|
|
204
|
-
class DepsContainer {
|
|
205
|
-
private deps: FillingDeps = createDefaultDeps();
|
|
206
|
-
|
|
207
|
-
/**
|
|
208
|
-
* 전역 의존성 설정
|
|
209
|
-
*/
|
|
210
|
-
set(deps: Partial<FillingDeps>): void {
|
|
211
|
-
this.deps = mergeDeps(this.deps, deps);
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
/**
|
|
215
|
-
* 전역 의존성 가져오기
|
|
216
|
-
*/
|
|
217
|
-
get(): FillingDeps {
|
|
218
|
-
return this.deps;
|
|
219
|
-
}
|
|
220
|
-
|
|
221
|
-
/**
|
|
222
|
-
* 기본값으로 리셋
|
|
223
|
-
*/
|
|
224
|
-
reset(): void {
|
|
225
|
-
this.deps = createDefaultDeps();
|
|
226
|
-
}
|
|
227
|
-
}
|
|
228
|
-
|
|
229
|
-
/**
|
|
230
|
-
* 전역 의존성 컨테이너
|
|
231
|
-
*/
|
|
232
|
-
export const globalDeps = new DepsContainer();
|
|
233
|
-
|
|
234
|
-
/**
|
|
235
|
-
* 의존성 주입 데코레이터 타입
|
|
236
|
-
* (향후 클래스 기반 핸들러 지원 시)
|
|
237
|
-
*/
|
|
238
|
-
export type InjectDeps<T extends keyof FillingDeps> = Pick<FillingDeps, T>;
|
|
1
|
+
/**
|
|
2
|
+
* DNA-002: Dependency Injection Pattern
|
|
3
|
+
*
|
|
4
|
+
* Filling 핸들러의 의존성을 명시적으로 주입
|
|
5
|
+
* - 테스트 시 목킹 용이
|
|
6
|
+
* - 의존성 역전 원칙 (DIP) 준수
|
|
7
|
+
* - 외부 서비스와의 결합도 감소
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* 데이터베이스 의존성 인터페이스
|
|
12
|
+
*/
|
|
13
|
+
export interface DbDeps {
|
|
14
|
+
/**
|
|
15
|
+
* SQL 쿼리 실행
|
|
16
|
+
*/
|
|
17
|
+
query: <T>(sql: string, params?: unknown[]) => Promise<T>;
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* 트랜잭션 실행
|
|
21
|
+
*/
|
|
22
|
+
transaction: <T>(fn: () => Promise<T>) => Promise<T>;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* 캐시 의존성 인터페이스
|
|
27
|
+
*/
|
|
28
|
+
export interface CacheDeps {
|
|
29
|
+
/**
|
|
30
|
+
* 캐시에서 값 조회
|
|
31
|
+
*/
|
|
32
|
+
get: <T>(key: string) => Promise<T | null>;
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* 캐시에 값 저장
|
|
36
|
+
* @param ttl - Time to live (초 단위)
|
|
37
|
+
*/
|
|
38
|
+
set: <T>(key: string, value: T, ttl?: number) => Promise<void>;
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* 캐시에서 값 삭제
|
|
42
|
+
*/
|
|
43
|
+
delete: (key: string) => Promise<void>;
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* 패턴에 맞는 키 삭제
|
|
47
|
+
*/
|
|
48
|
+
deletePattern?: (pattern: string) => Promise<void>;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* 로거 의존성 인터페이스
|
|
53
|
+
*/
|
|
54
|
+
export interface LoggerDeps {
|
|
55
|
+
debug: (msg: string, data?: unknown) => void;
|
|
56
|
+
info: (msg: string, data?: unknown) => void;
|
|
57
|
+
warn: (msg: string, data?: unknown) => void;
|
|
58
|
+
error: (msg: string, data?: unknown) => void;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* 이벤트 버스 의존성 인터페이스
|
|
63
|
+
*/
|
|
64
|
+
export interface EventBusDeps {
|
|
65
|
+
/**
|
|
66
|
+
* 이벤트 발행
|
|
67
|
+
*/
|
|
68
|
+
emit: (event: string, payload: unknown) => void;
|
|
69
|
+
|
|
70
|
+
/**
|
|
71
|
+
* 이벤트 구독
|
|
72
|
+
*/
|
|
73
|
+
on: (event: string, handler: (payload: unknown) => void) => () => void;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
/**
|
|
77
|
+
* Filling 핸들러 의존성 타입
|
|
78
|
+
*
|
|
79
|
+
* 모든 필드는 선택적 → 필요한 것만 주입
|
|
80
|
+
*/
|
|
81
|
+
export interface FillingDeps {
|
|
82
|
+
/**
|
|
83
|
+
* 데이터베이스 접근
|
|
84
|
+
*/
|
|
85
|
+
db?: DbDeps;
|
|
86
|
+
|
|
87
|
+
/**
|
|
88
|
+
* 캐시 접근
|
|
89
|
+
*/
|
|
90
|
+
cache?: CacheDeps;
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* HTTP 클라이언트
|
|
94
|
+
*/
|
|
95
|
+
fetch?: typeof fetch;
|
|
96
|
+
|
|
97
|
+
/**
|
|
98
|
+
* 로거
|
|
99
|
+
*/
|
|
100
|
+
logger?: LoggerDeps;
|
|
101
|
+
|
|
102
|
+
/**
|
|
103
|
+
* 이벤트 버스
|
|
104
|
+
*/
|
|
105
|
+
events?: EventBusDeps;
|
|
106
|
+
|
|
107
|
+
/**
|
|
108
|
+
* 현재 시간 (테스트용)
|
|
109
|
+
*/
|
|
110
|
+
now?: () => Date;
|
|
111
|
+
|
|
112
|
+
/**
|
|
113
|
+
* UUID 생성 (테스트용)
|
|
114
|
+
*/
|
|
115
|
+
uuid?: () => string;
|
|
116
|
+
|
|
117
|
+
/**
|
|
118
|
+
* 커스텀 의존성
|
|
119
|
+
*/
|
|
120
|
+
[key: string]: unknown;
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
/**
|
|
124
|
+
* 기본 의존성 생성
|
|
125
|
+
*
|
|
126
|
+
* @example
|
|
127
|
+
* ```ts
|
|
128
|
+
* const deps = createDefaultDeps();
|
|
129
|
+
* console.log(deps.now()); // 현재 시간
|
|
130
|
+
* ```
|
|
131
|
+
*/
|
|
132
|
+
export function createDefaultDeps(): FillingDeps {
|
|
133
|
+
return {
|
|
134
|
+
fetch: globalThis.fetch,
|
|
135
|
+
logger: {
|
|
136
|
+
debug: (msg, data) => console.debug(`[DEBUG] ${msg}`, data ?? ""),
|
|
137
|
+
info: (msg, data) => console.info(`[INFO] ${msg}`, data ?? ""),
|
|
138
|
+
warn: (msg, data) => console.warn(`[WARN] ${msg}`, data ?? ""),
|
|
139
|
+
error: (msg, data) => console.error(`[ERROR] ${msg}`, data ?? ""),
|
|
140
|
+
},
|
|
141
|
+
now: () => new Date(),
|
|
142
|
+
uuid: () => crypto.randomUUID(),
|
|
143
|
+
};
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
/**
|
|
147
|
+
* 테스트용 목 의존성 생성 헬퍼
|
|
148
|
+
*
|
|
149
|
+
* @example
|
|
150
|
+
* ```ts
|
|
151
|
+
* const mockDeps = createMockDeps({
|
|
152
|
+
* db: {
|
|
153
|
+
* query: vi.fn().mockResolvedValue([{ id: 1, name: "Test" }]),
|
|
154
|
+
* transaction: vi.fn(fn => fn()),
|
|
155
|
+
* },
|
|
156
|
+
* now: () => new Date("2025-01-01"),
|
|
157
|
+
* });
|
|
158
|
+
* ```
|
|
159
|
+
*/
|
|
160
|
+
export function createMockDeps(overrides: Partial<FillingDeps> = {}): FillingDeps {
|
|
161
|
+
const noop = () => {};
|
|
162
|
+
const asyncNoop = async () => {};
|
|
163
|
+
|
|
164
|
+
return {
|
|
165
|
+
db: {
|
|
166
|
+
query: async () => [] as any,
|
|
167
|
+
transaction: async (fn) => fn(),
|
|
168
|
+
},
|
|
169
|
+
cache: {
|
|
170
|
+
get: async () => null,
|
|
171
|
+
set: asyncNoop,
|
|
172
|
+
delete: asyncNoop,
|
|
173
|
+
},
|
|
174
|
+
fetch: async () => new Response(),
|
|
175
|
+
logger: {
|
|
176
|
+
debug: noop,
|
|
177
|
+
info: noop,
|
|
178
|
+
warn: noop,
|
|
179
|
+
error: noop,
|
|
180
|
+
},
|
|
181
|
+
events: {
|
|
182
|
+
emit: noop,
|
|
183
|
+
on: () => noop,
|
|
184
|
+
},
|
|
185
|
+
now: () => new Date("2025-01-01T00:00:00Z"),
|
|
186
|
+
uuid: () => "00000000-0000-0000-0000-000000000000",
|
|
187
|
+
...overrides,
|
|
188
|
+
};
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
/**
|
|
192
|
+
* 의존성 병합 (기본값 + 커스텀)
|
|
193
|
+
*/
|
|
194
|
+
export function mergeDeps(
|
|
195
|
+
base: FillingDeps,
|
|
196
|
+
overrides: Partial<FillingDeps>
|
|
197
|
+
): FillingDeps {
|
|
198
|
+
return { ...base, ...overrides };
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
/**
|
|
202
|
+
* 의존성 컨테이너 (싱글톤 관리)
|
|
203
|
+
*/
|
|
204
|
+
class DepsContainer {
|
|
205
|
+
private deps: FillingDeps = createDefaultDeps();
|
|
206
|
+
|
|
207
|
+
/**
|
|
208
|
+
* 전역 의존성 설정
|
|
209
|
+
*/
|
|
210
|
+
set(deps: Partial<FillingDeps>): void {
|
|
211
|
+
this.deps = mergeDeps(this.deps, deps);
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
/**
|
|
215
|
+
* 전역 의존성 가져오기
|
|
216
|
+
*/
|
|
217
|
+
get(): FillingDeps {
|
|
218
|
+
return this.deps;
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
/**
|
|
222
|
+
* 기본값으로 리셋
|
|
223
|
+
*/
|
|
224
|
+
reset(): void {
|
|
225
|
+
this.deps = createDefaultDeps();
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
/**
|
|
230
|
+
* 전역 의존성 컨테이너
|
|
231
|
+
*/
|
|
232
|
+
export const globalDeps = new DepsContainer();
|
|
233
|
+
|
|
234
|
+
/**
|
|
235
|
+
* 의존성 주입 데코레이터 타입
|
|
236
|
+
* (향후 클래스 기반 핸들러 지원 시)
|
|
237
|
+
*/
|
|
238
|
+
export type InjectDeps<T extends keyof FillingDeps> = Pick<FillingDeps, T>;
|
package/src/filling/index.ts
CHANGED
|
@@ -8,6 +8,8 @@ export { ManduContext, ValidationError, CookieManager } from "./context";
|
|
|
8
8
|
export type { CookieOptions } from "./context";
|
|
9
9
|
export { ManduFilling, ManduFillingFactory, LoaderTimeoutError } from "./filling";
|
|
10
10
|
export type { Handler, Guard, HttpMethod, Loader, LoaderOptions } from "./filling";
|
|
11
|
+
export { SSEConnection, createSSEConnection } from "./sse";
|
|
12
|
+
export type { SSEOptions, SSESendOptions, SSECleanup } from "./sse";
|
|
11
13
|
|
|
12
14
|
// Auth Guards
|
|
13
15
|
export {
|
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
import { describe, it, expect, mock } from "bun:test";
|
|
2
|
+
import { createSSEConnection } from "./sse";
|
|
3
|
+
import { ManduContext } from "./context";
|
|
4
|
+
|
|
5
|
+
async function readChunk(response: Response): Promise<string> {
|
|
6
|
+
const reader = response.body?.getReader();
|
|
7
|
+
if (!reader) throw new Error("Missing response body");
|
|
8
|
+
|
|
9
|
+
const { value, done } = await reader.read();
|
|
10
|
+
if (done || !value) return "";
|
|
11
|
+
return new TextDecoder().decode(value);
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
describe("SSEConnection", () => {
|
|
15
|
+
it("streams real-time chunks and finishes with done=true after close", async () => {
|
|
16
|
+
const server = Bun.serve({
|
|
17
|
+
port: 0,
|
|
18
|
+
fetch(req) {
|
|
19
|
+
const ctx = new ManduContext(req);
|
|
20
|
+
return ctx.sse(async (sse) => {
|
|
21
|
+
sse.event("tick", { step: 1 }, { id: "1" });
|
|
22
|
+
await new Promise((resolve) => setTimeout(resolve, 20));
|
|
23
|
+
sse.event("tick", { step: 2 }, { id: "2" });
|
|
24
|
+
await sse.close();
|
|
25
|
+
});
|
|
26
|
+
},
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
try {
|
|
30
|
+
const response = await fetch(`http://127.0.0.1:${server.port}/stream`);
|
|
31
|
+
expect(response.headers.get("content-type")).toContain("text/event-stream");
|
|
32
|
+
|
|
33
|
+
const reader = response.body?.getReader();
|
|
34
|
+
if (!reader) throw new Error("Missing response body");
|
|
35
|
+
|
|
36
|
+
const firstRead = await reader.read();
|
|
37
|
+
expect(firstRead.done).toBe(false);
|
|
38
|
+
const firstChunk = new TextDecoder().decode(firstRead.value);
|
|
39
|
+
expect(firstChunk).toContain("event: tick");
|
|
40
|
+
expect(firstChunk).toContain('data: {"step":1}');
|
|
41
|
+
|
|
42
|
+
const secondRead = await reader.read();
|
|
43
|
+
expect(secondRead.done).toBe(false);
|
|
44
|
+
const secondChunk = new TextDecoder().decode(secondRead.value);
|
|
45
|
+
expect(secondChunk).toContain('data: {"step":2}');
|
|
46
|
+
|
|
47
|
+
const finalRead = await reader.read();
|
|
48
|
+
expect(finalRead.done).toBe(true);
|
|
49
|
+
} finally {
|
|
50
|
+
server.stop(true);
|
|
51
|
+
}
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
it("closes stream when context-level SSE setup throws (error path)", async () => {
|
|
55
|
+
const ctx = new ManduContext(new Request("http://localhost/realtime-error"));
|
|
56
|
+
|
|
57
|
+
const response = ctx.sse(async () => {
|
|
58
|
+
throw new Error("setup failed");
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
const reader = response.body?.getReader();
|
|
62
|
+
if (!reader) throw new Error("Missing response body");
|
|
63
|
+
|
|
64
|
+
// setup error is swallowed intentionally, but stream must close cleanly.
|
|
65
|
+
await Promise.resolve();
|
|
66
|
+
const read = await reader.read();
|
|
67
|
+
expect(read.done).toBe(true);
|
|
68
|
+
});
|
|
69
|
+
it("sends SSE event payload with metadata", async () => {
|
|
70
|
+
const sse = createSSEConnection();
|
|
71
|
+
|
|
72
|
+
sse.send({ ok: true }, { event: "ready", id: "1", retry: 3000 });
|
|
73
|
+
const chunk = await readChunk(sse.response);
|
|
74
|
+
|
|
75
|
+
expect(chunk).toContain("event: ready");
|
|
76
|
+
expect(chunk).toContain("id: 1");
|
|
77
|
+
expect(chunk).toContain("retry: 3000");
|
|
78
|
+
expect(chunk).toContain('data: {"ok":true}');
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
it("sanitizes event/id fields to prevent SSE injection", async () => {
|
|
82
|
+
const sse = createSSEConnection();
|
|
83
|
+
|
|
84
|
+
sse.send("payload", {
|
|
85
|
+
event: "update\nretry:0",
|
|
86
|
+
id: "abc\r\ndata: injected",
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
const chunk = await readChunk(sse.response);
|
|
90
|
+
expect(chunk).toContain("event: update retry:0");
|
|
91
|
+
expect(chunk).toContain("id: abc data: injected");
|
|
92
|
+
expect(chunk).not.toContain("\nevent: update\nretry:0\n");
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
it("normalizes payload lines across CR/LF variants", async () => {
|
|
96
|
+
const sse = createSSEConnection();
|
|
97
|
+
|
|
98
|
+
sse.send("a\rb\nc\r\nd");
|
|
99
|
+
const chunk = await readChunk(sse.response);
|
|
100
|
+
|
|
101
|
+
expect(chunk).toContain("data: a");
|
|
102
|
+
expect(chunk).toContain("data: b");
|
|
103
|
+
expect(chunk).toContain("data: c");
|
|
104
|
+
expect(chunk).toContain("data: d");
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
it("registers heartbeat and cleanup on close", async () => {
|
|
108
|
+
const sse = createSSEConnection();
|
|
109
|
+
const cleanup = mock(() => {});
|
|
110
|
+
|
|
111
|
+
const stop = sse.heartbeat(1000, "ping");
|
|
112
|
+
sse.onClose(cleanup);
|
|
113
|
+
|
|
114
|
+
await sse.close();
|
|
115
|
+
|
|
116
|
+
expect(cleanup).toHaveBeenCalledTimes(1);
|
|
117
|
+
expect(typeof stop).toBe("function");
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
it("continues closing when cleanup handlers throw", async () => {
|
|
121
|
+
const sse = createSSEConnection();
|
|
122
|
+
const badCleanup = mock(() => {
|
|
123
|
+
throw new Error("cleanup failed");
|
|
124
|
+
});
|
|
125
|
+
const goodCleanup = mock(() => {});
|
|
126
|
+
|
|
127
|
+
sse.onClose(badCleanup);
|
|
128
|
+
sse.onClose(goodCleanup);
|
|
129
|
+
|
|
130
|
+
await expect(sse.close()).resolves.toBeUndefined();
|
|
131
|
+
expect(badCleanup).toHaveBeenCalledTimes(1);
|
|
132
|
+
expect(goodCleanup).toHaveBeenCalledTimes(1);
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
it("does not throw when registering onClose after already closed", async () => {
|
|
136
|
+
const sse = createSSEConnection();
|
|
137
|
+
await sse.close();
|
|
138
|
+
|
|
139
|
+
const badCleanup = () => Promise.reject(new Error("late cleanup failed"));
|
|
140
|
+
expect(() => sse.onClose(badCleanup)).not.toThrow();
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
it("closes automatically when request signal aborts", async () => {
|
|
144
|
+
const controller = new AbortController();
|
|
145
|
+
const sse = createSSEConnection(controller.signal);
|
|
146
|
+
const cleanup = mock(() => {});
|
|
147
|
+
|
|
148
|
+
sse.onClose(cleanup);
|
|
149
|
+
controller.abort();
|
|
150
|
+
|
|
151
|
+
await Promise.resolve();
|
|
152
|
+
|
|
153
|
+
expect(cleanup).toHaveBeenCalledTimes(1);
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
it("supports context-level SSE response helper", async () => {
|
|
157
|
+
const ctx = new ManduContext(new Request("http://localhost/realtime"));
|
|
158
|
+
|
|
159
|
+
const response = ctx.sse((sse) => {
|
|
160
|
+
sse.event("message", "hello");
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
expect(response.headers.get("content-type")).toContain("text/event-stream");
|
|
164
|
+
const chunk = await readChunk(response);
|
|
165
|
+
expect(chunk).toContain("event: message");
|
|
166
|
+
expect(chunk).toContain("data: hello");
|
|
167
|
+
});
|
|
168
|
+
});
|