@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.
Files changed (157) hide show
  1. package/README.ko.md +4 -4
  2. package/README.md +653 -653
  3. package/package.json +1 -1
  4. package/src/bundler/build.ts +91 -91
  5. package/src/bundler/css.ts +302 -302
  6. package/src/client/Link.tsx +227 -227
  7. package/src/client/globals.ts +44 -44
  8. package/src/client/hooks.ts +267 -267
  9. package/src/client/index.ts +5 -5
  10. package/src/client/island.ts +8 -8
  11. package/src/client/router.ts +435 -435
  12. package/src/client/runtime.ts +23 -23
  13. package/src/client/serialize.ts +404 -404
  14. package/src/client/window-state.ts +101 -101
  15. package/src/config/mandu.ts +9 -0
  16. package/src/config/validate.ts +12 -0
  17. package/src/config/watcher.ts +311 -311
  18. package/src/constants.ts +40 -40
  19. package/src/content/content-layer.ts +314 -314
  20. package/src/content/content.test.ts +433 -433
  21. package/src/content/data-store.ts +245 -245
  22. package/src/content/digest.ts +133 -133
  23. package/src/content/index.ts +164 -164
  24. package/src/content/loader-context.ts +172 -172
  25. package/src/content/loaders/api.ts +216 -216
  26. package/src/content/loaders/file.ts +169 -169
  27. package/src/content/loaders/glob.ts +252 -252
  28. package/src/content/loaders/index.ts +34 -34
  29. package/src/content/loaders/types.ts +137 -137
  30. package/src/content/meta-store.ts +209 -209
  31. package/src/content/types.ts +282 -282
  32. package/src/content/watcher.ts +135 -135
  33. package/src/contract/client-safe.test.ts +42 -42
  34. package/src/contract/client-safe.ts +114 -114
  35. package/src/contract/client.ts +16 -16
  36. package/src/contract/define.ts +459 -459
  37. package/src/contract/handler.ts +10 -10
  38. package/src/contract/normalize.test.ts +276 -276
  39. package/src/contract/normalize.ts +404 -404
  40. package/src/contract/registry.test.ts +206 -206
  41. package/src/contract/registry.ts +568 -568
  42. package/src/contract/schema.ts +48 -48
  43. package/src/contract/types.ts +58 -58
  44. package/src/contract/validator.ts +32 -32
  45. package/src/devtools/ai/context-builder.ts +375 -375
  46. package/src/devtools/ai/index.ts +25 -25
  47. package/src/devtools/ai/mcp-connector.ts +465 -465
  48. package/src/devtools/client/catchers/error-catcher.ts +327 -327
  49. package/src/devtools/client/catchers/index.ts +18 -18
  50. package/src/devtools/client/catchers/network-proxy.ts +363 -363
  51. package/src/devtools/client/components/index.ts +39 -39
  52. package/src/devtools/client/components/kitchen-root.tsx +362 -362
  53. package/src/devtools/client/components/mandu-character.tsx +241 -241
  54. package/src/devtools/client/components/overlay.tsx +368 -368
  55. package/src/devtools/client/components/panel/errors-panel.tsx +259 -259
  56. package/src/devtools/client/components/panel/guard-panel.tsx +244 -244
  57. package/src/devtools/client/components/panel/index.ts +32 -32
  58. package/src/devtools/client/components/panel/islands-panel.tsx +304 -304
  59. package/src/devtools/client/components/panel/network-panel.tsx +292 -292
  60. package/src/devtools/client/components/panel/panel-container.tsx +259 -259
  61. package/src/devtools/client/filters/context-filters.ts +282 -282
  62. package/src/devtools/client/filters/index.ts +16 -16
  63. package/src/devtools/client/index.ts +63 -63
  64. package/src/devtools/client/persistence.ts +335 -335
  65. package/src/devtools/client/state-manager.ts +478 -478
  66. package/src/devtools/design-tokens.ts +263 -263
  67. package/src/devtools/hook/create-hook.ts +207 -207
  68. package/src/devtools/hook/index.ts +13 -13
  69. package/src/devtools/index.ts +439 -439
  70. package/src/devtools/init.ts +266 -266
  71. package/src/devtools/protocol.ts +237 -237
  72. package/src/devtools/server/index.ts +17 -17
  73. package/src/devtools/server/source-context.ts +444 -444
  74. package/src/devtools/types.ts +319 -319
  75. package/src/devtools/worker/index.ts +25 -25
  76. package/src/devtools/worker/redaction-worker.ts +222 -222
  77. package/src/devtools/worker/worker-manager.ts +409 -409
  78. package/src/error/domains.ts +265 -265
  79. package/src/error/result.ts +46 -46
  80. package/src/error/types.ts +6 -6
  81. package/src/errors/extractor.ts +409 -409
  82. package/src/errors/index.ts +19 -19
  83. package/src/filling/auth.ts +308 -308
  84. package/src/filling/context.ts +24 -1
  85. package/src/filling/deps.ts +238 -238
  86. package/src/filling/index.ts +4 -0
  87. package/src/filling/sse-catchup.test.ts +56 -0
  88. package/src/filling/sse-catchup.ts +67 -0
  89. package/src/filling/sse.test.ts +168 -0
  90. package/src/filling/sse.ts +162 -0
  91. package/src/generator/index.ts +3 -3
  92. package/src/guard/analyzer.ts +360 -360
  93. package/src/guard/ast-analyzer.ts +806 -806
  94. package/src/guard/contract-guard.ts +9 -9
  95. package/src/guard/file-type.test.ts +24 -24
  96. package/src/guard/presets/atomic.ts +70 -70
  97. package/src/guard/presets/clean.ts +77 -77
  98. package/src/guard/presets/fsd.ts +79 -79
  99. package/src/guard/presets/hexagonal.ts +68 -68
  100. package/src/guard/presets/index.ts +291 -291
  101. package/src/guard/reporter.ts +445 -445
  102. package/src/guard/rules.ts +12 -12
  103. package/src/guard/statistics.ts +578 -578
  104. package/src/guard/suggestions.ts +358 -358
  105. package/src/guard/types.ts +348 -348
  106. package/src/guard/validator.ts +834 -834
  107. package/src/guard/watcher.ts +404 -404
  108. package/src/index.ts +6 -1
  109. package/src/intent/index.ts +310 -310
  110. package/src/island/index.ts +304 -304
  111. package/src/logging/index.ts +22 -22
  112. package/src/logging/transports.ts +365 -365
  113. package/src/plugins/index.ts +38 -38
  114. package/src/plugins/registry.ts +377 -377
  115. package/src/plugins/types.ts +363 -363
  116. package/src/report/index.ts +1 -1
  117. package/src/router/fs-patterns.ts +387 -387
  118. package/src/router/fs-scanner.ts +497 -497
  119. package/src/runtime/boundary.tsx +232 -232
  120. package/src/runtime/compose.ts +222 -222
  121. package/src/runtime/escape.ts +44 -0
  122. package/src/runtime/lifecycle.ts +381 -381
  123. package/src/runtime/logger.test.ts +345 -345
  124. package/src/runtime/logger.ts +677 -677
  125. package/src/runtime/router.test.ts +476 -476
  126. package/src/runtime/router.ts +105 -105
  127. package/src/runtime/security.ts +155 -155
  128. package/src/runtime/server.ts +257 -0
  129. package/src/runtime/session-key.ts +328 -328
  130. package/src/runtime/ssr.ts +16 -21
  131. package/src/runtime/streaming-ssr.ts +24 -33
  132. package/src/runtime/trace.ts +144 -144
  133. package/src/seo/index.ts +214 -214
  134. package/src/seo/integration/ssr.ts +307 -307
  135. package/src/seo/render/basic.ts +427 -427
  136. package/src/seo/render/index.ts +143 -143
  137. package/src/seo/render/jsonld.ts +539 -539
  138. package/src/seo/render/opengraph.ts +191 -191
  139. package/src/seo/render/robots.ts +116 -116
  140. package/src/seo/render/sitemap.ts +137 -137
  141. package/src/seo/render/twitter.ts +126 -126
  142. package/src/seo/resolve/index.ts +353 -353
  143. package/src/seo/resolve/opengraph.ts +143 -143
  144. package/src/seo/resolve/robots.ts +73 -73
  145. package/src/seo/resolve/title.ts +94 -94
  146. package/src/seo/resolve/twitter.ts +73 -73
  147. package/src/seo/resolve/url.ts +97 -97
  148. package/src/seo/routes/index.ts +290 -290
  149. package/src/seo/types.ts +575 -575
  150. package/src/slot/validator.ts +39 -39
  151. package/src/spec/index.ts +3 -3
  152. package/src/spec/load.ts +76 -76
  153. package/src/spec/lock.ts +56 -56
  154. package/src/utils/bun.ts +8 -8
  155. package/src/utils/lru-cache.ts +75 -75
  156. package/src/utils/safe-io.ts +188 -188
  157. package/src/utils/string-safe.ts +298 -298
@@ -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, createDefaultDeps, globalDeps } from "./deps";
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
  // ============================================
@@ -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>;
@@ -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
+ }