@auto-engineer/pipeline 0.0.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 (270) hide show
  1. package/.turbo/turbo-build.log +5 -0
  2. package/LICENSE +10 -0
  3. package/claude.md +160 -0
  4. package/dist/src/builder/define.d.ts +90 -0
  5. package/dist/src/builder/define.d.ts.map +1 -0
  6. package/dist/src/builder/define.js +425 -0
  7. package/dist/src/builder/define.js.map +1 -0
  8. package/dist/src/builder/define.specs.d.ts +2 -0
  9. package/dist/src/builder/define.specs.d.ts.map +1 -0
  10. package/dist/src/builder/define.specs.js +435 -0
  11. package/dist/src/builder/define.specs.js.map +1 -0
  12. package/dist/src/config/pipeline-config.d.ts +13 -0
  13. package/dist/src/config/pipeline-config.d.ts.map +1 -0
  14. package/dist/src/config/pipeline-config.js +15 -0
  15. package/dist/src/config/pipeline-config.js.map +1 -0
  16. package/dist/src/core/descriptors.d.ts +84 -0
  17. package/dist/src/core/descriptors.d.ts.map +1 -0
  18. package/dist/src/core/descriptors.js +2 -0
  19. package/dist/src/core/descriptors.js.map +1 -0
  20. package/dist/src/core/descriptors.specs.d.ts +2 -0
  21. package/dist/src/core/descriptors.specs.d.ts.map +1 -0
  22. package/dist/src/core/descriptors.specs.js +24 -0
  23. package/dist/src/core/descriptors.specs.js.map +1 -0
  24. package/dist/src/core/types.d.ts +10 -0
  25. package/dist/src/core/types.d.ts.map +1 -0
  26. package/dist/src/core/types.js +4 -0
  27. package/dist/src/core/types.js.map +1 -0
  28. package/dist/src/core/types.specs.d.ts +2 -0
  29. package/dist/src/core/types.specs.d.ts.map +1 -0
  30. package/dist/src/core/types.specs.js +40 -0
  31. package/dist/src/core/types.specs.js.map +1 -0
  32. package/dist/src/graph/types.d.ts +17 -0
  33. package/dist/src/graph/types.d.ts.map +1 -0
  34. package/dist/src/graph/types.js +2 -0
  35. package/dist/src/graph/types.js.map +1 -0
  36. package/dist/src/graph/types.specs.d.ts +2 -0
  37. package/dist/src/graph/types.specs.d.ts.map +1 -0
  38. package/dist/src/graph/types.specs.js +148 -0
  39. package/dist/src/graph/types.specs.js.map +1 -0
  40. package/dist/src/index.d.ts +20 -0
  41. package/dist/src/index.d.ts.map +1 -0
  42. package/dist/src/index.js +12 -0
  43. package/dist/src/index.js.map +1 -0
  44. package/dist/src/logging/event-logger.d.ts +21 -0
  45. package/dist/src/logging/event-logger.d.ts.map +1 -0
  46. package/dist/src/logging/event-logger.js +31 -0
  47. package/dist/src/logging/event-logger.js.map +1 -0
  48. package/dist/src/logging/event-logger.specs.d.ts +2 -0
  49. package/dist/src/logging/event-logger.specs.d.ts.map +1 -0
  50. package/dist/src/logging/event-logger.specs.js +81 -0
  51. package/dist/src/logging/event-logger.specs.js.map +1 -0
  52. package/dist/src/plugins/handler-adapter.d.ts +5 -0
  53. package/dist/src/plugins/handler-adapter.d.ts.map +1 -0
  54. package/dist/src/plugins/handler-adapter.js +17 -0
  55. package/dist/src/plugins/handler-adapter.js.map +1 -0
  56. package/dist/src/plugins/handler-adapter.specs.d.ts +2 -0
  57. package/dist/src/plugins/handler-adapter.specs.d.ts.map +1 -0
  58. package/dist/src/plugins/handler-adapter.specs.js +129 -0
  59. package/dist/src/plugins/handler-adapter.specs.js.map +1 -0
  60. package/dist/src/plugins/plugin-loader.d.ts +25 -0
  61. package/dist/src/plugins/plugin-loader.d.ts.map +1 -0
  62. package/dist/src/plugins/plugin-loader.js +150 -0
  63. package/dist/src/plugins/plugin-loader.js.map +1 -0
  64. package/dist/src/plugins/plugin-loader.specs.d.ts +2 -0
  65. package/dist/src/plugins/plugin-loader.specs.d.ts.map +1 -0
  66. package/dist/src/plugins/plugin-loader.specs.js +246 -0
  67. package/dist/src/plugins/plugin-loader.specs.js.map +1 -0
  68. package/dist/src/runtime/await-tracker.d.ts +10 -0
  69. package/dist/src/runtime/await-tracker.d.ts.map +1 -0
  70. package/dist/src/runtime/await-tracker.js +42 -0
  71. package/dist/src/runtime/await-tracker.js.map +1 -0
  72. package/dist/src/runtime/await-tracker.specs.d.ts +2 -0
  73. package/dist/src/runtime/await-tracker.specs.d.ts.map +1 -0
  74. package/dist/src/runtime/await-tracker.specs.js +46 -0
  75. package/dist/src/runtime/await-tracker.specs.js.map +1 -0
  76. package/dist/src/runtime/context.d.ts +12 -0
  77. package/dist/src/runtime/context.d.ts.map +1 -0
  78. package/dist/src/runtime/context.js +2 -0
  79. package/dist/src/runtime/context.js.map +1 -0
  80. package/dist/src/runtime/context.specs.d.ts +2 -0
  81. package/dist/src/runtime/context.specs.d.ts.map +1 -0
  82. package/dist/src/runtime/context.specs.js +26 -0
  83. package/dist/src/runtime/context.specs.js.map +1 -0
  84. package/dist/src/runtime/event-command-map.d.ts +15 -0
  85. package/dist/src/runtime/event-command-map.d.ts.map +1 -0
  86. package/dist/src/runtime/event-command-map.js +26 -0
  87. package/dist/src/runtime/event-command-map.js.map +1 -0
  88. package/dist/src/runtime/event-command-map.specs.d.ts +2 -0
  89. package/dist/src/runtime/event-command-map.specs.d.ts.map +1 -0
  90. package/dist/src/runtime/event-command-map.specs.js +108 -0
  91. package/dist/src/runtime/event-command-map.specs.js.map +1 -0
  92. package/dist/src/runtime/phased-executor.d.ts +29 -0
  93. package/dist/src/runtime/phased-executor.d.ts.map +1 -0
  94. package/dist/src/runtime/phased-executor.js +164 -0
  95. package/dist/src/runtime/phased-executor.js.map +1 -0
  96. package/dist/src/runtime/phased-executor.specs.d.ts +2 -0
  97. package/dist/src/runtime/phased-executor.specs.d.ts.map +1 -0
  98. package/dist/src/runtime/phased-executor.specs.js +256 -0
  99. package/dist/src/runtime/phased-executor.specs.js.map +1 -0
  100. package/dist/src/runtime/pipeline-runtime.d.ts +17 -0
  101. package/dist/src/runtime/pipeline-runtime.d.ts.map +1 -0
  102. package/dist/src/runtime/pipeline-runtime.js +87 -0
  103. package/dist/src/runtime/pipeline-runtime.js.map +1 -0
  104. package/dist/src/runtime/pipeline-runtime.specs.d.ts +2 -0
  105. package/dist/src/runtime/pipeline-runtime.specs.d.ts.map +1 -0
  106. package/dist/src/runtime/pipeline-runtime.specs.js +192 -0
  107. package/dist/src/runtime/pipeline-runtime.specs.js.map +1 -0
  108. package/dist/src/runtime/settled-tracker.d.ts +42 -0
  109. package/dist/src/runtime/settled-tracker.d.ts.map +1 -0
  110. package/dist/src/runtime/settled-tracker.js +161 -0
  111. package/dist/src/runtime/settled-tracker.js.map +1 -0
  112. package/dist/src/runtime/settled-tracker.specs.d.ts +2 -0
  113. package/dist/src/runtime/settled-tracker.specs.d.ts.map +1 -0
  114. package/dist/src/runtime/settled-tracker.specs.js +361 -0
  115. package/dist/src/runtime/settled-tracker.specs.js.map +1 -0
  116. package/dist/src/server/full-orchestration.e2e.specs.d.ts +2 -0
  117. package/dist/src/server/full-orchestration.e2e.specs.d.ts.map +1 -0
  118. package/dist/src/server/full-orchestration.e2e.specs.js +561 -0
  119. package/dist/src/server/full-orchestration.e2e.specs.js.map +1 -0
  120. package/dist/src/server/pipeline-server.d.ts +59 -0
  121. package/dist/src/server/pipeline-server.d.ts.map +1 -0
  122. package/dist/src/server/pipeline-server.e2e.specs.d.ts +2 -0
  123. package/dist/src/server/pipeline-server.e2e.specs.d.ts.map +1 -0
  124. package/dist/src/server/pipeline-server.e2e.specs.js +381 -0
  125. package/dist/src/server/pipeline-server.e2e.specs.js.map +1 -0
  126. package/dist/src/server/pipeline-server.js +527 -0
  127. package/dist/src/server/pipeline-server.js.map +1 -0
  128. package/dist/src/server/pipeline-server.specs.d.ts +2 -0
  129. package/dist/src/server/pipeline-server.specs.d.ts.map +1 -0
  130. package/dist/src/server/pipeline-server.specs.js +662 -0
  131. package/dist/src/server/pipeline-server.specs.js.map +1 -0
  132. package/dist/src/server/sse-manager.d.ts +12 -0
  133. package/dist/src/server/sse-manager.d.ts.map +1 -0
  134. package/dist/src/server/sse-manager.js +63 -0
  135. package/dist/src/server/sse-manager.js.map +1 -0
  136. package/dist/src/server/sse-manager.specs.d.ts +2 -0
  137. package/dist/src/server/sse-manager.specs.d.ts.map +1 -0
  138. package/dist/src/server/sse-manager.specs.js +158 -0
  139. package/dist/src/server/sse-manager.specs.js.map +1 -0
  140. package/dist/src/testing/event-capture.d.ts +14 -0
  141. package/dist/src/testing/event-capture.d.ts.map +1 -0
  142. package/dist/src/testing/event-capture.js +55 -0
  143. package/dist/src/testing/event-capture.js.map +1 -0
  144. package/dist/src/testing/event-capture.specs.d.ts +2 -0
  145. package/dist/src/testing/event-capture.specs.d.ts.map +1 -0
  146. package/dist/src/testing/event-capture.specs.js +114 -0
  147. package/dist/src/testing/event-capture.specs.js.map +1 -0
  148. package/dist/src/testing/fixtures/kanban-full.pipeline.d.ts +7 -0
  149. package/dist/src/testing/fixtures/kanban-full.pipeline.d.ts.map +1 -0
  150. package/dist/src/testing/fixtures/kanban-full.pipeline.js +168 -0
  151. package/dist/src/testing/fixtures/kanban-full.pipeline.js.map +1 -0
  152. package/dist/src/testing/fixtures/kanban-full.pipeline.specs.d.ts +2 -0
  153. package/dist/src/testing/fixtures/kanban-full.pipeline.specs.d.ts.map +1 -0
  154. package/dist/src/testing/fixtures/kanban-full.pipeline.specs.js +263 -0
  155. package/dist/src/testing/fixtures/kanban-full.pipeline.specs.js.map +1 -0
  156. package/dist/src/testing/fixtures/kanban-todo.config.d.ts +3 -0
  157. package/dist/src/testing/fixtures/kanban-todo.config.d.ts.map +1 -0
  158. package/dist/src/testing/fixtures/kanban-todo.config.js +19 -0
  159. package/dist/src/testing/fixtures/kanban-todo.config.js.map +1 -0
  160. package/dist/src/testing/fixtures/kanban.pipeline.d.ts +5 -0
  161. package/dist/src/testing/fixtures/kanban.pipeline.d.ts.map +1 -0
  162. package/dist/src/testing/fixtures/kanban.pipeline.js +76 -0
  163. package/dist/src/testing/fixtures/kanban.pipeline.js.map +1 -0
  164. package/dist/src/testing/fixtures/kanban.pipeline.specs.d.ts +2 -0
  165. package/dist/src/testing/fixtures/kanban.pipeline.specs.d.ts.map +1 -0
  166. package/dist/src/testing/fixtures/kanban.pipeline.specs.js +29 -0
  167. package/dist/src/testing/fixtures/kanban.pipeline.specs.js.map +1 -0
  168. package/dist/src/testing/kanban-todo.e2e.specs.d.ts +2 -0
  169. package/dist/src/testing/kanban-todo.e2e.specs.d.ts.map +1 -0
  170. package/dist/src/testing/kanban-todo.e2e.specs.js +160 -0
  171. package/dist/src/testing/kanban-todo.e2e.specs.js.map +1 -0
  172. package/dist/src/testing/mock-handlers.d.ts +21 -0
  173. package/dist/src/testing/mock-handlers.d.ts.map +1 -0
  174. package/dist/src/testing/mock-handlers.js +34 -0
  175. package/dist/src/testing/mock-handlers.js.map +1 -0
  176. package/dist/src/testing/mock-handlers.specs.d.ts +2 -0
  177. package/dist/src/testing/mock-handlers.specs.d.ts.map +1 -0
  178. package/dist/src/testing/mock-handlers.specs.js +193 -0
  179. package/dist/src/testing/mock-handlers.specs.js.map +1 -0
  180. package/dist/src/testing/real-execution.e2e.specs.d.ts +2 -0
  181. package/dist/src/testing/real-execution.e2e.specs.d.ts.map +1 -0
  182. package/dist/src/testing/real-execution.e2e.specs.js +140 -0
  183. package/dist/src/testing/real-execution.e2e.specs.js.map +1 -0
  184. package/dist/src/testing/real-plugin.e2e.specs.d.ts +2 -0
  185. package/dist/src/testing/real-plugin.e2e.specs.d.ts.map +1 -0
  186. package/dist/src/testing/real-plugin.e2e.specs.js +65 -0
  187. package/dist/src/testing/real-plugin.e2e.specs.js.map +1 -0
  188. package/dist/src/testing/server-startup.e2e.specs.d.ts +2 -0
  189. package/dist/src/testing/server-startup.e2e.specs.d.ts.map +1 -0
  190. package/dist/src/testing/server-startup.e2e.specs.js +104 -0
  191. package/dist/src/testing/server-startup.e2e.specs.js.map +1 -0
  192. package/dist/src/testing/snapshot-compare.d.ts +18 -0
  193. package/dist/src/testing/snapshot-compare.d.ts.map +1 -0
  194. package/dist/src/testing/snapshot-compare.js +86 -0
  195. package/dist/src/testing/snapshot-compare.js.map +1 -0
  196. package/dist/src/testing/snapshot-compare.specs.d.ts +2 -0
  197. package/dist/src/testing/snapshot-compare.specs.d.ts.map +1 -0
  198. package/dist/src/testing/snapshot-compare.specs.js +112 -0
  199. package/dist/src/testing/snapshot-compare.specs.js.map +1 -0
  200. package/dist/src/testing/snapshot-sanitize.d.ts +8 -0
  201. package/dist/src/testing/snapshot-sanitize.d.ts.map +1 -0
  202. package/dist/src/testing/snapshot-sanitize.js +10 -0
  203. package/dist/src/testing/snapshot-sanitize.js.map +1 -0
  204. package/dist/src/testing/snapshot-sanitize.specs.d.ts +2 -0
  205. package/dist/src/testing/snapshot-sanitize.specs.d.ts.map +1 -0
  206. package/dist/src/testing/snapshot-sanitize.specs.js +104 -0
  207. package/dist/src/testing/snapshot-sanitize.specs.js.map +1 -0
  208. package/dist/tsconfig.tsbuildinfo +1 -0
  209. package/docs/testing-analysis.md +395 -0
  210. package/package.json +31 -0
  211. package/pipeline-api-new.md +1078 -0
  212. package/pomodoro-plan.md +651 -0
  213. package/scripts/run-kanban-e2e.ts +219 -0
  214. package/scripts/start-server.ts +64 -0
  215. package/snapshots/e2e-run-2025-12-22T15-52-03.json +613 -0
  216. package/snapshots/e2e-run-2025-12-22T16-51-30.json +699 -0
  217. package/src/builder/define.specs.ts +531 -0
  218. package/src/builder/define.ts +700 -0
  219. package/src/config/pipeline-config.ts +32 -0
  220. package/src/core/descriptors.specs.ts +28 -0
  221. package/src/core/descriptors.ts +99 -0
  222. package/src/core/types.specs.ts +44 -0
  223. package/src/core/types.ts +16 -0
  224. package/src/graph/types.specs.ts +176 -0
  225. package/src/graph/types.ts +19 -0
  226. package/src/index.ts +54 -0
  227. package/src/logging/event-logger.specs.ts +100 -0
  228. package/src/logging/event-logger.ts +50 -0
  229. package/src/plugins/handler-adapter.specs.ts +164 -0
  230. package/src/plugins/handler-adapter.ts +21 -0
  231. package/src/plugins/plugin-loader.specs.ts +295 -0
  232. package/src/plugins/plugin-loader.ts +202 -0
  233. package/src/runtime/await-tracker.specs.ts +52 -0
  234. package/src/runtime/await-tracker.ts +50 -0
  235. package/src/runtime/context.specs.ts +28 -0
  236. package/src/runtime/context.ts +13 -0
  237. package/src/runtime/event-command-map.specs.ts +136 -0
  238. package/src/runtime/event-command-map.ts +38 -0
  239. package/src/runtime/phased-executor.specs.ts +358 -0
  240. package/src/runtime/phased-executor.ts +224 -0
  241. package/src/runtime/pipeline-runtime.specs.ts +214 -0
  242. package/src/runtime/pipeline-runtime.ts +119 -0
  243. package/src/runtime/settled-tracker.specs.ts +448 -0
  244. package/src/runtime/settled-tracker.ts +237 -0
  245. package/src/server/full-orchestration.e2e.specs.ts +672 -0
  246. package/src/server/pipeline-server.e2e.specs.ts +505 -0
  247. package/src/server/pipeline-server.specs.ts +761 -0
  248. package/src/server/pipeline-server.ts +656 -0
  249. package/src/server/sse-manager.specs.ts +208 -0
  250. package/src/server/sse-manager.ts +79 -0
  251. package/src/testing/event-capture.specs.ts +143 -0
  252. package/src/testing/event-capture.ts +65 -0
  253. package/src/testing/fixtures/kanban-full.pipeline.specs.ts +337 -0
  254. package/src/testing/fixtures/kanban-full.pipeline.ts +225 -0
  255. package/src/testing/fixtures/kanban-todo.config.ts +19 -0
  256. package/src/testing/fixtures/kanban.pipeline.specs.ts +33 -0
  257. package/src/testing/fixtures/kanban.pipeline.ts +124 -0
  258. package/src/testing/kanban-todo.e2e.specs.ts +209 -0
  259. package/src/testing/mock-handlers.specs.ts +229 -0
  260. package/src/testing/mock-handlers.ts +58 -0
  261. package/src/testing/real-execution.e2e.specs.ts +193 -0
  262. package/src/testing/real-plugin.e2e.specs.ts +94 -0
  263. package/src/testing/server-startup.e2e.specs.ts +162 -0
  264. package/src/testing/snapshot-compare.specs.ts +136 -0
  265. package/src/testing/snapshot-compare.ts +106 -0
  266. package/src/testing/snapshot-sanitize.specs.ts +131 -0
  267. package/src/testing/snapshot-sanitize.ts +17 -0
  268. package/tsconfig.json +11 -0
  269. package/tsconfig.test.json +9 -0
  270. package/vitest.config.ts +29 -0
