@pixelbyte-software/pixcode 1.33.11 → 1.35.0

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 (164) hide show
  1. package/dist/api-docs.html +162 -9
  2. package/dist/assets/index-B8w57E1r.css +32 -0
  3. package/dist/assets/index-Djuh0wHV.js +854 -0
  4. package/dist/favicon.svg +8 -8
  5. package/dist/icons/icon-128x128.svg +9 -9
  6. package/dist/icons/icon-144x144.svg +9 -9
  7. package/dist/icons/icon-152x152.svg +9 -9
  8. package/dist/icons/icon-192x192.svg +9 -9
  9. package/dist/icons/icon-384x384.svg +9 -9
  10. package/dist/icons/icon-512x512.svg +9 -9
  11. package/dist/icons/icon-72x72.svg +9 -9
  12. package/dist/icons/icon-96x96.svg +9 -9
  13. package/dist/icons/icon-template.svg +9 -9
  14. package/dist/index.html +2 -2
  15. package/dist/logo.svg +12 -12
  16. package/dist/openapi.yaml +383 -1
  17. package/dist-server/server/claude-sdk.js +38 -7
  18. package/dist-server/server/claude-sdk.js.map +1 -1
  19. package/dist-server/server/cli.js +12 -17
  20. package/dist-server/server/cli.js.map +1 -1
  21. package/dist-server/server/daemon-manager.js +98 -51
  22. package/dist-server/server/daemon-manager.js.map +1 -1
  23. package/dist-server/server/database/json-store.js +8 -5
  24. package/dist-server/server/database/json-store.js.map +1 -1
  25. package/dist-server/server/index.js +34 -9
  26. package/dist-server/server/index.js.map +1 -1
  27. package/dist-server/server/modules/orchestration/a2a/adapter-registry.js +73 -0
  28. package/dist-server/server/modules/orchestration/a2a/adapter-registry.js.map +1 -0
  29. package/dist-server/server/modules/orchestration/a2a/adapters/abstract-a2a.adapter.js +17 -0
  30. package/dist-server/server/modules/orchestration/a2a/adapters/abstract-a2a.adapter.js.map +1 -0
  31. package/dist-server/server/modules/orchestration/a2a/adapters/claude-code.adapter.js +234 -0
  32. package/dist-server/server/modules/orchestration/a2a/adapters/claude-code.adapter.js.map +1 -0
  33. package/dist-server/server/modules/orchestration/a2a/adapters/codex.adapter.js +202 -0
  34. package/dist-server/server/modules/orchestration/a2a/adapters/codex.adapter.js.map +1 -0
  35. package/dist-server/server/modules/orchestration/a2a/adapters/cursor.adapter.js +205 -0
  36. package/dist-server/server/modules/orchestration/a2a/adapters/cursor.adapter.js.map +1 -0
  37. package/dist-server/server/modules/orchestration/a2a/adapters/gemini.adapter.js +205 -0
  38. package/dist-server/server/modules/orchestration/a2a/adapters/gemini.adapter.js.map +1 -0
  39. package/dist-server/server/modules/orchestration/a2a/adapters/opencode.adapter.js +205 -0
  40. package/dist-server/server/modules/orchestration/a2a/adapters/opencode.adapter.js.map +1 -0
  41. package/dist-server/server/modules/orchestration/a2a/adapters/qwen.adapter.js +205 -0
  42. package/dist-server/server/modules/orchestration/a2a/adapters/qwen.adapter.js.map +1 -0
  43. package/dist-server/server/modules/orchestration/a2a/agent-card.js +50 -0
  44. package/dist-server/server/modules/orchestration/a2a/agent-card.js.map +1 -0
  45. package/dist-server/server/modules/orchestration/a2a/auth.middleware.js +25 -0
  46. package/dist-server/server/modules/orchestration/a2a/auth.middleware.js.map +1 -0
  47. package/dist-server/server/modules/orchestration/a2a/bus.js +34 -0
  48. package/dist-server/server/modules/orchestration/a2a/bus.js.map +1 -0
  49. package/dist-server/server/modules/orchestration/a2a/routes.js +497 -0
  50. package/dist-server/server/modules/orchestration/a2a/routes.js.map +1 -0
  51. package/dist-server/server/modules/orchestration/a2a/task-store.js +144 -0
  52. package/dist-server/server/modules/orchestration/a2a/task-store.js.map +1 -0
  53. package/dist-server/server/modules/orchestration/a2a/types.js +6 -0
  54. package/dist-server/server/modules/orchestration/a2a/types.js.map +1 -0
  55. package/dist-server/server/modules/orchestration/a2a/validator.js +101 -0
  56. package/dist-server/server/modules/orchestration/a2a/validator.js.map +1 -0
  57. package/dist-server/server/modules/orchestration/index.js +24 -0
  58. package/dist-server/server/modules/orchestration/index.js.map +1 -0
  59. package/dist-server/server/modules/orchestration/preview/port-watcher.js +90 -0
  60. package/dist-server/server/modules/orchestration/preview/port-watcher.js.map +1 -0
  61. package/dist-server/server/modules/orchestration/preview/preview-proxy.js +58 -0
  62. package/dist-server/server/modules/orchestration/preview/preview-proxy.js.map +1 -0
  63. package/dist-server/server/modules/orchestration/preview/types.js +2 -0
  64. package/dist-server/server/modules/orchestration/preview/types.js.map +1 -0
  65. package/dist-server/server/modules/orchestration/tasks/orchestration-task-store.js +37 -0
  66. package/dist-server/server/modules/orchestration/tasks/orchestration-task-store.js.map +1 -0
  67. package/dist-server/server/modules/orchestration/tasks/orchestration-task.routes.js +68 -0
  68. package/dist-server/server/modules/orchestration/tasks/orchestration-task.routes.js.map +1 -0
  69. package/dist-server/server/modules/orchestration/tasks/orchestration-task.service.js +128 -0
  70. package/dist-server/server/modules/orchestration/tasks/orchestration-task.service.js.map +1 -0
  71. package/dist-server/server/modules/orchestration/tasks/orchestration-task.types.js +2 -0
  72. package/dist-server/server/modules/orchestration/tasks/orchestration-task.types.js.map +1 -0
  73. package/dist-server/server/modules/orchestration/workflows/built-in-workflows.js +126 -0
  74. package/dist-server/server/modules/orchestration/workflows/built-in-workflows.js.map +1 -0
  75. package/dist-server/server/modules/orchestration/workflows/workflow-runner.js +1047 -0
  76. package/dist-server/server/modules/orchestration/workflows/workflow-runner.js.map +1 -0
  77. package/dist-server/server/modules/orchestration/workflows/workflow-store.js +76 -0
  78. package/dist-server/server/modules/orchestration/workflows/workflow-store.js.map +1 -0
  79. package/dist-server/server/modules/orchestration/workflows/workflow.routes.js +151 -0
  80. package/dist-server/server/modules/orchestration/workflows/workflow.routes.js.map +1 -0
  81. package/dist-server/server/modules/orchestration/workflows/workflow.types.js +2 -0
  82. package/dist-server/server/modules/orchestration/workflows/workflow.types.js.map +1 -0
  83. package/dist-server/server/modules/orchestration/workflows/workspace-target.js +98 -0
  84. package/dist-server/server/modules/orchestration/workflows/workspace-target.js.map +1 -0
  85. package/dist-server/server/modules/orchestration/workspace/docker-workspace.js +122 -0
  86. package/dist-server/server/modules/orchestration/workspace/docker-workspace.js.map +1 -0
  87. package/dist-server/server/modules/orchestration/workspace/path-safety.js +48 -0
  88. package/dist-server/server/modules/orchestration/workspace/path-safety.js.map +1 -0
  89. package/dist-server/server/modules/orchestration/workspace/types.js +11 -0
  90. package/dist-server/server/modules/orchestration/workspace/types.js.map +1 -0
  91. package/dist-server/server/modules/orchestration/workspace/workspace-manager.js +80 -0
  92. package/dist-server/server/modules/orchestration/workspace/workspace-manager.js.map +1 -0
  93. package/dist-server/server/modules/orchestration/workspace/worktree-workspace.js +96 -0
  94. package/dist-server/server/modules/orchestration/workspace/worktree-workspace.js.map +1 -0
  95. package/dist-server/server/modules/providers/index.js +3 -0
  96. package/dist-server/server/modules/providers/index.js.map +1 -0
  97. package/dist-server/server/openai-codex.js +35 -4
  98. package/dist-server/server/openai-codex.js.map +1 -1
  99. package/dist-server/server/routes/taskmaster.js +106 -89
  100. package/dist-server/server/routes/taskmaster.js.map +1 -1
  101. package/package.json +3 -1
  102. package/scripts/smoke/a2a-roundtrip.mjs +167 -0
  103. package/scripts/smoke/orchestration-api.mjs +172 -0
  104. package/scripts/smoke/orchestration-live-run.mjs +176 -0
  105. package/server/claude-sdk.js +48 -7
  106. package/server/cli.js +12 -17
  107. package/server/daemon-manager.js +90 -51
  108. package/server/database/db.js +794 -794
  109. package/server/database/json-store.js +8 -5
  110. package/server/index.js +49 -9
  111. package/server/modules/orchestration/a2a/adapter-registry.ts +108 -0
  112. package/server/modules/orchestration/a2a/adapters/abstract-a2a.adapter.ts +55 -0
  113. package/server/modules/orchestration/a2a/adapters/claude-code.adapter.ts +284 -0
  114. package/server/modules/orchestration/a2a/adapters/codex.adapter.ts +244 -0
  115. package/server/modules/orchestration/a2a/adapters/cursor.adapter.ts +249 -0
  116. package/server/modules/orchestration/a2a/adapters/gemini.adapter.ts +248 -0
  117. package/server/modules/orchestration/a2a/adapters/opencode.adapter.ts +248 -0
  118. package/server/modules/orchestration/a2a/adapters/qwen.adapter.ts +248 -0
  119. package/server/modules/orchestration/a2a/agent-card.ts +55 -0
  120. package/server/modules/orchestration/a2a/auth.middleware.ts +29 -0
  121. package/server/modules/orchestration/a2a/bus.ts +46 -0
  122. package/server/modules/orchestration/a2a/routes.ts +577 -0
  123. package/server/modules/orchestration/a2a/task-store.ts +178 -0
  124. package/server/modules/orchestration/a2a/types.ts +125 -0
  125. package/server/modules/orchestration/a2a/validator.ts +113 -0
  126. package/server/modules/orchestration/index.ts +66 -0
  127. package/server/modules/orchestration/preview/port-watcher.ts +112 -0
  128. package/server/modules/orchestration/preview/preview-proxy.ts +60 -0
  129. package/server/modules/orchestration/preview/types.ts +19 -0
  130. package/server/modules/orchestration/tasks/orchestration-task-store.ts +45 -0
  131. package/server/modules/orchestration/tasks/orchestration-task.routes.ts +73 -0
  132. package/server/modules/orchestration/tasks/orchestration-task.service.ts +145 -0
  133. package/server/modules/orchestration/tasks/orchestration-task.types.ts +29 -0
  134. package/server/modules/orchestration/workflows/built-in-workflows.ts +127 -0
  135. package/server/modules/orchestration/workflows/workflow-runner.ts +1206 -0
  136. package/server/modules/orchestration/workflows/workflow-store.ts +97 -0
  137. package/server/modules/orchestration/workflows/workflow.routes.ts +169 -0
  138. package/server/modules/orchestration/workflows/workflow.types.ts +70 -0
  139. package/server/modules/orchestration/workflows/workspace-target.ts +120 -0
  140. package/server/modules/orchestration/workspace/docker-workspace.ts +135 -0
  141. package/server/modules/orchestration/workspace/path-safety.ts +55 -0
  142. package/server/modules/orchestration/workspace/types.ts +52 -0
  143. package/server/modules/orchestration/workspace/workspace-manager.ts +97 -0
  144. package/server/modules/orchestration/workspace/worktree-workspace.ts +125 -0
  145. package/server/modules/providers/index.ts +2 -0
  146. package/server/modules/providers/list/opencode/opencode-auth.provider.ts +130 -130
  147. package/server/modules/providers/list/opencode/opencode-mcp.provider.ts +126 -126
  148. package/server/modules/providers/list/opencode/opencode.provider.ts +29 -29
  149. package/server/modules/providers/list/qwen/qwen-auth.provider.ts +145 -145
  150. package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -114
  151. package/server/modules/providers/list/qwen/qwen.provider.ts +21 -21
  152. package/server/modules/providers/shared/provider-configs.ts +142 -142
  153. package/server/openai-codex.js +40 -4
  154. package/server/qwen-code-cli.js +395 -395
  155. package/server/qwen-response-handler.js +73 -73
  156. package/server/routes/qwen.js +27 -27
  157. package/server/routes/taskmaster.js +116 -91
  158. package/server/services/external-access.js +171 -171
  159. package/server/services/provider-models.js +381 -381
  160. package/server/services/telegram/telegram-http-client.js +130 -130
  161. package/server/services/vapid-keys.js +36 -36
  162. package/server/utils/port-access.js +209 -209
  163. package/dist/assets/index-B1ghfb4w.css +0 -32
  164. package/dist/assets/index-oLYHJ2X5.js +0 -852
