@prmichaelsen/acp-visualizer 0.1.0 → 0.1.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 (159) hide show
  1. package/package.json +8 -10
  2. package/src/components/ExtraFieldsBadge.tsx +1 -1
  3. package/src/components/FilterBar.tsx +1 -1
  4. package/src/components/Header.tsx +1 -1
  5. package/src/components/MilestoneTable.tsx +1 -1
  6. package/src/components/MilestoneTree.tsx +2 -2
  7. package/src/components/StatusBadge.tsx +1 -1
  8. package/src/components/StatusDot.tsx +1 -1
  9. package/src/components/TaskList.tsx +1 -1
  10. package/src/routes/__root.tsx +5 -5
  11. package/src/routes/api/watch.ts +1 -1
  12. package/src/routes/index.tsx +2 -2
  13. package/src/routes/milestones.tsx +7 -7
  14. package/src/routes/search.tsx +4 -4
  15. package/src/routes/tasks.tsx +3 -3
  16. package/src/services/progress-database.service.ts +3 -3
  17. package/agent/commands/acp.clarification-address.md +0 -417
  18. package/agent/commands/acp.clarification-capture.md +0 -386
  19. package/agent/commands/acp.clarification-create.md +0 -437
  20. package/agent/commands/acp.clarifications-research.md +0 -326
  21. package/agent/commands/acp.command-create.md +0 -432
  22. package/agent/commands/acp.design-create.md +0 -286
  23. package/agent/commands/acp.design-reference.md +0 -355
  24. package/agent/commands/acp.handoff.md +0 -270
  25. package/agent/commands/acp.index.md +0 -423
  26. package/agent/commands/acp.init.md +0 -546
  27. package/agent/commands/acp.package-create.md +0 -895
  28. package/agent/commands/acp.package-info.md +0 -212
  29. package/agent/commands/acp.package-install.md +0 -539
  30. package/agent/commands/acp.package-list.md +0 -280
  31. package/agent/commands/acp.package-publish.md +0 -541
  32. package/agent/commands/acp.package-remove.md +0 -293
  33. package/agent/commands/acp.package-search.md +0 -307
  34. package/agent/commands/acp.package-update.md +0 -361
  35. package/agent/commands/acp.package-validate.md +0 -540
  36. package/agent/commands/acp.pattern-create.md +0 -386
  37. package/agent/commands/acp.plan.md +0 -587
  38. package/agent/commands/acp.proceed.md +0 -882
  39. package/agent/commands/acp.project-create.md +0 -675
  40. package/agent/commands/acp.project-info.md +0 -312
  41. package/agent/commands/acp.project-list.md +0 -226
  42. package/agent/commands/acp.project-remove.md +0 -379
  43. package/agent/commands/acp.project-set.md +0 -227
  44. package/agent/commands/acp.project-update.md +0 -307
  45. package/agent/commands/acp.projects-restore.md +0 -228
  46. package/agent/commands/acp.projects-sync.md +0 -347
  47. package/agent/commands/acp.report.md +0 -407
  48. package/agent/commands/acp.resume.md +0 -239
  49. package/agent/commands/acp.sessions.md +0 -301
  50. package/agent/commands/acp.status.md +0 -293
  51. package/agent/commands/acp.sync.md +0 -364
  52. package/agent/commands/acp.task-create.md +0 -500
  53. package/agent/commands/acp.update.md +0 -302
  54. package/agent/commands/acp.validate.md +0 -466
  55. package/agent/commands/acp.version-check-for-updates.md +0 -276
  56. package/agent/commands/acp.version-check.md +0 -191
  57. package/agent/commands/acp.version-update.md +0 -289
  58. package/agent/commands/command.template.md +0 -339
  59. package/agent/commands/git.commit.md +0 -526
  60. package/agent/commands/git.init.md +0 -514
  61. package/agent/commands/tanstack-cloudflare.deploy.md +0 -272
  62. package/agent/commands/tanstack-cloudflare.tail.md +0 -275
  63. package/agent/design/.gitkeep +0 -0
  64. package/agent/design/design.template.md +0 -154
  65. package/agent/design/local.dashboard-layout-routing.md +0 -288
  66. package/agent/design/local.data-model-yaml-parsing.md +0 -310
  67. package/agent/design/local.search-filtering.md +0 -331
  68. package/agent/design/local.server-api-auto-refresh.md +0 -235
  69. package/agent/design/local.table-tree-views.md +0 -299
  70. package/agent/design/local.visualizer-requirements.md +0 -349
  71. package/agent/design/requirements.template.md +0 -387
  72. package/agent/index/.gitkeep +0 -0
  73. package/agent/index/acp.core.yaml +0 -137
  74. package/agent/index/local.main.template.yaml +0 -37
  75. package/agent/manifest.template.yaml +0 -13
  76. package/agent/manifest.yaml +0 -302
  77. package/agent/milestones/.gitkeep +0 -0
  78. package/agent/milestones/milestone-1-project-scaffold-data-pipeline.md +0 -67
  79. package/agent/milestones/milestone-1-{title}.template.md +0 -206
  80. package/agent/milestones/milestone-2-dashboard-views-interaction.md +0 -79
  81. package/agent/package.template.yaml +0 -86
  82. package/agent/patterns/.gitkeep +0 -0
  83. package/agent/patterns/bootstrap.template.md +0 -1237
  84. package/agent/patterns/pattern.template.md +0 -382
  85. package/agent/patterns/tanstack-cloudflare.acl-permissions.md +0 -332
  86. package/agent/patterns/tanstack-cloudflare.action-bar-item.md +0 -416
  87. package/agent/patterns/tanstack-cloudflare.api-route-handlers.md +0 -401
  88. package/agent/patterns/tanstack-cloudflare.auth-session-management.md +0 -387
  89. package/agent/patterns/tanstack-cloudflare.card-and-list.md +0 -271
  90. package/agent/patterns/tanstack-cloudflare.chat-engine.md +0 -353
  91. package/agent/patterns/tanstack-cloudflare.confirmation-tokens.md +0 -346
  92. package/agent/patterns/tanstack-cloudflare.durable-objects-websocket.md +0 -516
  93. package/agent/patterns/tanstack-cloudflare.email-service.md +0 -431
  94. package/agent/patterns/tanstack-cloudflare.expander.md +0 -98
  95. package/agent/patterns/tanstack-cloudflare.fcm-push.md +0 -115
  96. package/agent/patterns/tanstack-cloudflare.firebase-anonymous-sessions.md +0 -441
  97. package/agent/patterns/tanstack-cloudflare.firebase-auth.md +0 -348
  98. package/agent/patterns/tanstack-cloudflare.firebase-firestore.md +0 -550
  99. package/agent/patterns/tanstack-cloudflare.firebase-storage.md +0 -369
  100. package/agent/patterns/tanstack-cloudflare.form-controls.md +0 -145
  101. package/agent/patterns/tanstack-cloudflare.global-search-context.md +0 -93
  102. package/agent/patterns/tanstack-cloudflare.image-carousel.md +0 -126
  103. package/agent/patterns/tanstack-cloudflare.library-services.md +0 -553
  104. package/agent/patterns/tanstack-cloudflare.lightbox.md +0 -169
  105. package/agent/patterns/tanstack-cloudflare.markdown-content.md +0 -115
  106. package/agent/patterns/tanstack-cloudflare.mention-suggestions.md +0 -98
  107. package/agent/patterns/tanstack-cloudflare.modal.md +0 -156
  108. package/agent/patterns/tanstack-cloudflare.nextjs-to-tanstack-routing.md +0 -461
  109. package/agent/patterns/tanstack-cloudflare.notifications-engine.md +0 -151
  110. package/agent/patterns/tanstack-cloudflare.oauth-token-refresh.md +0 -90
  111. package/agent/patterns/tanstack-cloudflare.og-metadata.md +0 -296
  112. package/agent/patterns/tanstack-cloudflare.pagination.md +0 -442
  113. package/agent/patterns/tanstack-cloudflare.pill-input.md +0 -220
  114. package/agent/patterns/tanstack-cloudflare.provider-adapter.md +0 -401
  115. package/agent/patterns/tanstack-cloudflare.rate-limiting.md +0 -323
  116. package/agent/patterns/tanstack-cloudflare.scheduled-tasks.md +0 -338
  117. package/agent/patterns/tanstack-cloudflare.searchable-settings.md +0 -375
  118. package/agent/patterns/tanstack-cloudflare.slide-over.md +0 -129
  119. package/agent/patterns/tanstack-cloudflare.ssr-preload.md +0 -571
  120. package/agent/patterns/tanstack-cloudflare.third-party-api-integration.md +0 -508
  121. package/agent/patterns/tanstack-cloudflare.toast-system.md +0 -142
  122. package/agent/patterns/tanstack-cloudflare.unified-header.md +0 -280
  123. package/agent/patterns/tanstack-cloudflare.user-scoped-collections.md +0 -628
  124. package/agent/patterns/tanstack-cloudflare.websocket-manager.md +0 -237
  125. package/agent/patterns/tanstack-cloudflare.wrangler-configuration.md +0 -358
  126. package/agent/patterns/tanstack-cloudflare.zod-schema-validation.md +0 -336
  127. package/agent/progress.template.yaml +0 -161
  128. package/agent/progress.yaml +0 -145
  129. package/agent/schemas/package.schema.yaml +0 -276
  130. package/agent/scripts/acp.common.sh +0 -1781
  131. package/agent/scripts/acp.install.sh +0 -333
  132. package/agent/scripts/acp.package-create.sh +0 -924
  133. package/agent/scripts/acp.package-info.sh +0 -288
  134. package/agent/scripts/acp.package-install.sh +0 -893
  135. package/agent/scripts/acp.package-list.sh +0 -311
  136. package/agent/scripts/acp.package-publish.sh +0 -420
  137. package/agent/scripts/acp.package-remove.sh +0 -348
  138. package/agent/scripts/acp.package-search.sh +0 -156
  139. package/agent/scripts/acp.package-update.sh +0 -517
  140. package/agent/scripts/acp.package-validate.sh +0 -1018
  141. package/agent/scripts/acp.uninstall.sh +0 -85
  142. package/agent/scripts/acp.version-check-for-updates.sh +0 -98
  143. package/agent/scripts/acp.version-check.sh +0 -47
  144. package/agent/scripts/acp.version-update.sh +0 -176
  145. package/agent/scripts/acp.yaml-parser.sh +0 -985
  146. package/agent/scripts/acp.yaml-validate.sh +0 -205
  147. package/agent/tasks/.gitkeep +0 -0
  148. package/agent/tasks/milestone-1-project-scaffold-data-pipeline/task-1-initialize-tanstack-start-project.md +0 -210
  149. package/agent/tasks/milestone-1-project-scaffold-data-pipeline/task-2-implement-data-model-yaml-parser.md +0 -294
  150. package/agent/tasks/milestone-1-project-scaffold-data-pipeline/task-3-build-server-api-data-loading.md +0 -193
  151. package/agent/tasks/milestone-1-project-scaffold-data-pipeline/task-4-add-auto-refresh-sse.md +0 -262
  152. package/agent/tasks/milestone-2-dashboard-views-interaction/task-10-polish-integration-testing.md +0 -156
  153. package/agent/tasks/milestone-2-dashboard-views-interaction/task-5-build-dashboard-layout-routing.md +0 -178
  154. package/agent/tasks/milestone-2-dashboard-views-interaction/task-6-build-overview-page.md +0 -141
  155. package/agent/tasks/milestone-2-dashboard-views-interaction/task-7-implement-milestone-table-view.md +0 -153
  156. package/agent/tasks/milestone-2-dashboard-views-interaction/task-8-implement-milestone-tree-view.md +0 -174
  157. package/agent/tasks/milestone-2-dashboard-views-interaction/task-9-implement-search-filtering.md +0 -233
  158. package/agent/tasks/task-1-{title}.template.md +0 -244
  159. package/vitest.config.ts +0 -27
