@mandujs/core 0.13.0 → 0.13.2
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 +4 -0
- package/src/filling/sse-catchup.test.ts +56 -0
- package/src/filling/sse-catchup.ts +67 -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,10 @@ 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";
|
|
13
|
+
export { resolveResumeCursor, catchupFromCursor, mergeUniqueById } from "./sse-catchup";
|
|
14
|
+
export type { SSECursor, CatchupResult, CatchupOptions } from "./sse-catchup";
|
|
11
15
|
|
|
12
16
|
// Auth Guards
|
|
13
17
|
export {
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import { describe, expect, it } from "bun:test";
|
|
2
|
+
import { catchupFromCursor, mergeUniqueById, resolveResumeCursor } from "./sse-catchup";
|
|
3
|
+
|
|
4
|
+
describe("sse catch-up primitives", () => {
|
|
5
|
+
const snapshot = [
|
|
6
|
+
{ id: "m1", text: "first" },
|
|
7
|
+
{ id: "m2", text: "second" },
|
|
8
|
+
{ id: "m3", text: "third" },
|
|
9
|
+
];
|
|
10
|
+
|
|
11
|
+
it("resolves Last-Event-ID cursor", () => {
|
|
12
|
+
const req = new Request("http://localhost/stream", {
|
|
13
|
+
headers: { "Last-Event-ID": "m2" },
|
|
14
|
+
});
|
|
15
|
+
|
|
16
|
+
expect(resolveResumeCursor(req)).toBe("m2");
|
|
17
|
+
});
|
|
18
|
+
|
|
19
|
+
it("returns delta list when cursor exists", () => {
|
|
20
|
+
const result = catchupFromCursor({ cursorId: "m1", snapshot });
|
|
21
|
+
|
|
22
|
+
expect(result.mode).toBe("delta");
|
|
23
|
+
expect(result.items.map((m) => m.id)).toEqual(["m2", "m3"]);
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
it("falls back to snapshot when cursor is missing", () => {
|
|
27
|
+
const result = catchupFromCursor({ snapshot });
|
|
28
|
+
|
|
29
|
+
expect(result.mode).toBe("snapshot");
|
|
30
|
+
expect(result.reason).toBe("missing-cursor");
|
|
31
|
+
expect(result.items.map((m) => m.id)).toEqual(["m1", "m2", "m3"]);
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it("falls back to snapshot when cursor is unknown", () => {
|
|
35
|
+
const result = catchupFromCursor({ cursorId: "missing", snapshot });
|
|
36
|
+
|
|
37
|
+
expect(result.mode).toBe("snapshot");
|
|
38
|
+
expect(result.reason).toBe("unknown-cursor");
|
|
39
|
+
expect(result.items.map((m) => m.id)).toEqual(["m1", "m2", "m3"]);
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
it("merges incoming messages idempotently", () => {
|
|
43
|
+
const base = [
|
|
44
|
+
{ id: "m1", text: "first" },
|
|
45
|
+
{ id: "m2", text: "second" },
|
|
46
|
+
];
|
|
47
|
+
const incoming = [
|
|
48
|
+
{ id: "m2", text: "second-dup" },
|
|
49
|
+
{ id: "m3", text: "third" },
|
|
50
|
+
];
|
|
51
|
+
|
|
52
|
+
const merged = mergeUniqueById(base, incoming);
|
|
53
|
+
|
|
54
|
+
expect(merged.map((m) => m.id)).toEqual(["m1", "m2", "m3"]);
|
|
55
|
+
});
|
|
56
|
+
});
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
export interface SSECursor {
|
|
2
|
+
id: string;
|
|
3
|
+
}
|
|
4
|
+
|
|
5
|
+
export interface CatchupResult<T> {
|
|
6
|
+
mode: "delta" | "snapshot";
|
|
7
|
+
items: T[];
|
|
8
|
+
reason?: "missing-cursor" | "unknown-cursor";
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export interface CatchupOptions<T extends SSECursor> {
|
|
12
|
+
cursorId?: string | null;
|
|
13
|
+
snapshot: readonly T[];
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Resolve resume cursor from Last-Event-ID header.
|
|
18
|
+
*/
|
|
19
|
+
export function resolveResumeCursor(request: Request): string | undefined {
|
|
20
|
+
const raw = request.headers.get("last-event-id")?.trim();
|
|
21
|
+
return raw ? raw : undefined;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Return reconnect catch-up list if cursor exists, otherwise full snapshot fallback.
|
|
26
|
+
*/
|
|
27
|
+
export function catchupFromCursor<T extends SSECursor>(options: CatchupOptions<T>): CatchupResult<T> {
|
|
28
|
+
const { cursorId, snapshot } = options;
|
|
29
|
+
|
|
30
|
+
if (!cursorId) {
|
|
31
|
+
return {
|
|
32
|
+
mode: "snapshot",
|
|
33
|
+
reason: "missing-cursor",
|
|
34
|
+
items: [...snapshot],
|
|
35
|
+
};
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
const index = snapshot.findIndex((item) => item.id === cursorId);
|
|
39
|
+
if (index < 0) {
|
|
40
|
+
return {
|
|
41
|
+
mode: "snapshot",
|
|
42
|
+
reason: "unknown-cursor",
|
|
43
|
+
items: [...snapshot],
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
return {
|
|
48
|
+
mode: "delta",
|
|
49
|
+
items: snapshot.slice(index + 1),
|
|
50
|
+
};
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Idempotent merge utility for reconnect + snapshot mix.
|
|
55
|
+
*/
|
|
56
|
+
export function mergeUniqueById<T extends SSECursor>(base: readonly T[], incoming: readonly T[]): T[] {
|
|
57
|
+
const seen = new Set(base.map((item) => item.id));
|
|
58
|
+
const merged = [...base];
|
|
59
|
+
|
|
60
|
+
for (const item of incoming) {
|
|
61
|
+
if (seen.has(item.id)) continue;
|
|
62
|
+
seen.add(item.id);
|
|
63
|
+
merged.push(item);
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
return merged;
|
|
67
|
+
}
|