@@ -0,0 +1,208 @@
1
+ import type { Event } from '@auto-engineer/message-bus';
2
+ import type { Response } from 'express';
3
+ import { beforeEach, describe, expect, it, type Mock, vi } from 'vitest';
4
+ import { SSEManager } from './sse-manager';
5
+
6
+ interface MockResponse extends Response {
7
+ written: string[];
8
+ ended: boolean;
9
+ writeHeadMock: Mock;
10
+ endMock: Mock;
11
+ triggerClose: () => void;
12
+ }
13
+
14
+ function createMockResponse(): MockResponse {
15
+ const written: string[] = [];
16
+ let ended = false;
17
+ const listeners: Record<string, Array<() => void>> = {};
18
+
19
+ const writeHeadMock = vi.fn();
20
+ const endMock = vi.fn(() => {
21
+ ended = true;
22
+ });
23
+
24
+ return {
25
+ written,
26
+ ended,
27
+ writeHeadMock,
28
+ endMock,
29
+ writeHead: writeHeadMock,
30
+ write: vi.fn((data: string) => {
31
+ written.push(data);
32
+ return true;
33
+ }),
34
+ end: endMock,
35
+ on: vi.fn((event: string, handler: () => void) => {
36
+ listeners[event] = listeners[event] ?? [];
37
+ listeners[event].push(handler);
38
+ }),
39
+ triggerClose: () => {
40
+ listeners.close?.forEach((h) => h());
41
+ },
42
+ } as unknown as MockResponse;
43
+ }
44
+
45
+ describe('SSEManager', () => {
46
+ let manager: SSEManager;
47
+
48
+ beforeEach(() => {
49
+ manager = new SSEManager();
50
+ });
51
+
52
+ describe('client management', () => {
53
+ it('should add client', () => {
54
+ const res = createMockResponse();
55
+ manager.addClient('c1', res);
56
+ expect(manager.clientCount).toBe(1);
57
+ });
58
+
59
+ it('should set correct SSE headers', () => {
60
+ const res = createMockResponse();
61
+ manager.addClient('c1', res);
62
+ expect(res.writeHeadMock).toHaveBeenCalledWith(200, {
63
+ 'Content-Type': 'text/event-stream',
64
+ 'Cache-Control': 'no-cache',
65
+ Connection: 'keep-alive',
66
+ 'Access-Control-Allow-Origin': '*',
67
+ });
68
+ });
69
+
70
+ it('should send heartbeat comment on connect', () => {
71
+ const res = createMockResponse();
72
+ manager.addClient('c1', res);
73
+ expect(res.written[0]).toBe(':\n\n');
74
+ });
75
+
76
+ it('should remove client', () => {
77
+ const res = createMockResponse();
78
+ manager.addClient('c1', res);
79
+ manager.removeClient('c1');
80
+ expect(manager.clientCount).toBe(0);
81
+ });
82
+
83
+ it('should cleanup client on response close', () => {
84
+ const res = createMockResponse();
85
+ manager.addClient('c1', res);
86
+ res.triggerClose();
87
+ expect(manager.clientCount).toBe(0);
88
+ });
89
+ });
90
+
91
+ describe('broadcasting', () => {
92
+ it('should broadcast event to all clients', () => {
93
+ const res1 = createMockResponse();
94
+ const res2 = createMockResponse();
95
+
96
+ manager.addClient('c1', res1);
97
+ manager.addClient('c2', res2);
98
+
99
+ const event: Event = { type: 'TestEvent', data: { foo: 'bar' } };
100
+ manager.broadcast(event);
101
+
102
+ expect(res1.written).toContainEqual(`data: ${JSON.stringify(event)}\n\n`);
103
+ expect(res2.written).toContainEqual(`data: ${JSON.stringify(event)}\n\n`);
104
+ });
105
+
106
+ it('should filter by correlationId when set', () => {
107
+ const res1 = createMockResponse();
108
+ const res2 = createMockResponse();
109
+
110
+ manager.addClient('c1', res1, 'workflow-123');
111
+ manager.addClient('c2', res2);
112
+
113
+ const event: Event = { type: 'TestEvent', correlationId: 'workflow-456', data: {} };
114
+ manager.broadcast(event);
115
+
116
+ expect(res1.written.length).toBe(1);
117
+ expect(res2.written.length).toBe(2);
118
+ });
119
+
120
+ it('should send to filtered client when correlationId matches', () => {
121
+ const res = createMockResponse();
122
+ manager.addClient('c1', res, 'workflow-123');
123
+
124
+ const event: Event = { type: 'TestEvent', correlationId: 'workflow-123', data: {} };
125
+ manager.broadcast(event);
126
+
127
+ expect(res.written).toContainEqual(`data: ${JSON.stringify(event)}\n\n`);
128
+ });
129
+
130
+ it('should not send to filtered client when correlationId does not match', () => {
131
+ const res = createMockResponse();
132
+ manager.addClient('c1', res, 'workflow-123');
133
+
134
+ const event: Event = { type: 'TestEvent', correlationId: 'workflow-456', data: {} };
135
+ manager.broadcast(event);
136
+
137
+ expect(res.written.length).toBe(1);
138
+ });
139
+ });
140
+
141
+ describe('broadcast error handling', () => {
142
+ it('should not throw when client.response.write fails', () => {
143
+ const res = createMockResponse();
144
+ manager.addClient('c1', res);
145
+
146
+ vi.mocked(res.write).mockImplementation(() => {
147
+ throw new Error('Connection reset');
148
+ });
149
+
150
+ const event: Event = { type: 'TestEvent', data: {} };
151
+
152
+ expect(() => manager.broadcast(event)).not.toThrow();
153
+ });
154
+
155
+ it('should remove failed client from clients map', () => {
156
+ const res = createMockResponse();
157
+ vi.mocked(res.write)
158
+ .mockImplementationOnce(() => true)
159
+ .mockImplementation(() => {
160
+ throw new Error('Connection reset');
161
+ });
162
+
163
+ manager.addClient('c1', res);
164
+ expect(manager.clientCount).toBe(1);
165
+
166
+ const event: Event = { type: 'TestEvent', data: {} };
167
+ manager.broadcast(event);
168
+
169
+ expect(manager.clientCount).toBe(0);
170
+ });
171
+
172
+ it('should continue broadcasting to other clients after one fails', () => {
173
+ const failingRes = createMockResponse();
174
+ const workingRes = createMockResponse();
175
+
176
+ vi.mocked(failingRes.write)
177
+ .mockImplementationOnce(() => true)
178
+ .mockImplementation(() => {
179
+ throw new Error('Connection reset');
180
+ });
181
+
182
+ manager.addClient('c1', failingRes);
183
+ manager.addClient('c2', workingRes);
184
+
185
+ const event: Event = { type: 'TestEvent', data: {} };
186
+ manager.broadcast(event);
187
+
188
+ expect(workingRes.written).toContainEqual(`data: ${JSON.stringify(event)}\n\n`);
189
+ expect(manager.clientCount).toBe(1);
190
+ });
191
+ });
192
+
193
+ describe('closeAll', () => {
194
+ it('should close all clients', () => {
195
+ const res1 = createMockResponse();
196
+ const res2 = createMockResponse();
197
+
198
+ manager.addClient('c1', res1);
199
+ manager.addClient('c2', res2);
200
+
201
+ manager.closeAll();
202
+
203
+ expect(manager.clientCount).toBe(0);
204
+ expect(res1.endMock).toHaveBeenCalled();
205
+ expect(res2.endMock).toHaveBeenCalled();
206
+ });
207
+ });
208
+ });
@@ -0,0 +1,79 @@
1
+ import type { Event } from '@auto-engineer/message-bus';
2
+ import type { Response } from 'express';
3
+
4
+ interface SSEClient {
5
+ id: string;
6
+ response: Response;
7
+ correlationIdFilter?: string;
8
+ }
9
+
10
+ export class SSEManager {
11
+ private clients = new Map<string, SSEClient>();
12
+
13
+ get clientCount(): number {
14
+ return this.clients.size;
15
+ }
16
+
17
+ addClient(id: string, response: Response, correlationIdFilter?: string): void {
18
+ response.writeHead(200, {
19
+ 'Content-Type': 'text/event-stream',
20
+ 'Cache-Control': 'no-cache',
21
+ Connection: 'keep-alive',
22
+ 'Access-Control-Allow-Origin': '*',
23
+ });
24
+
25
+ response.write(':\n\n');
26
+
27
+ this.clients.set(id, {
28
+ id,
29
+ response,
30
+ correlationIdFilter,
31
+ });
32
+
33
+ response.on('close', () => {
34
+ this.removeClient(id);
35
+ });
36
+ }
37
+
38
+ removeClient(id: string): void {
39
+ const client = this.clients.get(id);
40
+ if (client !== undefined) {
41
+ client.response.end();
42
+ this.clients.delete(id);
43
+ }
44
+ }
45
+
46
+ broadcast(event: Event): void {
47
+ const data = JSON.stringify(event);
48
+ const message = `data: ${data}\n\n`;
49
+ const failedClientIds: string[] = [];
50
+
51
+ for (const client of this.clients.values()) {
52
+ if (this.shouldSendToClient(client, event)) {
53
+ try {
54
+ client.response.write(message);
55
+ } catch {
56
+ failedClientIds.push(client.id);
57
+ }
58
+ }
59
+ }
60
+
61
+ for (const id of failedClientIds) {
62
+ this.removeClient(id);
63
+ }
64
+ }
65
+
66
+ private shouldSendToClient(client: SSEClient, event: Event): boolean {
67
+ if (client.correlationIdFilter === undefined) {
68
+ return true;
69
+ }
70
+ return event.correlationId === client.correlationIdFilter;
71
+ }
72
+
73
+ closeAll(): void {
74
+ for (const client of this.clients.values()) {
75
+ client.response.end();
76
+ }
77
+ this.clients.clear();
78
+ }
79
+ }
@@ -0,0 +1,143 @@
1
+ import type { Event } from '@auto-engineer/message-bus';
2
+ import { beforeEach, describe, expect, it } from 'vitest';
3
+ import { EventCapture } from './event-capture';
4
+
5
+ describe('EventCapture', () => {
6
+ let capture: EventCapture;
7
+
8
+ beforeEach(() => {
9
+ capture = new EventCapture();
10
+ });
11
+
12
+ describe('record', () => {
13
+ it('should record a single event', () => {
14
+ const event: Event = { type: 'TestEvent', data: { foo: 'bar' } };
15
+ capture.record(event);
16
+ expect(capture.getEvents()).toHaveLength(1);
17
+ expect(capture.getEvents()[0]).toEqual(event);
18
+ });
19
+
20
+ it('should record multiple events in order', () => {
21
+ const event1: Event = { type: 'First', data: {} };
22
+ const event2: Event = { type: 'Second', data: {} };
23
+ const event3: Event = { type: 'Third', data: {} };
24
+
25
+ capture.record(event1);
26
+ capture.record(event2);
27
+ capture.record(event3);
28
+
29
+ const events = capture.getEvents();
30
+ expect(events).toHaveLength(3);
31
+ expect(events[0].type).toBe('First');
32
+ expect(events[1].type).toBe('Second');
33
+ expect(events[2].type).toBe('Third');
34
+ });
35
+ });
36
+
37
+ describe('getEvents', () => {
38
+ it('should return empty array initially', () => {
39
+ expect(capture.getEvents()).toEqual([]);
40
+ });
41
+
42
+ it('should return a copy of events array', () => {
43
+ const event: Event = { type: 'TestEvent', data: {} };
44
+ capture.record(event);
45
+
46
+ const events = capture.getEvents();
47
+ events.push({ type: 'Modified', data: {} });
48
+
49
+ expect(capture.getEvents()).toHaveLength(1);
50
+ });
51
+ });
52
+
53
+ describe('getEventTypes', () => {
54
+ it('should return empty array initially', () => {
55
+ expect(capture.getEventTypes()).toEqual([]);
56
+ });
57
+
58
+ it('should return event types in order', () => {
59
+ capture.record({ type: 'A', data: {} });
60
+ capture.record({ type: 'B', data: {} });
61
+ capture.record({ type: 'C', data: {} });
62
+
63
+ expect(capture.getEventTypes()).toEqual(['A', 'B', 'C']);
64
+ });
65
+ });
66
+
67
+ describe('clear', () => {
68
+ it('should clear all recorded events', () => {
69
+ capture.record({ type: 'A', data: {} });
70
+ capture.record({ type: 'B', data: {} });
71
+
72
+ capture.clear();
73
+
74
+ expect(capture.getEvents()).toEqual([]);
75
+ });
76
+ });
77
+
78
+ describe('hasEvent', () => {
79
+ it('should return false when event not present', () => {
80
+ expect(capture.hasEvent('Missing')).toBe(false);
81
+ });
82
+
83
+ it('should return true when event is present', () => {
84
+ capture.record({ type: 'Present', data: {} });
85
+ expect(capture.hasEvent('Present')).toBe(true);
86
+ });
87
+ });
88
+
89
+ describe('getEventsOfType', () => {
90
+ it('should return empty array when no matching events', () => {
91
+ capture.record({ type: 'Other', data: {} });
92
+ expect(capture.getEventsOfType('Missing')).toEqual([]);
93
+ });
94
+
95
+ it('should return all events of specified type', () => {
96
+ capture.record({ type: 'A', data: { id: 1 } });
97
+ capture.record({ type: 'B', data: { id: 2 } });
98
+ capture.record({ type: 'A', data: { id: 3 } });
99
+
100
+ const aEvents = capture.getEventsOfType('A');
101
+ expect(aEvents).toHaveLength(2);
102
+ expect(aEvents[0].data).toEqual({ id: 1 });
103
+ expect(aEvents[1].data).toEqual({ id: 3 });
104
+ });
105
+ });
106
+
107
+ describe('waitForEvent', () => {
108
+ it('should resolve immediately if event already present', async () => {
109
+ capture.record({ type: 'Already', data: {} });
110
+
111
+ const event = await capture.waitForEvent('Already', 100);
112
+ expect(event.type).toBe('Already');
113
+ });
114
+
115
+ it('should resolve when event is recorded', async () => {
116
+ const promise = capture.waitForEvent('Delayed', 500);
117
+
118
+ setTimeout(() => {
119
+ capture.record({ type: 'Delayed', data: { success: true } });
120
+ }, 50);
121
+
122
+ const event = await promise;
123
+ expect(event.type).toBe('Delayed');
124
+ expect(event.data).toEqual({ success: true });
125
+ });
126
+
127
+ it('should reject on timeout if event not recorded', async () => {
128
+ await expect(capture.waitForEvent('Never', 50)).rejects.toThrow('Timeout waiting for event: Never');
129
+ });
130
+ });
131
+
132
+ describe('count', () => {
133
+ it('should return 0 initially', () => {
134
+ expect(capture.count).toBe(0);
135
+ });
136
+
137
+ it('should return number of recorded events', () => {
138
+ capture.record({ type: 'A', data: {} });
139
+ capture.record({ type: 'B', data: {} });
140
+ expect(capture.count).toBe(2);
141
+ });
142
+ });
143
+ });
@@ -0,0 +1,65 @@
1
+ import type { Event } from '@auto-engineer/message-bus';
2
+
3
+ export class EventCapture {
4
+ private events: Event[] = [];
5
+ private eventListeners: Array<(event: Event) => void> = [];
6
+
7
+ record(event: Event): void {
8
+ this.events.push(event);
9
+ this.eventListeners.forEach((listener) => listener(event));
10
+ }
11
+
12
+ getEvents(): Event[] {
13
+ return [...this.events];
14
+ }
15
+
16
+ getEventTypes(): string[] {
17
+ return this.events.map((e) => e.type);
18
+ }
19
+
20
+ clear(): void {
21
+ this.events = [];
22
+ }
23
+
24
+ hasEvent(eventType: string): boolean {
25
+ return this.events.some((e) => e.type === eventType);
26
+ }
27
+
28
+ getEventsOfType(eventType: string): Event[] {
29
+ return this.events.filter((e) => e.type === eventType);
30
+ }
31
+
32
+ waitForEvent(eventType: string, timeoutMs: number): Promise<Event> {
33
+ const existing = this.events.find((e) => e.type === eventType);
34
+ if (existing) {
35
+ return Promise.resolve(existing);
36
+ }
37
+
38
+ return new Promise((resolve, reject) => {
39
+ const timeoutId = setTimeout(() => {
40
+ const index = this.eventListeners.indexOf(listener);
41
+ if (index !== -1) {
42
+ this.eventListeners.splice(index, 1);
43
+ }
44
+ reject(new Error(`Timeout waiting for event: ${eventType}`));
45
+ }, timeoutMs);
46
+
47
+ const listener = (event: Event) => {
48
+ if (event.type === eventType) {
49
+ clearTimeout(timeoutId);
50
+ const index = this.eventListeners.indexOf(listener);
51
+ if (index !== -1) {
52
+ this.eventListeners.splice(index, 1);
53
+ }
54
+ resolve(event);
55
+ }
56
+ };
57
+
58
+ this.eventListeners.push(listener);
59
+ });
60
+ }
61
+
62
+ get count(): number {
63
+ return this.events.length;
64
+ }
65
+ }