@@ -1,516 +0,0 @@
1
- # Durable Objects WebSocket Pattern
2
-
3
- **Category**: Architecture
4
- **Applicable To**: TanStack Start + Cloudflare Workers applications requiring real-time communication
5
- **Status**: Stable
6
-
7
- ---
8
-
9
- ## Overview
10
-
11
- Cloudflare Durable Objects provide stateful, long-lived server instances that are ideal for WebSocket-based real-time features. This pattern demonstrates how to build a Durable Object that accepts WebSocket connections, processes typed messages, and delegates business logic to an injected engine using dependency injection.
12
-
13
- The Durable Object acts as a thin coordination layer — it manages WebSocket lifecycle (accept, message routing, close) while delegating all domain logic to portable engine classes. This separation enables the business logic to be tested independently and potentially extracted into reusable packages.
14
-
15
- ---
16
-
17
- ## When to Use This Pattern
18
-
19
- ✅ **Use this pattern when:**
20
- - Building real-time features (chat, collaboration, live updates)
21
- - Need persistent server-side state across requests
22
- - Want WebSocket connections managed at the edge
23
- - Need to coordinate between multiple connected clients
24
- - Building features that require streaming responses (AI chat, file upload progress)
25
-
26
- ❌ **Don't use this pattern when:**
27
- - Simple request/response patterns suffice (use API routes instead)
28
- - No real-time requirement (polling or SSR preloading is simpler)
29
- - Stateless operations (use regular Workers)
30
-
31
- ---
32
-
33
- ## Core Principles
34
-
35
- 1. **Thin Wrapper**: The Durable Object is a thin coordination layer — it manages WebSocket lifecycle and delegates business logic to injected engines
36
- 2. **Dependency Injection**: All providers (AI, storage, etc.) are injected into the engine at construction time
37
- 3. **Typed Messages**: Both client and server messages use TypeScript interfaces with discriminated `type` fields
38
- 4. **Persistent Provider Instances**: Provider instances are created once in the constructor and reused across messages for caching benefits
39
- 5. **Graceful Error Handling**: Errors are caught and sent back to the client as error messages, never crashing the Durable Object
40
- 6. **WebSocket Pair Pattern**: Use Cloudflare's `WebSocketPair` API for server-accepted WebSocket connections
41
-
42
- ---
43
-
44
- ## Implementation
45
-
46
- ### Structure
47
-
48
- ```
49
- src/
50
- ├── durable-objects/
51
- │ ├── ChatRoom.ts # Durable Object (thin wrapper)
52
- │ └── UploadManager.ts # Another Durable Object example
53
- ├── lib/
54
- │ ├── chat/
55
- │ │ ├── chat-engine.ts # Core business logic (portable)
56
- │ │ └── interfaces/ # Provider interfaces
57
- │ └── chat-providers/
58
- │ ├── ai-provider.ts # AI provider implementation
59
- │ └── storage-provider.ts # Storage provider implementation
60
- └── routes/
61
- └── api/
62
- └── chat-ws.tsx # WebSocket upgrade endpoint
63
- ```
64
-
65
- ### Code Example
66
-
67
- #### Step 1: Define Typed Messages
68
-
69
- ```typescript
70
- // src/durable-objects/types.ts
71
-
72
- /** Messages sent from client → server */
73
- interface ClientMessage {
74
- type: 'message' | 'load_messages' | 'init'
75
- userId: string
76
- conversationId?: string
77
- message?: MessageContent
78
- limit?: number
79
- startAfter?: string
80
- }
81
-
82
- /** Messages sent from server → client */
83
- interface ServerMessage {
84
- type: 'chunk' | 'complete' | 'error' | 'message' | 'messages_loaded' | 'ready'
85
- content?: string
86
- error?: string
87
- message?: any
88
- messages?: any[]
89
- hasMore?: boolean
90
- }
91
- ```
92
-
93
- #### Step 2: Create the Durable Object
94
-
95
- ```typescript
96
- // src/durable-objects/ChatRoom.ts
97
- import { DurableObject } from 'cloudflare:workers'
98
- import { ChatEngine } from '@/lib/chat'
99
- import { BedrockAIProvider, FirebaseStorageProvider } from '@/lib/chat-providers'
100
- import { chatLogger } from '@/lib/logger'
101
-
102
- export class ChatRoom extends DurableObject {
103
- private sessions: Set<WebSocket>
104
- private chatEngine: ChatEngine
105
-
106
- constructor(state: DurableObjectState, env: Env) {
107
- super(state, env)
108
- this.sessions = new Set()
109
-
110
- // Inject dependencies into engine (created once, reused across messages)
111
- this.chatEngine = new ChatEngine({
112
- aiProvider: new BedrockAIProvider(),
113
- storageProvider: new FirebaseStorageProvider(),
114
- logger: chatLogger
115
- })
116
- }
117
-
118
- async fetch(request: Request): Promise<Response> {
119
- // Verify WebSocket upgrade
120
- const upgradeHeader = request.headers.get('Upgrade')
121
- if (upgradeHeader !== 'websocket') {
122
- return new Response('Expected WebSocket', { status: 426 })
123
- }
124
-
125
- // Create WebSocket pair
126
- const pair = new WebSocketPair()
127
- const [client, server] = Object.values(pair)
128
-
129
- // Accept and track connection
130
- server.accept()
131
- this.sessions.add(server)
132
-
133
- // Handle messages
134
- server.addEventListener('message', async (event) => {
135
- try {
136
- const data = JSON.parse(event.data as string) as ClientMessage
137
-
138
- switch (data.type) {
139
- case 'init':
140
- this.sendMessage(server, { type: 'ready' })
141
- break
142
- case 'message':
143
- await this.handleMessage(data, server)
144
- break
145
- case 'load_messages':
146
- await this.handleLoadMessages(data, server)
147
- break
148
- default:
149
- chatLogger.warn('Unknown message type', { type: (data as any).type })
150
- }
151
- } catch (error) {
152
- this.sendMessage(server, {
153
- type: 'error',
154
- error: error instanceof Error ? error.message : 'Unknown error'
155
- })
156
- }
157
- })
158
-
159
- // Handle close
160
- server.addEventListener('close', () => {
161
- this.sessions.delete(server)
162
- })
163
-
164
- return new Response(null, { status: 101, webSocket: client })
165
- }
166
-
167
- private async handleMessage(data: ClientMessage, socket: WebSocket): Promise<void> {
168
- const { userId, conversationId = 'main', message } = data
169
-
170
- if (!message) {
171
- this.sendMessage(socket, { type: 'error', error: 'No message provided' })
172
- return
173
- }
174
-
175
- // Delegate to engine with streaming callback
176
- await this.chatEngine.processMessage({
177
- userId,
178
- conversationId,
179
- message,
180
- onMessage: (msg) => {
181
- this.sendMessage(socket, msg)
182
- }
183
- })
184
- }
185
-
186
- private async handleLoadMessages(data: ClientMessage, socket: WebSocket): Promise<void> {
187
- const { userId, conversationId = 'main', limit = 50, startAfter } = data
188
-
189
- try {
190
- const messages = await this.chatEngine.loadMessages({
191
- userId, conversationId, limit, startAfter
192
- })
193
-
194
- this.sendMessage(socket, {
195
- type: 'messages_loaded',
196
- messages,
197
- hasMore: messages.length === limit
198
- })
199
- } catch (error) {
200
- this.sendMessage(socket, {
201
- type: 'error',
202
- error: error instanceof Error ? error.message : 'Failed to load messages'
203
- })
204
- }
205
- }
206
-
207
- private sendMessage(socket: WebSocket, message: ServerMessage): void {
208
- try {
209
- socket.send(JSON.stringify(message))
210
- } catch (error) {
211
- console.error('Failed to send WebSocket message', error)
212
- }
213
- }
214
- }
215
- ```
216
-
217
- #### Step 3: Configure Wrangler
218
-
219
- ```toml
220
- # wrangler.toml
221
-
222
- # Durable Object bindings
223
- [[durable_objects.bindings]]
224
- name = "CHAT_ROOM"
225
- class_name = "ChatRoom"
226
-
227
- # Migrations
228
- [[migrations]]
229
- tag = "v1"
230
- new_sqlite_classes = ["ChatRoom"]
231
- ```
232
-
233
- #### Step 4: Create WebSocket Upgrade Route
234
-
235
- ```typescript
236
- // src/routes/api/chat-ws.tsx
237
- import { createFileRoute } from '@tanstack/react-router'
238
- import { getAuthSession } from '@/lib/auth/server-fn'
239
-
240
- export const Route = createFileRoute('/api/chat-ws')({
241
- server: {
242
- handlers: {
243
- GET: async ({ request, context }) => {
244
- const user = await getAuthSession()
245
- if (!user) {
246
- return new Response('Unauthorized', { status: 401 })
247
- }
248
-
249
- // Get Durable Object stub
250
- const env = context.cloudflare.env as Env
251
- const id = env.CHAT_ROOM.idFromName(user.uid)
252
- const stub = env.CHAT_ROOM.get(id)
253
-
254
- // Forward WebSocket upgrade to Durable Object
255
- return stub.fetch(request)
256
- },
257
- },
258
- },
259
- })
260
- ```
261
-
262
- #### Step 5: Client-Side WebSocket Hook
263
-
264
- ```typescript
265
- // src/hooks/use-websocket.ts
266
- import { useState, useEffect, useCallback, useRef } from 'react'
267
-
268
- interface UseWebSocketOptions {
269
- userId: string
270
- onMessage: (message: ServerMessage) => void
271
- onConnectionChange?: (connected: boolean) => void
272
- }
273
-
274
- export function useWebSocket({ userId, onMessage, onConnectionChange }: UseWebSocketOptions) {
275
- const wsRef = useRef<WebSocket | null>(null)
276
- const [connected, setConnected] = useState(false)
277
-
278
- useEffect(() => {
279
- const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:'
280
- const ws = new WebSocket(`${protocol}//${window.location.host}/api/chat-ws`)
281
- wsRef.current = ws
282
-
283
- ws.onopen = () => {
284
- setConnected(true)
285
- onConnectionChange?.(true)
286
- ws.send(JSON.stringify({ type: 'init', userId }))
287
- }
288
-
289
- ws.onmessage = (event) => {
290
- const data = JSON.parse(event.data)
291
- onMessage(data)
292
- }
293
-
294
- ws.onclose = () => {
295
- setConnected(false)
296
- onConnectionChange?.(false)
297
- }
298
-
299
- return () => ws.close()
300
- }, [userId])
301
-
302
- const sendMessage = useCallback((message: ClientMessage) => {
303
- if (wsRef.current?.readyState === WebSocket.OPEN) {
304
- wsRef.current.send(JSON.stringify(message))
305
- }
306
- }, [])
307
-
308
- return { connected, sendMessage }
309
- }
310
- ```
311
-
312
- ---
313
-
314
- ## Examples
315
-
316
- ### Example 1: File Upload Durable Object
317
-
318
- ```typescript
319
- // src/durable-objects/UploadManager.ts
320
- import { DurableObject } from 'cloudflare:workers'
321
-
322
- export class UploadManager extends DurableObject {
323
- async fetch(request: Request): Promise<Response> {
324
- if (request.headers.get('Upgrade') !== 'websocket') {
325
- return new Response('Expected WebSocket', { status: 426 })
326
- }
327
-
328
- const pair = new WebSocketPair()
329
- const [client, server] = Object.values(pair)
330
- server.accept()
331
-
332
- server.addEventListener('message', async (event) => {
333
- try {
334
- const data = JSON.parse(event.data as string)
335
-
336
- switch (data.type) {
337
- case 'upload_chunk':
338
- // Process chunk, report progress
339
- server.send(JSON.stringify({
340
- type: 'progress',
341
- progress: data.chunkIndex / data.totalChunks * 100
342
- }))
343
- break
344
- case 'upload_complete':
345
- // Finalize upload
346
- server.send(JSON.stringify({ type: 'complete', url: signedUrl }))
347
- break
348
- }
349
- } catch (error) {
350
- server.send(JSON.stringify({ type: 'error', error: String(error) }))
351
- }
352
- })
353
-
354
- return new Response(null, { status: 101, webSocket: client })
355
- }
356
- }
357
- ```
358
-
359
- ### Example 2: Internal HTTP Endpoint on Durable Object
360
-
361
- ```typescript
362
- // Durable Objects can also handle non-WebSocket HTTP requests
363
- export class ChatRoom extends DurableObject {
364
- async fetch(request: Request): Promise<Response> {
365
- const url = new URL(request.url)
366
-
367
- // Internal cache invalidation endpoint
368
- if (url.pathname === '/invalidate-cache' && request.method === 'POST') {
369
- const body = await request.json()
370
- this.mcpProvider.clearCache()
371
- return Response.json({ success: true })
372
- }
373
-
374
- // WebSocket upgrade (normal flow)
375
- if (request.headers.get('Upgrade') === 'websocket') {
376
- return this.handleWebSocket(request)
377
- }
378
-
379
- return new Response('Expected WebSocket or known endpoint', { status: 426 })
380
- }
381
- }
382
- ```
383
-
384
- ---
385
-
386
- ## Benefits
387
-
388
- ### 1. Persistent State Across Messages
389
- Durable Objects maintain state between requests, enabling caching of provider instances and connection state.
390
-
391
- ### 2. Edge-Based Real-Time
392
- WebSocket connections run at Cloudflare's edge, providing low-latency real-time communication globally.
393
-
394
- ### 3. Testable Business Logic
395
- Business logic lives in injected engines, not in the Durable Object itself, making it independently testable.
396
-
397
- ### 4. Automatic Scaling
398
- Each Durable Object instance is uniquely addressable (by user ID, room ID, etc.) and scales automatically.
399
-
400
- ---
401
-
402
- ## Trade-offs
403
-
404
- ### 1. Durable Object Limitations
405
- **Downside**: Each Durable Object runs in a single location and has CPU/memory limits.
406
- **Mitigation**: Keep Durable Objects thin. Delegate heavy work to external services.
407
-
408
- ### 2. Cold Start Latency
409
- **Downside**: First request to a Durable Object may have cold start delay.
410
- **Mitigation**: Use `init` messages to pre-warm connections after WebSocket establishes.
411
-
412
- ---
413
-
414
- ## Anti-Patterns
415
-
416
- ### ❌ Anti-Pattern 1: Business Logic in Durable Object
417
-
418
- ```typescript
419
- // ❌ BAD: Business logic mixed into Durable Object
420
- export class ChatRoom extends DurableObject {
421
- async handleMessage(data, socket) {
422
- const response = await fetch('https://api.anthropic.com/...') // Direct API call
423
- const messages = await getDocument(...) // Direct DB call
424
- // ... 200 lines of business logic
425
- }
426
- }
427
-
428
- // ✅ GOOD: Delegate to engine
429
- export class ChatRoom extends DurableObject {
430
- async handleMessage(data, socket) {
431
- await this.chatEngine.processMessage({
432
- ...data,
433
- onMessage: (msg) => this.sendMessage(socket, msg)
434
- })
435
- }
436
- }
437
- ```
438
-
439
- ### ❌ Anti-Pattern 2: Creating Providers Per Message
440
-
441
- ```typescript
442
- // ❌ BAD: New provider instances per message (loses caching)
443
- server.addEventListener('message', async (event) => {
444
- const aiProvider = new BedrockAIProvider() // Created every message!
445
- const engine = new ChatEngine({ aiProvider })
446
- })
447
-
448
- // ✅ GOOD: Create once in constructor
449
- constructor(state, env) {
450
- this.chatEngine = new ChatEngine({
451
- aiProvider: new BedrockAIProvider() // Created once, reused
452
- })
453
- }
454
- ```
455
-
456
- ---
457
-
458
- ## Testing Strategy
459
-
460
- ### Unit Testing Engine (Without Durable Object)
461
-
462
- ```typescript
463
- describe('ChatEngine', () => {
464
- it('should process messages', async () => {
465
- const mockAI = { streamChat: jest.fn() }
466
- const mockStorage = { saveMessage: jest.fn(), loadMessages: jest.fn() }
467
-
468
- const engine = new ChatEngine({
469
- aiProvider: mockAI,
470
- storageProvider: mockStorage
471
- })
472
-
473
- const messages: any[] = []
474
- await engine.processMessage({
475
- userId: 'user1',
476
- conversationId: 'conv1',
477
- message: 'Hello',
478
- onMessage: (msg) => messages.push(msg)
479
- })
480
-
481
- expect(mockStorage.saveMessage).toHaveBeenCalled()
482
- })
483
- })
484
- ```
485
-
486
- ---
487
-
488
- ## Related Patterns
489
-
490
- - **[Provider Adapter Pattern](./tanstack-cloudflare.provider-adapter.md)**: Interfaces for dependency injection into engines
491
- - **[Rate Limiting Pattern](./tanstack-cloudflare.rate-limiting.md)**: Rate limit WebSocket connections
492
- - **[Wrangler Configuration](./tanstack-cloudflare.wrangler-configuration.md)**: Durable Object bindings and migrations
493
-
494
- ---
495
-
496
- ## Checklist for Implementation
497
-
498
- - [ ] Durable Object extends `DurableObject` from `cloudflare:workers`
499
- - [ ] Business logic delegated to injected engine
500
- - [ ] Provider instances created once in constructor
501
- - [ ] Typed `ClientMessage` and `ServerMessage` interfaces
502
- - [ ] `switch` statement on `data.type` for message routing
503
- - [ ] Error handling wraps every message handler
504
- - [ ] Errors sent to client as `{ type: 'error', error: string }`
505
- - [ ] WebSocket pair created with `new WebSocketPair()`
506
- - [ ] Server socket accepted with `server.accept()`
507
- - [ ] Sessions tracked in `Set<WebSocket>` and cleaned up on close
508
- - [ ] Wrangler configured with DO bindings and migrations
509
- - [ ] WebSocket upgrade route validates auth before forwarding
510
-
511
- ---
512
-
513
- **Status**: Stable - Proven pattern for real-time features on Cloudflare Workers
514
- **Recommendation**: Use for all real-time features requiring persistent server-side state
515
- **Last Updated**: 2026-02-28
516
- **Contributors**: Patrick Michaelsen