@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.
Files changed (155) 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 +2 -0
  87. package/src/filling/sse.test.ts +168 -0
  88. package/src/filling/sse.ts +162 -0
  89. package/src/generator/index.ts +3 -3
  90. package/src/guard/analyzer.ts +360 -360
  91. package/src/guard/ast-analyzer.ts +806 -806
  92. package/src/guard/contract-guard.ts +9 -9
  93. package/src/guard/file-type.test.ts +24 -24
  94. package/src/guard/presets/atomic.ts +70 -70
  95. package/src/guard/presets/clean.ts +77 -77
  96. package/src/guard/presets/fsd.ts +79 -79
  97. package/src/guard/presets/hexagonal.ts +68 -68
  98. package/src/guard/presets/index.ts +291 -291
  99. package/src/guard/reporter.ts +445 -445
  100. package/src/guard/rules.ts +12 -12
  101. package/src/guard/statistics.ts +578 -578
  102. package/src/guard/suggestions.ts +358 -358
  103. package/src/guard/types.ts +348 -348
  104. package/src/guard/validator.ts +834 -834
  105. package/src/guard/watcher.ts +404 -404
  106. package/src/index.ts +6 -1
  107. package/src/intent/index.ts +310 -310
  108. package/src/island/index.ts +304 -304
  109. package/src/logging/index.ts +22 -22
  110. package/src/logging/transports.ts +365 -365
  111. package/src/plugins/index.ts +38 -38
  112. package/src/plugins/registry.ts +377 -377
  113. package/src/plugins/types.ts +363 -363
  114. package/src/report/index.ts +1 -1
  115. package/src/router/fs-patterns.ts +387 -387
  116. package/src/router/fs-scanner.ts +497 -497
  117. package/src/runtime/boundary.tsx +232 -232
  118. package/src/runtime/compose.ts +222 -222
  119. package/src/runtime/escape.ts +44 -0
  120. package/src/runtime/lifecycle.ts +381 -381
  121. package/src/runtime/logger.test.ts +345 -345
  122. package/src/runtime/logger.ts +677 -677
  123. package/src/runtime/router.test.ts +476 -476
  124. package/src/runtime/router.ts +105 -105
  125. package/src/runtime/security.ts +155 -155
  126. package/src/runtime/server.ts +257 -0
  127. package/src/runtime/session-key.ts +328 -328
  128. package/src/runtime/ssr.ts +16 -21
  129. package/src/runtime/streaming-ssr.ts +24 -33
  130. package/src/runtime/trace.ts +144 -144
  131. package/src/seo/index.ts +214 -214
  132. package/src/seo/integration/ssr.ts +307 -307
  133. package/src/seo/render/basic.ts +427 -427
  134. package/src/seo/render/index.ts +143 -143
  135. package/src/seo/render/jsonld.ts +539 -539
  136. package/src/seo/render/opengraph.ts +191 -191
  137. package/src/seo/render/robots.ts +116 -116
  138. package/src/seo/render/sitemap.ts +137 -137
  139. package/src/seo/render/twitter.ts +126 -126
  140. package/src/seo/resolve/index.ts +353 -353
  141. package/src/seo/resolve/opengraph.ts +143 -143
  142. package/src/seo/resolve/robots.ts +73 -73
  143. package/src/seo/resolve/title.ts +94 -94
  144. package/src/seo/resolve/twitter.ts +73 -73
  145. package/src/seo/resolve/url.ts +97 -97
  146. package/src/seo/routes/index.ts +290 -290
  147. package/src/seo/types.ts +575 -575
  148. package/src/slot/validator.ts +39 -39
  149. package/src/spec/index.ts +3 -3
  150. package/src/spec/load.ts +76 -76
  151. package/src/spec/lock.ts +56 -56
  152. package/src/utils/bun.ts +8 -8
  153. package/src/utils/lru-cache.ts +75 -75
  154. package/src/utils/safe-io.ts +188 -188
  155. 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,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
+ });