@@ -0,0 +1,248 @@
1
+ // server/modules/orchestration/a2a/adapters/gemini.adapter.ts
2
+ // Wraps the existing server/gemini-cli.js runtime, which emits
3
+ // provider-neutral NormalizedMessage frames over a writer interface.
4
+
5
+ import crypto from 'node:crypto';
6
+
7
+ // eslint-disable-next-line boundaries/no-unknown -- gemini-cli.js is a top-level CLI runtime not yet classified by eslint.config.js; cleanup deferred.
8
+ import { abortGeminiSession, spawnGemini } from '@/gemini-cli.js';
9
+ import { AbstractA2AAdapter } from '@/modules/orchestration/a2a/adapters/abstract-a2a.adapter.js';
10
+ import type {
11
+ AdapterContext,
12
+ TaskHandle,
13
+ } from '@/modules/orchestration/a2a/adapters/abstract-a2a.adapter.js';
14
+ import type { AgentCard, Part, Task } from '@/modules/orchestration/a2a/types.js';
15
+ import type { NormalizedMessage } from '@/shared/types.js';
16
+
17
+ interface FakeWriter {
18
+ send(data: unknown): void;
19
+ getSessionId(): string | null;
20
+ setSessionId(sessionId: string): void;
21
+ }
22
+
23
+ function joinPartsToPrompt(parts: Part[]): string {
24
+ return parts
25
+ .map((part) => {
26
+ if (part.kind === 'text') return part.text;
27
+ if (part.kind === 'data') return JSON.stringify(part.data);
28
+ return `[file:${part.name}${part.uri ? ` uri=${part.uri}` : ''}]`;
29
+ })
30
+ .join('\n');
31
+ }
32
+
33
+ function newId(prefix: string): string {
34
+ return `${prefix}_${crypto.randomBytes(8).toString('hex')}`;
35
+ }
36
+
37
+ export class GeminiA2AAdapter extends AbstractA2AAdapter {
38
+ readonly id = 'gemini';
39
+
40
+ readonly agentCard: AgentCard = {
41
+ name: 'pixcode-gemini',
42
+ description: 'Google Gemini CLI, accessed via Pixcode',
43
+ url: '/a2a/agents/gemini',
44
+ version: '1.0.0',
45
+ capabilities: ['streaming', 'fileEdit', 'commandExec', 'mcp'],
46
+ skills: [
47
+ {
48
+ id: 'rapid-prototyping',
49
+ description: 'Fast implementation and iteration for coding tasks',
50
+ },
51
+ {
52
+ id: 'tool-augmented-editing',
53
+ description: 'Use CLI tools and MCP servers while editing code',
54
+ },
55
+ {
56
+ id: 'multi-file-refactor',
57
+ description: 'Coordinate edits across multiple files',
58
+ },
59
+ ],
60
+ authentication: { type: 'bearer' },
61
+ };
62
+
63
+ private readonly active = new Map<string, { sessionId: string | null }>();
64
+
65
+ async submitTask(task: Task, ctx: AdapterContext): Promise<TaskHandle> {
66
+ const promptText = joinPartsToPrompt(
67
+ task.history[task.history.length - 1]?.parts ?? [],
68
+ );
69
+ const session = { sessionId: null as string | null };
70
+ this.active.set(task.id, session);
71
+
72
+ this.emitState(task.id, 'working');
73
+
74
+ const fakeWriter: FakeWriter = {
75
+ send: (data) => this.handleRuntimeFrame(task.id, data, session),
76
+ getSessionId: () => session.sessionId,
77
+ setSessionId: (sessionId) => {
78
+ session.sessionId = sessionId;
79
+ },
80
+ };
81
+
82
+ const finished = (async () => {
83
+ try {
84
+ await spawnGemini(
85
+ promptText,
86
+ {
87
+ cwd: ctx.cwd,
88
+ model: ctx.model,
89
+ permissionMode: ctx.permissionMode === 'bypassPermissions' ? 'yolo' : ctx.permissionMode,
90
+ toolsSettings: ctx.toolsSettings,
91
+ },
92
+ fakeWriter,
93
+ );
94
+
95
+ if (this.active.has(task.id)) {
96
+ this.emitState(task.id, 'completed');
97
+ }
98
+ } catch (err) {
99
+ if (this.active.has(task.id)) {
100
+ this.emitState(task.id, 'failed', {
101
+ code: 'ADAPTER_RUNTIME_ERROR',
102
+ message: err instanceof Error ? err.message : String(err),
103
+ });
104
+ }
105
+ } finally {
106
+ this.active.delete(task.id);
107
+ }
108
+ })();
109
+
110
+ return {
111
+ cancel: () => this.cancelTask(task.id),
112
+ finished,
113
+ };
114
+ }
115
+
116
+ async cancelTask(taskId: string): Promise<void> {
117
+ const session = this.active.get(taskId);
118
+ if (!session) {
119
+ this.emitState(taskId, 'canceled');
120
+ return;
121
+ }
122
+
123
+ this.active.delete(taskId);
124
+ if (session.sessionId) {
125
+ abortGeminiSession(session.sessionId);
126
+ }
127
+ this.emitState(taskId, 'canceled');
128
+ }
129
+
130
+ private handleRuntimeFrame(
131
+ taskId: string,
132
+ frame: unknown,
133
+ session: { sessionId: string | null },
134
+ ): void {
135
+ if (!frame || typeof frame !== 'object') return;
136
+ const message = frame as Partial<NormalizedMessage>;
137
+
138
+ if (
139
+ message.kind === 'session_created' &&
140
+ typeof message.newSessionId === 'string' &&
141
+ !session.sessionId
142
+ ) {
143
+ session.sessionId = message.newSessionId;
144
+ return;
145
+ }
146
+
147
+ switch (message.kind) {
148
+ case 'stream_delta':
149
+ case 'text': {
150
+ const text = typeof message.content === 'string' ? message.content : null;
151
+ if (text && text.trim()) {
152
+ this.emitMessage(taskId, {
153
+ messageId: typeof message.id === 'string' ? message.id : newId('msg'),
154
+ role: 'agent',
155
+ parts: [{ kind: 'text', text }],
156
+ taskId,
157
+ });
158
+ }
159
+ return;
160
+ }
161
+
162
+ case 'tool_use': {
163
+ this.emitArtifact(taskId, {
164
+ artifactId: typeof message.id === 'string' ? message.id : newId('art'),
165
+ type: 'command-output',
166
+ parts: [
167
+ {
168
+ kind: 'data',
169
+ data: {
170
+ toolName: message.toolName,
171
+ toolInput: message.toolInput,
172
+ toolId: message.toolId,
173
+ toolResult: message.toolResult,
174
+ },
175
+ },
176
+ ],
177
+ metadata: { source: 'gemini-tool-use' },
178
+ });
179
+ return;
180
+ }
181
+
182
+ case 'tool_result': {
183
+ this.emitArtifact(taskId, {
184
+ artifactId: typeof message.id === 'string' ? message.id : newId('art'),
185
+ type: 'command-output',
186
+ parts: [
187
+ {
188
+ kind: 'data',
189
+ data: {
190
+ toolId: message.toolId,
191
+ content: message.content,
192
+ isError: message.isError,
193
+ },
194
+ },
195
+ ],
196
+ metadata: { source: 'gemini-tool-result' },
197
+ });
198
+ return;
199
+ }
200
+
201
+ case 'status': {
202
+ this.emitArtifact(taskId, {
203
+ artifactId: typeof message.id === 'string' ? message.id : newId('art'),
204
+ type: 'data',
205
+ parts: [
206
+ {
207
+ kind: 'data',
208
+ data: {
209
+ kind: 'status',
210
+ text: message.text,
211
+ tokens: message.tokens,
212
+ },
213
+ },
214
+ ],
215
+ metadata: { source: 'gemini-status' },
216
+ });
217
+ return;
218
+ }
219
+
220
+ case 'stream_end':
221
+ case 'complete':
222
+ case 'session_created':
223
+ return;
224
+
225
+ case 'error': {
226
+ const text =
227
+ typeof message.content === 'string' && message.content.length > 0
228
+ ? message.content
229
+ : 'Gemini reported an error';
230
+ this.emitState(taskId, 'failed', {
231
+ code: 'ADAPTER_RUNTIME_ERROR',
232
+ message: text,
233
+ });
234
+ this.active.delete(taskId);
235
+ return;
236
+ }
237
+
238
+ default: {
239
+ this.emitArtifact(taskId, {
240
+ artifactId: typeof message.id === 'string' ? message.id : newId('art'),
241
+ type: 'data',
242
+ parts: [{ kind: 'data', data: message as Record<string, unknown> }],
243
+ metadata: { source: `gemini-${String(message.kind ?? 'event')}` },
244
+ });
245
+ }
246
+ }
247
+ }
248
+ }
@@ -0,0 +1,248 @@
1
+ // server/modules/orchestration/a2a/adapters/opencode.adapter.ts
2
+ // Wraps the existing server/opencode-cli.js runtime, which emits
3
+ // provider-neutral NormalizedMessage frames over a writer interface.
4
+
5
+ import crypto from 'node:crypto';
6
+
7
+ // eslint-disable-next-line boundaries/no-unknown -- opencode-cli.js is a top-level CLI runtime not yet classified by eslint.config.js; cleanup deferred.
8
+ import { abortOpencodeSession, spawnOpencode } from '@/opencode-cli.js';
9
+ import { AbstractA2AAdapter } from '@/modules/orchestration/a2a/adapters/abstract-a2a.adapter.js';
10
+ import type {
11
+ AdapterContext,
12
+ TaskHandle,
13
+ } from '@/modules/orchestration/a2a/adapters/abstract-a2a.adapter.js';
14
+ import type { AgentCard, Part, Task } from '@/modules/orchestration/a2a/types.js';
15
+ import type { NormalizedMessage } from '@/shared/types.js';
16
+
17
+ interface FakeWriter {
18
+ send(data: unknown): void;
19
+ getSessionId(): string | null;
20
+ setSessionId(sessionId: string): void;
21
+ }
22
+
23
+ function joinPartsToPrompt(parts: Part[]): string {
24
+ return parts
25
+ .map((part) => {
26
+ if (part.kind === 'text') return part.text;
27
+ if (part.kind === 'data') return JSON.stringify(part.data);
28
+ return `[file:${part.name}${part.uri ? ` uri=${part.uri}` : ''}]`;
29
+ })
30
+ .join('\n');
31
+ }
32
+
33
+ function newId(prefix: string): string {
34
+ return `${prefix}_${crypto.randomBytes(8).toString('hex')}`;
35
+ }
36
+
37
+ export class OpenCodeA2AAdapter extends AbstractA2AAdapter {
38
+ readonly id = 'opencode';
39
+
40
+ readonly agentCard: AgentCard = {
41
+ name: 'pixcode-opencode',
42
+ description: 'OpenCode CLI, accessed via Pixcode',
43
+ url: '/a2a/agents/opencode',
44
+ version: '1.0.0',
45
+ capabilities: ['streaming', 'fileEdit', 'commandExec', 'multiProvider'],
46
+ skills: [
47
+ {
48
+ id: 'rapid-prototyping',
49
+ description: 'Fast implementation with multi-provider CLI routing',
50
+ },
51
+ {
52
+ id: 'plan-mode',
53
+ description: 'Read-only planning mode for change proposals',
54
+ },
55
+ {
56
+ id: 'tool-augmented-editing',
57
+ description: 'Use CLI tools while editing code across a workspace',
58
+ },
59
+ ],
60
+ authentication: { type: 'bearer' },
61
+ };
62
+
63
+ private readonly active = new Map<string, { sessionId: string | null }>();
64
+
65
+ async submitTask(task: Task, ctx: AdapterContext): Promise<TaskHandle> {
66
+ const promptText = joinPartsToPrompt(
67
+ task.history[task.history.length - 1]?.parts ?? [],
68
+ );
69
+ const session = { sessionId: null as string | null };
70
+ this.active.set(task.id, session);
71
+
72
+ this.emitState(task.id, 'working');
73
+
74
+ const fakeWriter: FakeWriter = {
75
+ send: (data) => this.handleRuntimeFrame(task.id, data, session),
76
+ getSessionId: () => session.sessionId,
77
+ setSessionId: (sessionId) => {
78
+ session.sessionId = sessionId;
79
+ },
80
+ };
81
+
82
+ const finished = (async () => {
83
+ try {
84
+ await spawnOpencode(
85
+ promptText,
86
+ {
87
+ cwd: ctx.cwd,
88
+ model: ctx.model,
89
+ permissionMode: ctx.permissionMode,
90
+ toolsSettings: ctx.toolsSettings,
91
+ },
92
+ fakeWriter,
93
+ );
94
+
95
+ if (this.active.has(task.id)) {
96
+ this.emitState(task.id, 'completed');
97
+ }
98
+ } catch (err) {
99
+ if (this.active.has(task.id)) {
100
+ this.emitState(task.id, 'failed', {
101
+ code: 'ADAPTER_RUNTIME_ERROR',
102
+ message: err instanceof Error ? err.message : String(err),
103
+ });
104
+ }
105
+ } finally {
106
+ this.active.delete(task.id);
107
+ }
108
+ })();
109
+
110
+ return {
111
+ cancel: () => this.cancelTask(task.id),
112
+ finished,
113
+ };
114
+ }
115
+
116
+ async cancelTask(taskId: string): Promise<void> {
117
+ const session = this.active.get(taskId);
118
+ if (!session) {
119
+ this.emitState(taskId, 'canceled');
120
+ return;
121
+ }
122
+
123
+ this.active.delete(taskId);
124
+ if (session.sessionId) {
125
+ abortOpencodeSession(session.sessionId);
126
+ }
127
+ this.emitState(taskId, 'canceled');
128
+ }
129
+
130
+ private handleRuntimeFrame(
131
+ taskId: string,
132
+ frame: unknown,
133
+ session: { sessionId: string | null },
134
+ ): void {
135
+ if (!frame || typeof frame !== 'object') return;
136
+ const message = frame as Partial<NormalizedMessage>;
137
+
138
+ if (
139
+ message.kind === 'session_created' &&
140
+ typeof message.newSessionId === 'string' &&
141
+ !session.sessionId
142
+ ) {
143
+ session.sessionId = message.newSessionId;
144
+ return;
145
+ }
146
+
147
+ switch (message.kind) {
148
+ case 'stream_delta':
149
+ case 'text': {
150
+ const text = typeof message.content === 'string' ? message.content : null;
151
+ if (text && text.trim()) {
152
+ this.emitMessage(taskId, {
153
+ messageId: typeof message.id === 'string' ? message.id : newId('msg'),
154
+ role: 'agent',
155
+ parts: [{ kind: 'text', text }],
156
+ taskId,
157
+ });
158
+ }
159
+ return;
160
+ }
161
+
162
+ case 'tool_use': {
163
+ this.emitArtifact(taskId, {
164
+ artifactId: typeof message.id === 'string' ? message.id : newId('art'),
165
+ type: 'command-output',
166
+ parts: [
167
+ {
168
+ kind: 'data',
169
+ data: {
170
+ toolName: message.toolName,
171
+ toolInput: message.toolInput,
172
+ toolId: message.toolId,
173
+ toolResult: message.toolResult,
174
+ },
175
+ },
176
+ ],
177
+ metadata: { source: 'opencode-tool-use' },
178
+ });
179
+ return;
180
+ }
181
+
182
+ case 'tool_result': {
183
+ this.emitArtifact(taskId, {
184
+ artifactId: typeof message.id === 'string' ? message.id : newId('art'),
185
+ type: 'command-output',
186
+ parts: [
187
+ {
188
+ kind: 'data',
189
+ data: {
190
+ toolId: message.toolId,
191
+ content: message.content,
192
+ isError: message.isError,
193
+ },
194
+ },
195
+ ],
196
+ metadata: { source: 'opencode-tool-result' },
197
+ });
198
+ return;
199
+ }
200
+
201
+ case 'status': {
202
+ this.emitArtifact(taskId, {
203
+ artifactId: typeof message.id === 'string' ? message.id : newId('art'),
204
+ type: 'data',
205
+ parts: [
206
+ {
207
+ kind: 'data',
208
+ data: {
209
+ kind: 'status',
210
+ text: message.text,
211
+ tokens: message.tokens,
212
+ },
213
+ },
214
+ ],
215
+ metadata: { source: 'opencode-status' },
216
+ });
217
+ return;
218
+ }
219
+
220
+ case 'stream_end':
221
+ case 'complete':
222
+ case 'session_created':
223
+ return;
224
+
225
+ case 'error': {
226
+ const text =
227
+ typeof message.content === 'string' && message.content.length > 0
228
+ ? message.content
229
+ : 'OpenCode reported an error';
230
+ this.emitState(taskId, 'failed', {
231
+ code: 'ADAPTER_RUNTIME_ERROR',
232
+ message: text,
233
+ });
234
+ this.active.delete(taskId);
235
+ return;
236
+ }
237
+
238
+ default: {
239
+ this.emitArtifact(taskId, {
240
+ artifactId: typeof message.id === 'string' ? message.id : newId('art'),
241
+ type: 'data',
242
+ parts: [{ kind: 'data', data: message as Record<string, unknown> }],
243
+ metadata: { source: `opencode-${String(message.kind ?? 'event')}` },
244
+ });
245
+ }
246
+ }
247
+ }
248
+ }