@pixelbyte-software/pixcode 1.51.2 → 1.51.4
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.
- package/CODE_OF_CONDUCT.md +41 -41
- package/CONTRIBUTING.md +155 -155
- package/LICENSE +718 -718
- package/README.de.md +169 -169
- package/README.ja.md +167 -167
- package/README.ko.md +167 -167
- package/README.md +419 -419
- package/README.ru.md +169 -169
- package/README.tr.md +298 -298
- package/README.zh-CN.md +167 -167
- package/SECURITY.md +46 -46
- package/dist/api-automation.html +110 -110
- package/dist/api-docs.html +548 -548
- package/dist/assets/index-B9N-gfOQ.css +32 -0
- package/dist/assets/{index-EN9ngyxf.js → index-HfGHXhD6.js} +175 -175
- package/dist/clear-cache.html +85 -85
- package/dist/convert-icons.md +52 -52
- package/dist/docs.html +308 -308
- package/dist/favicon.svg +8 -8
- package/dist/features.html +133 -133
- package/dist/generate-icons.js +48 -48
- package/dist/humans.txt +15 -15
- package/dist/icons/codex-white.svg +3 -3
- package/dist/icons/codex.svg +3 -3
- package/dist/icons/cursor-white.svg +11 -11
- package/dist/icons/icon-128x128.svg +9 -9
- package/dist/icons/icon-144x144.svg +9 -9
- package/dist/icons/icon-152x152.svg +9 -9
- package/dist/icons/icon-192x192.svg +9 -9
- package/dist/icons/icon-384x384.svg +9 -9
- package/dist/icons/icon-512x512.svg +9 -9
- package/dist/icons/icon-72x72.svg +9 -9
- package/dist/icons/icon-96x96.svg +9 -9
- package/dist/icons/icon-template.svg +9 -9
- package/dist/icons/qwen-logo.svg +14 -14
- package/dist/index.html +59 -59
- package/dist/landing.html +268 -268
- package/dist/llms-full.txt +119 -119
- package/dist/llms.txt +53 -53
- package/dist/logo.svg +12 -12
- package/dist/manifest.json +60 -60
- package/dist/openapi.yaml +1696 -1696
- package/dist/orchestration.html +125 -125
- package/dist/robots.txt +4 -4
- package/dist/site.css +692 -692
- package/dist/sitemap.xml +51 -51
- package/dist/sw.js +132 -132
- package/dist-server/server/cli.js +96 -96
- package/dist-server/server/daemon/manager.js +33 -33
- package/dist-server/server/daemon-manager.js +64 -64
- package/dist-server/server/database/db.js +14 -2
- package/dist-server/server/database/db.js.map +1 -1
- package/dist-server/server/index.js +191 -31
- package/dist-server/server/index.js.map +1 -1
- package/dist-server/server/middleware/auth.js +16 -5
- package/dist-server/server/middleware/auth.js.map +1 -1
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js +84 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js.map +1 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js +43 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js.map +1 -0
- package/dist-server/server/modules/orchestration/hermes/hermes.routes.js +55 -1
- package/dist-server/server/modules/orchestration/hermes/hermes.routes.js.map +1 -1
- package/dist-server/server/modules/orchestration/index.js +1 -0
- package/dist-server/server/modules/orchestration/index.js.map +1 -1
- package/dist-server/server/routes/auth.js +12 -5
- package/dist-server/server/routes/auth.js.map +1 -1
- package/dist-server/server/routes/commands.js +25 -25
- package/dist-server/server/routes/git.js +29 -17
- package/dist-server/server/routes/git.js.map +1 -1
- package/dist-server/server/routes/live-view.js +46 -46
- package/dist-server/server/routes/platformization.js +7 -6
- package/dist-server/server/routes/platformization.js.map +1 -1
- package/dist-server/server/services/hermes-gateway.js +310 -0
- package/dist-server/server/services/hermes-gateway.js.map +1 -1
- package/dist-server/server/services/platformization.js +58 -2
- package/dist-server/server/services/platformization.js.map +1 -1
- package/dist-server/server/services/public-api-manifest.js +59 -51
- package/dist-server/server/services/public-api-manifest.js.map +1 -1
- package/package.json +222 -222
- package/scripts/fix-node-pty.js +67 -67
- package/scripts/github/create-v1.38-issues.mjs +351 -351
- package/scripts/github/create-vscode-workbench-issues.mjs +121 -121
- package/scripts/hermes/configure-pixcode-mcp.mjs +165 -163
- package/scripts/hermes/pixcode-mcp-server.mjs +1009 -958
- package/scripts/smoke/changes-panel-layout.mjs +48 -48
- package/scripts/smoke/chat-composer-fixed-layout.mjs +55 -55
- package/scripts/smoke/chat-message-timeline-order.mjs +41 -41
- package/scripts/smoke/chat-realtime-hydration.mjs +44 -44
- package/scripts/smoke/chat-session-provider-pools.mjs +35 -35
- package/scripts/smoke/chat-session-state.mjs +19 -19
- package/scripts/smoke/code-editor-theme.mjs +55 -55
- package/scripts/smoke/code-editor-vscode-engine.mjs +91 -91
- package/scripts/smoke/command-center-agent-writes.mjs +79 -79
- package/scripts/smoke/command-center-non-git.mjs +46 -46
- package/scripts/smoke/context-packet.mjs +43 -43
- package/scripts/smoke/control-room-ux-redesign.mjs +91 -91
- package/scripts/smoke/daemon-entrypoint.mjs +20 -20
- package/scripts/smoke/default-landing-routing.mjs +33 -33
- package/scripts/smoke/desktop-native-notifications.mjs +30 -30
- package/scripts/smoke/desktop-tray-icon.mjs +33 -33
- package/scripts/smoke/discord-release-workflow.mjs +24 -24
- package/scripts/smoke/git-install-update.mjs +255 -255
- package/scripts/smoke/handoff-artifact-protocol.mjs +50 -50
- package/scripts/smoke/hermes-api-install.mjs +56 -56
- package/scripts/smoke/hermes-gateway-persistence.mjs +104 -104
- package/scripts/smoke/hermes-mcp-pixcode-roundtrip.mjs +426 -367
- package/scripts/smoke/hermes-rest-chat-api.mjs +162 -162
- package/scripts/smoke/hermes-rest-chat-live.mjs +45 -45
- package/scripts/smoke/hermes-rest-codex-launch.mjs +209 -209
- package/scripts/smoke/hermes-rest-gateway.mjs +79 -70
- package/scripts/smoke/hermes-rest-live.mjs +42 -42
- package/scripts/smoke/hermes-roundtrip.mjs +167 -167
- package/scripts/smoke/hermes-settings-commands.mjs +349 -346
- package/scripts/smoke/hermes-smoke-launcher-guard.mjs +34 -34
- package/scripts/smoke/live-view-diagnostics.mjs +53 -53
- package/scripts/smoke/live-view-environment.mjs +92 -92
- package/scripts/smoke/live-view-integration.mjs +450 -450
- package/scripts/smoke/mac-desktop-runtime.mjs +37 -37
- package/scripts/smoke/mobile-tunnel-guidance.mjs +29 -29
- package/scripts/smoke/model-registry.mjs +36 -36
- package/scripts/smoke/multi-project-ui.mjs +45 -45
- package/scripts/smoke/multi-worker-slots.mjs +42 -42
- package/scripts/smoke/notification-center.mjs +87 -87
- package/scripts/smoke/notification-inapp-preference.mjs +23 -23
- package/scripts/smoke/notification-taxonomy.mjs +58 -58
- package/scripts/smoke/orchestration-api.mjs +172 -172
- package/scripts/smoke/orchestration-execution-dashboard.mjs +33 -33
- package/scripts/smoke/orchestration-live-run.mjs +176 -176
- package/scripts/smoke/orchestration-mobile-scroll.mjs +29 -29
- package/scripts/smoke/orchestration-model-sync.mjs +30 -30
- package/scripts/smoke/orchestration-permission-fallback.mjs +34 -34
- package/scripts/smoke/orchestration-runtime-guards.mjs +48 -48
- package/scripts/smoke/orchestration-user-facing-output.mjs +25 -25
- package/scripts/smoke/permission-policy.mjs +50 -50
- package/scripts/smoke/pixcode-workbench-1-48.mjs +167 -167
- package/scripts/smoke/provider-models-opencode-live.mjs +66 -66
- package/scripts/smoke/provider-rest-api.mjs +124 -124
- package/scripts/smoke/provider-selection-status.mjs +52 -52
- package/scripts/smoke/run-state-refresh.mjs +52 -52
- package/scripts/smoke/runtime-manager.mjs +99 -99
- package/scripts/smoke/shell-manual-disconnect.mjs +30 -30
- package/scripts/smoke/side-panel-editor-layout.mjs +34 -34
- package/scripts/smoke/static-root-routing.mjs +21 -21
- package/scripts/smoke/strict-handoff-compact.mjs +60 -60
- package/scripts/smoke/taskmaster-config.mjs +24 -24
- package/scripts/smoke/taskmaster-execution-telegram.mjs +3 -3
- package/scripts/smoke/taskmaster-onboarding.mjs +3 -3
- package/scripts/smoke/taskmaster-run-graph.mjs +3 -3
- package/scripts/smoke/telegram-control.mjs +242 -242
- package/scripts/smoke/tunnel-persistence.mjs +56 -56
- package/scripts/smoke/update-issue-progress.mjs +69 -69
- package/scripts/smoke/update-ux.mjs +55 -55
- package/scripts/smoke/v138-completion.mjs +132 -132
- package/scripts/smoke/v138-desktop-release-hardening.mjs +69 -69
- package/scripts/smoke/v138-diagnostics.mjs +63 -63
- package/scripts/smoke/v138-issue-planner.mjs +33 -33
- package/scripts/smoke/v143-remote-control.mjs +76 -76
- package/scripts/smoke/v144-production-loop.mjs +47 -47
- package/scripts/smoke/v145-platformization.mjs +46 -46
- package/scripts/smoke/v146-control-room-ui.mjs +150 -150
- package/scripts/smoke/version-modal-autoshow.mjs +29 -29
- package/scripts/smoke/vscode-workbench-layout.mjs +63 -63
- package/scripts/smoke/vscode-workbench-polish.mjs +461 -436
- package/scripts/smoke/workflow-fallback-replay.mjs +56 -56
- package/scripts/smoke/workflow-templates.mjs +43 -43
- package/scripts/smoke/workflow-trace-timeline.mjs +46 -46
- package/scripts/update-git-install.mjs +293 -293
- package/server/claude-sdk.js +920 -920
- package/server/cli.js +1039 -1039
- package/server/constants/config.js +4 -4
- package/server/cursor-cli.js +344 -344
- package/server/daemon/manager.js +563 -563
- package/server/daemon-manager.js +964 -964
- package/server/database/db.js +908 -895
- package/server/database/json-store.js +197 -197
- package/server/gemini-cli.js +550 -550
- package/server/gemini-response-handler.js +79 -79
- package/server/index.js +201 -30
- package/server/load-env.js +35 -35
- package/server/middleware/auth.js +171 -156
- package/server/modules/orchestration/a2a/adapter-registry.ts +108 -108
- package/server/modules/orchestration/a2a/adapters/abstract-a2a.adapter.ts +63 -63
- package/server/modules/orchestration/a2a/adapters/claude-code.adapter.ts +286 -286
- package/server/modules/orchestration/a2a/adapters/codex.adapter.ts +244 -244
- package/server/modules/orchestration/a2a/adapters/cursor.adapter.ts +249 -249
- package/server/modules/orchestration/a2a/adapters/gemini.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/adapters/json-event.adapter.test.ts +60 -0
- package/server/modules/orchestration/a2a/adapters/json-event.adapter.ts +101 -0
- package/server/modules/orchestration/a2a/adapters/opencode.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/adapters/qwen.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/agent-card.ts +55 -55
- package/server/modules/orchestration/a2a/routes.ts +590 -590
- package/server/modules/orchestration/a2a/task-store.ts +178 -178
- package/server/modules/orchestration/a2a/types.ts +126 -126
- package/server/modules/orchestration/a2a/validator.ts +113 -113
- package/server/modules/orchestration/hermes/hermes.routes.ts +642 -583
- package/server/modules/orchestration/index.ts +101 -100
- package/server/modules/orchestration/preview/port-watcher.ts +112 -112
- package/server/modules/orchestration/preview/preview-proxy.ts +60 -60
- package/server/modules/orchestration/preview/types.ts +19 -19
- package/server/modules/orchestration/security/permission-policy.ts +401 -401
- package/server/modules/orchestration/tasks/orchestration-task-store.ts +41 -41
- package/server/modules/orchestration/tasks/orchestration-task.routes.ts +64 -64
- package/server/modules/orchestration/tasks/orchestration-task.service.ts +209 -209
- package/server/modules/orchestration/tasks/orchestration-task.types.ts +40 -40
- package/server/modules/orchestration/tasks/task-run-graph.ts +155 -155
- package/server/modules/orchestration/workflows/approval-queue.ts +106 -106
- package/server/modules/orchestration/workflows/built-in-workflows.ts +127 -127
- package/server/modules/orchestration/workflows/context-packet.ts +186 -186
- package/server/modules/orchestration/workflows/handoff-artifact.ts +175 -175
- package/server/modules/orchestration/workflows/workflow-fallback-policy.ts +161 -161
- package/server/modules/orchestration/workflows/workflow-replay.ts +254 -254
- package/server/modules/orchestration/workflows/workflow-runner.ts +2070 -2070
- package/server/modules/orchestration/workflows/workflow-store.ts +97 -97
- package/server/modules/orchestration/workflows/workflow-templates.ts +272 -272
- package/server/modules/orchestration/workflows/workflow-trace.ts +424 -424
- package/server/modules/orchestration/workflows/workflow.routes.ts +586 -586
- package/server/modules/orchestration/workflows/workflow.types.ts +111 -111
- package/server/modules/orchestration/workflows/workspace-target.ts +122 -122
- package/server/modules/orchestration/workspace/docker-workspace.ts +136 -136
- package/server/modules/orchestration/workspace/path-safety.ts +55 -55
- package/server/modules/orchestration/workspace/types.ts +52 -52
- package/server/modules/orchestration/workspace/workspace-manager.ts +102 -102
- package/server/modules/orchestration/workspace/worktree-workspace.ts +126 -126
- package/server/modules/providers/index.ts +2 -2
- package/server/modules/providers/list/claude/claude-auth.provider.ts +146 -146
- package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
- package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
- package/server/modules/providers/list/claude/claude.provider.ts +15 -15
- package/server/modules/providers/list/codex/codex-auth.provider.ts +117 -117
- package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
- package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
- package/server/modules/providers/list/codex/codex.provider.ts +15 -15
- package/server/modules/providers/list/cursor/cursor-auth.provider.ts +147 -147
- package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
- package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
- package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
- package/server/modules/providers/list/gemini/gemini-auth.provider.ts +173 -173
- package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
- package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
- package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
- package/server/modules/providers/list/opencode/opencode-auth.provider.ts +131 -131
- package/server/modules/providers/list/opencode/opencode-mcp.provider.ts +126 -126
- package/server/modules/providers/list/opencode/opencode-sessions.provider.ts +286 -286
- package/server/modules/providers/list/opencode/opencode.provider.ts +29 -29
- package/server/modules/providers/list/qwen/qwen-auth.provider.ts +146 -146
- package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -114
- package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +265 -265
- package/server/modules/providers/list/qwen/qwen.provider.ts +21 -21
- package/server/modules/providers/provider.registry.ts +40 -40
- package/server/modules/providers/provider.routes.ts +944 -944
- package/server/modules/providers/services/mcp.service.ts +86 -86
- package/server/modules/providers/services/provider-auth.service.ts +26 -26
- package/server/modules/providers/services/sessions.service.ts +45 -45
- package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
- package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
- package/server/modules/providers/shared/provider-configs.ts +142 -142
- package/server/modules/providers/tests/mcp.test.ts +293 -293
- package/server/openai-codex.js +462 -462
- package/server/opencode-cli.js +491 -491
- package/server/opencode-response-handler.js +111 -111
- package/server/projects.js +3008 -3008
- package/server/qwen-code-cli.js +410 -410
- package/server/qwen-response-handler.js +73 -73
- package/server/routes/agent.js +1435 -1435
- package/server/routes/auth.js +154 -146
- package/server/routes/codex.js +20 -20
- package/server/routes/commands.js +570 -570
- package/server/routes/cursor.js +61 -61
- package/server/routes/diagnostics.js +41 -41
- package/server/routes/gemini.js +25 -25
- package/server/routes/git.js +1650 -1635
- package/server/routes/live-view.js +411 -411
- package/server/routes/mcp-utils.js +13 -13
- package/server/routes/messages.js +62 -62
- package/server/routes/network.js +125 -125
- package/server/routes/platformization.js +198 -197
- package/server/routes/plugins.js +320 -320
- package/server/routes/production-agent-loop.js +90 -90
- package/server/routes/projects.js +917 -917
- package/server/routes/public-api.js +34 -34
- package/server/routes/qwen.js +27 -27
- package/server/routes/remote.js +55 -55
- package/server/routes/settings.js +321 -321
- package/server/routes/telegram.js +140 -140
- package/server/routes/user.js +125 -125
- package/server/routes/webhooks.js +63 -63
- package/server/services/control-room.js +102 -102
- package/server/services/diagnostics.js +165 -165
- package/server/services/external-access.js +375 -375
- package/server/services/hermes-gateway.js +1562 -1247
- package/server/services/hermes-install-jobs.js +729 -729
- package/server/services/install-jobs.js +715 -715
- package/server/services/live-view.js +956 -956
- package/server/services/managed-runtimes.js +493 -493
- package/server/services/model-registry.js +144 -144
- package/server/services/notification-orchestrator.js +365 -365
- package/server/services/notification-taxonomy.js +204 -204
- package/server/services/platformization.js +844 -779
- package/server/services/production-agent-loop.js +248 -248
- package/server/services/provider-cli-versions.js +149 -149
- package/server/services/provider-credentials.js +189 -189
- package/server/services/provider-models.js +396 -396
- package/server/services/public-api-manifest.js +190 -182
- package/server/services/remote-connection.js +127 -127
- package/server/services/runtime-manager.js +323 -323
- package/server/services/startup-update.js +234 -234
- package/server/services/telegram/bot.js +331 -331
- package/server/services/telegram/control-center.js +979 -979
- package/server/services/telegram/telegram-http-client.js +151 -151
- package/server/services/telegram/translations.js +340 -340
- package/server/services/vapid-keys.js +36 -36
- package/server/services/webhooks.js +216 -216
- package/server/sessionManager.js +225 -225
- package/server/shared/interfaces.ts +54 -54
- package/server/shared/types.ts +172 -172
- package/server/shared/utils.ts +193 -193
- package/server/tsconfig.json +36 -36
- package/server/utils/colors.js +21 -21
- package/server/utils/commandParser.js +305 -305
- package/server/utils/frontmatter.js +18 -18
- package/server/utils/gitConfig.js +34 -34
- package/server/utils/plugin-loader.js +457 -457
- package/server/utils/plugin-process-manager.js +185 -185
- package/server/utils/port-access.js +209 -209
- package/server/utils/runtime-paths.js +37 -37
- package/server/utils/url-detection.js +71 -71
- package/server/vite-daemon.js +79 -79
- package/shared/modelConstants.js +161 -161
- package/shared/networkHosts.js +22 -22
- package/dist/assets/index-DMz0zv6T.css +0 -32
|
@@ -1,175 +1,175 @@
|
|
|
1
|
-
export const PIXCODE_HANDOFF_PROTOCOL = 'pixcode.handoff.v1' as const;
|
|
2
|
-
|
|
3
|
-
export type WorkflowHandoffTaskStatus =
|
|
4
|
-
| 'ready'
|
|
5
|
-
| 'completed'
|
|
6
|
-
| 'blocked'
|
|
7
|
-
| 'failed'
|
|
8
|
-
| 'needs-review';
|
|
9
|
-
|
|
10
|
-
export interface WorkflowHandoffArtifact {
|
|
11
|
-
protocol: typeof PIXCODE_HANDOFF_PROTOCOL;
|
|
12
|
-
taskStatus: WorkflowHandoffTaskStatus;
|
|
13
|
-
contextSummary: string;
|
|
14
|
-
taskResult: string;
|
|
15
|
-
changedFiles: string[];
|
|
16
|
-
blockers: string[];
|
|
17
|
-
risks: string[];
|
|
18
|
-
nextAction: string;
|
|
19
|
-
nextInstructions: string;
|
|
20
|
-
producedBy?: {
|
|
21
|
-
workflowRunId?: string;
|
|
22
|
-
nodeId?: string;
|
|
23
|
-
agentLabel?: string;
|
|
24
|
-
stage?: string;
|
|
25
|
-
};
|
|
26
|
-
createdAt: string;
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
export type HandoffArtifactParseResult =
|
|
30
|
-
| { ok: true; artifact: WorkflowHandoffArtifact }
|
|
31
|
-
| { ok: false; error: string };
|
|
32
|
-
|
|
33
|
-
type HandoffArtifactMetadata = {
|
|
34
|
-
workflowRunId?: string;
|
|
35
|
-
nodeId?: string;
|
|
36
|
-
agentLabel?: string;
|
|
37
|
-
stage?: string;
|
|
38
|
-
};
|
|
39
|
-
|
|
40
|
-
const VALID_TASK_STATUSES = new Set<WorkflowHandoffTaskStatus>([
|
|
41
|
-
'ready',
|
|
42
|
-
'completed',
|
|
43
|
-
'blocked',
|
|
44
|
-
'failed',
|
|
45
|
-
'needs-review',
|
|
46
|
-
]);
|
|
47
|
-
|
|
48
|
-
function extractJsonCandidate(text: string): string | null {
|
|
49
|
-
const fenced = text.match(/```(?:json)?\s*([\s\S]*?)```/i)?.[1]?.trim();
|
|
50
|
-
if (fenced) return fenced;
|
|
51
|
-
|
|
52
|
-
const start = text.indexOf('{');
|
|
53
|
-
const end = text.lastIndexOf('}');
|
|
54
|
-
if (start === -1 || end === -1 || end <= start) return null;
|
|
55
|
-
return text.slice(start, end + 1);
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
function readRequiredString(record: Record<string, unknown>, key: string): string | null {
|
|
59
|
-
const value = record[key];
|
|
60
|
-
if (typeof value !== 'string' || !value.trim()) return null;
|
|
61
|
-
return value.trim();
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
function sanitizeChangedFile(filePath: string): string {
|
|
65
|
-
const normalized = filePath.trim().replaceAll('\\', '/');
|
|
66
|
-
if (!normalized) return '';
|
|
67
|
-
if (!normalized.startsWith('/') && !/^[a-zA-Z]:\//.test(normalized)) return normalized;
|
|
68
|
-
return normalized.split('/').filter(Boolean).slice(-4).join('/');
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
function readStringArray(record: Record<string, unknown>, key: string): string[] | null {
|
|
72
|
-
const value = record[key];
|
|
73
|
-
if (!Array.isArray(value)) return null;
|
|
74
|
-
return value
|
|
75
|
-
.filter((item): item is string => typeof item === 'string')
|
|
76
|
-
.map((item) => key === 'changedFiles' ? sanitizeChangedFile(item) : item.trim())
|
|
77
|
-
.filter(Boolean)
|
|
78
|
-
.slice(0, 40);
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
export function parseHandoffArtifact(
|
|
82
|
-
text: string,
|
|
83
|
-
metadata: HandoffArtifactMetadata = {},
|
|
84
|
-
): HandoffArtifactParseResult {
|
|
85
|
-
const candidate = extractJsonCandidate(text);
|
|
86
|
-
if (!candidate) {
|
|
87
|
-
return { ok: false, error: 'Invalid handoff artifact: expected one JSON object.' };
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
let parsed: unknown;
|
|
91
|
-
try {
|
|
92
|
-
parsed = JSON.parse(candidate);
|
|
93
|
-
} catch (error) {
|
|
94
|
-
return {
|
|
95
|
-
ok: false,
|
|
96
|
-
error: `Invalid handoff artifact: JSON parse failed (${error instanceof Error ? error.message : String(error)}).`,
|
|
97
|
-
};
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
if (!parsed || typeof parsed !== 'object' || Array.isArray(parsed)) {
|
|
101
|
-
return { ok: false, error: 'Invalid handoff artifact: payload must be an object.' };
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
const record = parsed as Record<string, unknown>;
|
|
105
|
-
if (record.protocol !== PIXCODE_HANDOFF_PROTOCOL) {
|
|
106
|
-
return { ok: false, error: `Invalid handoff artifact: protocol must be ${PIXCODE_HANDOFF_PROTOCOL}.` };
|
|
107
|
-
}
|
|
108
|
-
|
|
109
|
-
const taskStatus = record.taskStatus;
|
|
110
|
-
if (typeof taskStatus !== 'string' || !VALID_TASK_STATUSES.has(taskStatus as WorkflowHandoffTaskStatus)) {
|
|
111
|
-
return { ok: false, error: 'Invalid handoff artifact: taskStatus is missing or unsupported.' };
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
const contextSummary = readRequiredString(record, 'contextSummary');
|
|
115
|
-
const taskResult = readRequiredString(record, 'taskResult');
|
|
116
|
-
const changedFiles = readStringArray(record, 'changedFiles');
|
|
117
|
-
const blockers = readStringArray(record, 'blockers');
|
|
118
|
-
const risks = readStringArray(record, 'risks');
|
|
119
|
-
const nextAction = readRequiredString(record, 'nextAction');
|
|
120
|
-
const nextInstructions = readRequiredString(record, 'nextInstructions');
|
|
121
|
-
|
|
122
|
-
if (!contextSummary || !taskResult || !changedFiles || !blockers || !risks || !nextAction || !nextInstructions) {
|
|
123
|
-
return {
|
|
124
|
-
ok: false,
|
|
125
|
-
error: 'Invalid handoff artifact: required fields are protocol, taskStatus, contextSummary, taskResult, changedFiles, blockers, risks, nextAction, and nextInstructions.',
|
|
126
|
-
};
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
return {
|
|
130
|
-
ok: true,
|
|
131
|
-
artifact: {
|
|
132
|
-
protocol: PIXCODE_HANDOFF_PROTOCOL,
|
|
133
|
-
taskStatus: taskStatus as WorkflowHandoffTaskStatus,
|
|
134
|
-
contextSummary,
|
|
135
|
-
taskResult,
|
|
136
|
-
changedFiles,
|
|
137
|
-
blockers,
|
|
138
|
-
risks,
|
|
139
|
-
nextAction,
|
|
140
|
-
nextInstructions,
|
|
141
|
-
producedBy: {
|
|
142
|
-
workflowRunId: metadata.workflowRunId,
|
|
143
|
-
nodeId: metadata.nodeId,
|
|
144
|
-
agentLabel: metadata.agentLabel,
|
|
145
|
-
stage: metadata.stage,
|
|
146
|
-
},
|
|
147
|
-
createdAt: new Date().toISOString(),
|
|
148
|
-
},
|
|
149
|
-
};
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
export function formatHandoffArtifactForContext(artifact: WorkflowHandoffArtifact): string {
|
|
153
|
-
return [
|
|
154
|
-
`Pixcode handoff artifact (${PIXCODE_HANDOFF_PROTOCOL})`,
|
|
155
|
-
JSON.stringify(artifact, null, 2),
|
|
156
|
-
].join('\n');
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
export function handoffArtifactToWorkflowArtifact(artifact: WorkflowHandoffArtifact): {
|
|
160
|
-
type: 'handoff-artifact';
|
|
161
|
-
data: Record<string, unknown>;
|
|
162
|
-
metadata: Record<string, unknown>;
|
|
163
|
-
} {
|
|
164
|
-
return {
|
|
165
|
-
type: 'handoff-artifact',
|
|
166
|
-
data: artifact as unknown as Record<string, unknown>,
|
|
167
|
-
metadata: {
|
|
168
|
-
protocol: artifact.protocol,
|
|
169
|
-
taskStatus: artifact.taskStatus,
|
|
170
|
-
changedFileCount: artifact.changedFiles.length,
|
|
171
|
-
blockerCount: artifact.blockers.length,
|
|
172
|
-
nextAction: artifact.nextAction,
|
|
173
|
-
},
|
|
174
|
-
};
|
|
175
|
-
}
|
|
1
|
+
export const PIXCODE_HANDOFF_PROTOCOL = 'pixcode.handoff.v1' as const;
|
|
2
|
+
|
|
3
|
+
export type WorkflowHandoffTaskStatus =
|
|
4
|
+
| 'ready'
|
|
5
|
+
| 'completed'
|
|
6
|
+
| 'blocked'
|
|
7
|
+
| 'failed'
|
|
8
|
+
| 'needs-review';
|
|
9
|
+
|
|
10
|
+
export interface WorkflowHandoffArtifact {
|
|
11
|
+
protocol: typeof PIXCODE_HANDOFF_PROTOCOL;
|
|
12
|
+
taskStatus: WorkflowHandoffTaskStatus;
|
|
13
|
+
contextSummary: string;
|
|
14
|
+
taskResult: string;
|
|
15
|
+
changedFiles: string[];
|
|
16
|
+
blockers: string[];
|
|
17
|
+
risks: string[];
|
|
18
|
+
nextAction: string;
|
|
19
|
+
nextInstructions: string;
|
|
20
|
+
producedBy?: {
|
|
21
|
+
workflowRunId?: string;
|
|
22
|
+
nodeId?: string;
|
|
23
|
+
agentLabel?: string;
|
|
24
|
+
stage?: string;
|
|
25
|
+
};
|
|
26
|
+
createdAt: string;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export type HandoffArtifactParseResult =
|
|
30
|
+
| { ok: true; artifact: WorkflowHandoffArtifact }
|
|
31
|
+
| { ok: false; error: string };
|
|
32
|
+
|
|
33
|
+
type HandoffArtifactMetadata = {
|
|
34
|
+
workflowRunId?: string;
|
|
35
|
+
nodeId?: string;
|
|
36
|
+
agentLabel?: string;
|
|
37
|
+
stage?: string;
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
const VALID_TASK_STATUSES = new Set<WorkflowHandoffTaskStatus>([
|
|
41
|
+
'ready',
|
|
42
|
+
'completed',
|
|
43
|
+
'blocked',
|
|
44
|
+
'failed',
|
|
45
|
+
'needs-review',
|
|
46
|
+
]);
|
|
47
|
+
|
|
48
|
+
function extractJsonCandidate(text: string): string | null {
|
|
49
|
+
const fenced = text.match(/```(?:json)?\s*([\s\S]*?)```/i)?.[1]?.trim();
|
|
50
|
+
if (fenced) return fenced;
|
|
51
|
+
|
|
52
|
+
const start = text.indexOf('{');
|
|
53
|
+
const end = text.lastIndexOf('}');
|
|
54
|
+
if (start === -1 || end === -1 || end <= start) return null;
|
|
55
|
+
return text.slice(start, end + 1);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
function readRequiredString(record: Record<string, unknown>, key: string): string | null {
|
|
59
|
+
const value = record[key];
|
|
60
|
+
if (typeof value !== 'string' || !value.trim()) return null;
|
|
61
|
+
return value.trim();
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
function sanitizeChangedFile(filePath: string): string {
|
|
65
|
+
const normalized = filePath.trim().replaceAll('\\', '/');
|
|
66
|
+
if (!normalized) return '';
|
|
67
|
+
if (!normalized.startsWith('/') && !/^[a-zA-Z]:\//.test(normalized)) return normalized;
|
|
68
|
+
return normalized.split('/').filter(Boolean).slice(-4).join('/');
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
function readStringArray(record: Record<string, unknown>, key: string): string[] | null {
|
|
72
|
+
const value = record[key];
|
|
73
|
+
if (!Array.isArray(value)) return null;
|
|
74
|
+
return value
|
|
75
|
+
.filter((item): item is string => typeof item === 'string')
|
|
76
|
+
.map((item) => key === 'changedFiles' ? sanitizeChangedFile(item) : item.trim())
|
|
77
|
+
.filter(Boolean)
|
|
78
|
+
.slice(0, 40);
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
export function parseHandoffArtifact(
|
|
82
|
+
text: string,
|
|
83
|
+
metadata: HandoffArtifactMetadata = {},
|
|
84
|
+
): HandoffArtifactParseResult {
|
|
85
|
+
const candidate = extractJsonCandidate(text);
|
|
86
|
+
if (!candidate) {
|
|
87
|
+
return { ok: false, error: 'Invalid handoff artifact: expected one JSON object.' };
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
let parsed: unknown;
|
|
91
|
+
try {
|
|
92
|
+
parsed = JSON.parse(candidate);
|
|
93
|
+
} catch (error) {
|
|
94
|
+
return {
|
|
95
|
+
ok: false,
|
|
96
|
+
error: `Invalid handoff artifact: JSON parse failed (${error instanceof Error ? error.message : String(error)}).`,
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
if (!parsed || typeof parsed !== 'object' || Array.isArray(parsed)) {
|
|
101
|
+
return { ok: false, error: 'Invalid handoff artifact: payload must be an object.' };
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
const record = parsed as Record<string, unknown>;
|
|
105
|
+
if (record.protocol !== PIXCODE_HANDOFF_PROTOCOL) {
|
|
106
|
+
return { ok: false, error: `Invalid handoff artifact: protocol must be ${PIXCODE_HANDOFF_PROTOCOL}.` };
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
const taskStatus = record.taskStatus;
|
|
110
|
+
if (typeof taskStatus !== 'string' || !VALID_TASK_STATUSES.has(taskStatus as WorkflowHandoffTaskStatus)) {
|
|
111
|
+
return { ok: false, error: 'Invalid handoff artifact: taskStatus is missing or unsupported.' };
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
const contextSummary = readRequiredString(record, 'contextSummary');
|
|
115
|
+
const taskResult = readRequiredString(record, 'taskResult');
|
|
116
|
+
const changedFiles = readStringArray(record, 'changedFiles');
|
|
117
|
+
const blockers = readStringArray(record, 'blockers');
|
|
118
|
+
const risks = readStringArray(record, 'risks');
|
|
119
|
+
const nextAction = readRequiredString(record, 'nextAction');
|
|
120
|
+
const nextInstructions = readRequiredString(record, 'nextInstructions');
|
|
121
|
+
|
|
122
|
+
if (!contextSummary || !taskResult || !changedFiles || !blockers || !risks || !nextAction || !nextInstructions) {
|
|
123
|
+
return {
|
|
124
|
+
ok: false,
|
|
125
|
+
error: 'Invalid handoff artifact: required fields are protocol, taskStatus, contextSummary, taskResult, changedFiles, blockers, risks, nextAction, and nextInstructions.',
|
|
126
|
+
};
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
return {
|
|
130
|
+
ok: true,
|
|
131
|
+
artifact: {
|
|
132
|
+
protocol: PIXCODE_HANDOFF_PROTOCOL,
|
|
133
|
+
taskStatus: taskStatus as WorkflowHandoffTaskStatus,
|
|
134
|
+
contextSummary,
|
|
135
|
+
taskResult,
|
|
136
|
+
changedFiles,
|
|
137
|
+
blockers,
|
|
138
|
+
risks,
|
|
139
|
+
nextAction,
|
|
140
|
+
nextInstructions,
|
|
141
|
+
producedBy: {
|
|
142
|
+
workflowRunId: metadata.workflowRunId,
|
|
143
|
+
nodeId: metadata.nodeId,
|
|
144
|
+
agentLabel: metadata.agentLabel,
|
|
145
|
+
stage: metadata.stage,
|
|
146
|
+
},
|
|
147
|
+
createdAt: new Date().toISOString(),
|
|
148
|
+
},
|
|
149
|
+
};
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
export function formatHandoffArtifactForContext(artifact: WorkflowHandoffArtifact): string {
|
|
153
|
+
return [
|
|
154
|
+
`Pixcode handoff artifact (${PIXCODE_HANDOFF_PROTOCOL})`,
|
|
155
|
+
JSON.stringify(artifact, null, 2),
|
|
156
|
+
].join('\n');
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
export function handoffArtifactToWorkflowArtifact(artifact: WorkflowHandoffArtifact): {
|
|
160
|
+
type: 'handoff-artifact';
|
|
161
|
+
data: Record<string, unknown>;
|
|
162
|
+
metadata: Record<string, unknown>;
|
|
163
|
+
} {
|
|
164
|
+
return {
|
|
165
|
+
type: 'handoff-artifact',
|
|
166
|
+
data: artifact as unknown as Record<string, unknown>,
|
|
167
|
+
metadata: {
|
|
168
|
+
protocol: artifact.protocol,
|
|
169
|
+
taskStatus: artifact.taskStatus,
|
|
170
|
+
changedFileCount: artifact.changedFiles.length,
|
|
171
|
+
blockerCount: artifact.blockers.length,
|
|
172
|
+
nextAction: artifact.nextAction,
|
|
173
|
+
},
|
|
174
|
+
};
|
|
175
|
+
}
|
|
@@ -1,161 +1,161 @@
|
|
|
1
|
-
import type { WorkflowNode, WorkflowRun } from '@/modules/orchestration/workflows/workflow.types.js';
|
|
2
|
-
|
|
3
|
-
export const PIXCODE_FALLBACK_POLICY_PROTOCOL = 'pixcode.fallback-policy.v1';
|
|
4
|
-
|
|
5
|
-
export type WorkflowFallbackTrigger = 'provider_failure' | 'timeout' | 'tool_failure' | 'invalid_output';
|
|
6
|
-
|
|
7
|
-
export interface WorkflowFallbackPolicy {
|
|
8
|
-
protocol: typeof PIXCODE_FALLBACK_POLICY_PROTOCOL;
|
|
9
|
-
enabled: boolean;
|
|
10
|
-
triggers: WorkflowFallbackTrigger[];
|
|
11
|
-
maxFallbacksPerRun: number;
|
|
12
|
-
requireDifferentAgent: boolean;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
export interface WorkflowFallbackDecision {
|
|
16
|
-
shouldFallback: boolean;
|
|
17
|
-
trigger: WorkflowFallbackTrigger;
|
|
18
|
-
reason: string;
|
|
19
|
-
policy: WorkflowFallbackPolicy;
|
|
20
|
-
skippedReason?: string;
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
const DEFAULT_TRIGGERS: WorkflowFallbackTrigger[] = [
|
|
24
|
-
'provider_failure',
|
|
25
|
-
'timeout',
|
|
26
|
-
'tool_failure',
|
|
27
|
-
'invalid_output',
|
|
28
|
-
];
|
|
29
|
-
|
|
30
|
-
export const DEFAULT_WORKFLOW_FALLBACK_POLICY: WorkflowFallbackPolicy = {
|
|
31
|
-
protocol: PIXCODE_FALLBACK_POLICY_PROTOCOL,
|
|
32
|
-
enabled: true,
|
|
33
|
-
triggers: DEFAULT_TRIGGERS,
|
|
34
|
-
maxFallbacksPerRun: 3,
|
|
35
|
-
requireDifferentAgent: true,
|
|
36
|
-
};
|
|
37
|
-
|
|
38
|
-
function readRecord(value: unknown): Record<string, unknown> | undefined {
|
|
39
|
-
return value && typeof value === 'object' ? value as Record<string, unknown> : undefined;
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
function readBoolean(value: unknown): boolean | undefined {
|
|
43
|
-
return typeof value === 'boolean' ? value : undefined;
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
function readNumber(value: unknown): number | undefined {
|
|
47
|
-
return typeof value === 'number' && Number.isFinite(value) ? value : undefined;
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
function isWorkflowFallbackTrigger(value: unknown): value is WorkflowFallbackTrigger {
|
|
51
|
-
return value === 'provider_failure'
|
|
52
|
-
|| value === 'timeout'
|
|
53
|
-
|| value === 'tool_failure'
|
|
54
|
-
|| value === 'invalid_output';
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
function readFallbackTriggers(value: unknown): WorkflowFallbackTrigger[] | undefined {
|
|
58
|
-
if (!Array.isArray(value)) return undefined;
|
|
59
|
-
const triggers = value.filter(isWorkflowFallbackTrigger);
|
|
60
|
-
return triggers.length > 0 ? [...new Set(triggers)] : undefined;
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
export function readWorkflowFallbackPolicy(metadata?: Record<string, unknown>): WorkflowFallbackPolicy {
|
|
64
|
-
const settings = readRecord(metadata?.settings) ?? {};
|
|
65
|
-
const configured = readRecord(settings.fallbackPolicy) ?? {};
|
|
66
|
-
const maxFallbacksPerRun = readNumber(configured.maxFallbacksPerRun);
|
|
67
|
-
|
|
68
|
-
return {
|
|
69
|
-
protocol: PIXCODE_FALLBACK_POLICY_PROTOCOL,
|
|
70
|
-
enabled: readBoolean(configured.enabled) ?? DEFAULT_WORKFLOW_FALLBACK_POLICY.enabled,
|
|
71
|
-
triggers: readFallbackTriggers(configured.triggers) ?? DEFAULT_WORKFLOW_FALLBACK_POLICY.triggers,
|
|
72
|
-
maxFallbacksPerRun: maxFallbacksPerRun === undefined
|
|
73
|
-
? DEFAULT_WORKFLOW_FALLBACK_POLICY.maxFallbacksPerRun
|
|
74
|
-
: Math.max(0, Math.min(8, Math.round(maxFallbacksPerRun))),
|
|
75
|
-
requireDifferentAgent: readBoolean(configured.requireDifferentAgent)
|
|
76
|
-
?? DEFAULT_WORKFLOW_FALLBACK_POLICY.requireDifferentAgent,
|
|
77
|
-
};
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
export function classifyWorkflowFailure(
|
|
81
|
-
reason: string,
|
|
82
|
-
explicitTrigger?: WorkflowFallbackTrigger,
|
|
83
|
-
): WorkflowFallbackTrigger {
|
|
84
|
-
if (explicitTrigger) return explicitTrigger;
|
|
85
|
-
|
|
86
|
-
const text = reason.toLocaleLowerCase('en');
|
|
87
|
-
if (/timed out|timeout|deadline/u.test(text)) return 'timeout';
|
|
88
|
-
if (/invalid (handoff|output|artifact|json|schema)|parse|protocol/u.test(text)) return 'invalid_output';
|
|
89
|
-
if (
|
|
90
|
-
/tool|command|shell|exit code|permission|file write|write failed|network|fetch|curl|wget|gh |npm |git /u
|
|
91
|
-
.test(text)
|
|
92
|
-
) {
|
|
93
|
-
return 'tool_failure';
|
|
94
|
-
}
|
|
95
|
-
return 'provider_failure';
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
function fallbackEventCount(run: WorkflowRun): number {
|
|
99
|
-
return Array.isArray(run.metadata?.fallbackEvents) ? run.metadata.fallbackEvents.length : 0;
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
export function resolveWorkflowFallbackDecision({
|
|
103
|
-
run,
|
|
104
|
-
node,
|
|
105
|
-
reason,
|
|
106
|
-
trigger,
|
|
107
|
-
fallbackAgentInstanceId,
|
|
108
|
-
}: {
|
|
109
|
-
run: WorkflowRun;
|
|
110
|
-
node: WorkflowNode;
|
|
111
|
-
reason: string;
|
|
112
|
-
trigger?: WorkflowFallbackTrigger;
|
|
113
|
-
fallbackAgentInstanceId?: string;
|
|
114
|
-
}): WorkflowFallbackDecision {
|
|
115
|
-
const fallbackTrigger = classifyWorkflowFailure(reason, trigger);
|
|
116
|
-
const policy = readWorkflowFallbackPolicy(run.metadata);
|
|
117
|
-
|
|
118
|
-
if (!policy.enabled) {
|
|
119
|
-
return {
|
|
120
|
-
shouldFallback: false,
|
|
121
|
-
trigger: fallbackTrigger,
|
|
122
|
-
reason,
|
|
123
|
-
policy,
|
|
124
|
-
skippedReason: 'Fallback policy is disabled.',
|
|
125
|
-
};
|
|
126
|
-
}
|
|
127
|
-
if (!policy.triggers.includes(fallbackTrigger)) {
|
|
128
|
-
return {
|
|
129
|
-
shouldFallback: false,
|
|
130
|
-
trigger: fallbackTrigger,
|
|
131
|
-
reason,
|
|
132
|
-
policy,
|
|
133
|
-
skippedReason: `Fallback trigger ${fallbackTrigger} is not enabled.`,
|
|
134
|
-
};
|
|
135
|
-
}
|
|
136
|
-
if (fallbackEventCount(run) >= policy.maxFallbacksPerRun) {
|
|
137
|
-
return {
|
|
138
|
-
shouldFallback: false,
|
|
139
|
-
trigger: fallbackTrigger,
|
|
140
|
-
reason,
|
|
141
|
-
policy,
|
|
142
|
-
skippedReason: `Fallback limit ${policy.maxFallbacksPerRun} reached.`,
|
|
143
|
-
};
|
|
144
|
-
}
|
|
145
|
-
if (policy.requireDifferentAgent && fallbackAgentInstanceId && fallbackAgentInstanceId === node.agentInstanceId) {
|
|
146
|
-
return {
|
|
147
|
-
shouldFallback: false,
|
|
148
|
-
trigger: fallbackTrigger,
|
|
149
|
-
reason,
|
|
150
|
-
policy,
|
|
151
|
-
skippedReason: 'Fallback agent must be different from the failed agent.',
|
|
152
|
-
};
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
return {
|
|
156
|
-
shouldFallback: true,
|
|
157
|
-
trigger: fallbackTrigger,
|
|
158
|
-
reason,
|
|
159
|
-
policy,
|
|
160
|
-
};
|
|
161
|
-
}
|
|
1
|
+
import type { WorkflowNode, WorkflowRun } from '@/modules/orchestration/workflows/workflow.types.js';
|
|
2
|
+
|
|
3
|
+
export const PIXCODE_FALLBACK_POLICY_PROTOCOL = 'pixcode.fallback-policy.v1';
|
|
4
|
+
|
|
5
|
+
export type WorkflowFallbackTrigger = 'provider_failure' | 'timeout' | 'tool_failure' | 'invalid_output';
|
|
6
|
+
|
|
7
|
+
export interface WorkflowFallbackPolicy {
|
|
8
|
+
protocol: typeof PIXCODE_FALLBACK_POLICY_PROTOCOL;
|
|
9
|
+
enabled: boolean;
|
|
10
|
+
triggers: WorkflowFallbackTrigger[];
|
|
11
|
+
maxFallbacksPerRun: number;
|
|
12
|
+
requireDifferentAgent: boolean;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export interface WorkflowFallbackDecision {
|
|
16
|
+
shouldFallback: boolean;
|
|
17
|
+
trigger: WorkflowFallbackTrigger;
|
|
18
|
+
reason: string;
|
|
19
|
+
policy: WorkflowFallbackPolicy;
|
|
20
|
+
skippedReason?: string;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const DEFAULT_TRIGGERS: WorkflowFallbackTrigger[] = [
|
|
24
|
+
'provider_failure',
|
|
25
|
+
'timeout',
|
|
26
|
+
'tool_failure',
|
|
27
|
+
'invalid_output',
|
|
28
|
+
];
|
|
29
|
+
|
|
30
|
+
export const DEFAULT_WORKFLOW_FALLBACK_POLICY: WorkflowFallbackPolicy = {
|
|
31
|
+
protocol: PIXCODE_FALLBACK_POLICY_PROTOCOL,
|
|
32
|
+
enabled: true,
|
|
33
|
+
triggers: DEFAULT_TRIGGERS,
|
|
34
|
+
maxFallbacksPerRun: 3,
|
|
35
|
+
requireDifferentAgent: true,
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
function readRecord(value: unknown): Record<string, unknown> | undefined {
|
|
39
|
+
return value && typeof value === 'object' ? value as Record<string, unknown> : undefined;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function readBoolean(value: unknown): boolean | undefined {
|
|
43
|
+
return typeof value === 'boolean' ? value : undefined;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
function readNumber(value: unknown): number | undefined {
|
|
47
|
+
return typeof value === 'number' && Number.isFinite(value) ? value : undefined;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
function isWorkflowFallbackTrigger(value: unknown): value is WorkflowFallbackTrigger {
|
|
51
|
+
return value === 'provider_failure'
|
|
52
|
+
|| value === 'timeout'
|
|
53
|
+
|| value === 'tool_failure'
|
|
54
|
+
|| value === 'invalid_output';
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
function readFallbackTriggers(value: unknown): WorkflowFallbackTrigger[] | undefined {
|
|
58
|
+
if (!Array.isArray(value)) return undefined;
|
|
59
|
+
const triggers = value.filter(isWorkflowFallbackTrigger);
|
|
60
|
+
return triggers.length > 0 ? [...new Set(triggers)] : undefined;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
export function readWorkflowFallbackPolicy(metadata?: Record<string, unknown>): WorkflowFallbackPolicy {
|
|
64
|
+
const settings = readRecord(metadata?.settings) ?? {};
|
|
65
|
+
const configured = readRecord(settings.fallbackPolicy) ?? {};
|
|
66
|
+
const maxFallbacksPerRun = readNumber(configured.maxFallbacksPerRun);
|
|
67
|
+
|
|
68
|
+
return {
|
|
69
|
+
protocol: PIXCODE_FALLBACK_POLICY_PROTOCOL,
|
|
70
|
+
enabled: readBoolean(configured.enabled) ?? DEFAULT_WORKFLOW_FALLBACK_POLICY.enabled,
|
|
71
|
+
triggers: readFallbackTriggers(configured.triggers) ?? DEFAULT_WORKFLOW_FALLBACK_POLICY.triggers,
|
|
72
|
+
maxFallbacksPerRun: maxFallbacksPerRun === undefined
|
|
73
|
+
? DEFAULT_WORKFLOW_FALLBACK_POLICY.maxFallbacksPerRun
|
|
74
|
+
: Math.max(0, Math.min(8, Math.round(maxFallbacksPerRun))),
|
|
75
|
+
requireDifferentAgent: readBoolean(configured.requireDifferentAgent)
|
|
76
|
+
?? DEFAULT_WORKFLOW_FALLBACK_POLICY.requireDifferentAgent,
|
|
77
|
+
};
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
export function classifyWorkflowFailure(
|
|
81
|
+
reason: string,
|
|
82
|
+
explicitTrigger?: WorkflowFallbackTrigger,
|
|
83
|
+
): WorkflowFallbackTrigger {
|
|
84
|
+
if (explicitTrigger) return explicitTrigger;
|
|
85
|
+
|
|
86
|
+
const text = reason.toLocaleLowerCase('en');
|
|
87
|
+
if (/timed out|timeout|deadline/u.test(text)) return 'timeout';
|
|
88
|
+
if (/invalid (handoff|output|artifact|json|schema)|parse|protocol/u.test(text)) return 'invalid_output';
|
|
89
|
+
if (
|
|
90
|
+
/tool|command|shell|exit code|permission|file write|write failed|network|fetch|curl|wget|gh |npm |git /u
|
|
91
|
+
.test(text)
|
|
92
|
+
) {
|
|
93
|
+
return 'tool_failure';
|
|
94
|
+
}
|
|
95
|
+
return 'provider_failure';
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
function fallbackEventCount(run: WorkflowRun): number {
|
|
99
|
+
return Array.isArray(run.metadata?.fallbackEvents) ? run.metadata.fallbackEvents.length : 0;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
export function resolveWorkflowFallbackDecision({
|
|
103
|
+
run,
|
|
104
|
+
node,
|
|
105
|
+
reason,
|
|
106
|
+
trigger,
|
|
107
|
+
fallbackAgentInstanceId,
|
|
108
|
+
}: {
|
|
109
|
+
run: WorkflowRun;
|
|
110
|
+
node: WorkflowNode;
|
|
111
|
+
reason: string;
|
|
112
|
+
trigger?: WorkflowFallbackTrigger;
|
|
113
|
+
fallbackAgentInstanceId?: string;
|
|
114
|
+
}): WorkflowFallbackDecision {
|
|
115
|
+
const fallbackTrigger = classifyWorkflowFailure(reason, trigger);
|
|
116
|
+
const policy = readWorkflowFallbackPolicy(run.metadata);
|
|
117
|
+
|
|
118
|
+
if (!policy.enabled) {
|
|
119
|
+
return {
|
|
120
|
+
shouldFallback: false,
|
|
121
|
+
trigger: fallbackTrigger,
|
|
122
|
+
reason,
|
|
123
|
+
policy,
|
|
124
|
+
skippedReason: 'Fallback policy is disabled.',
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
if (!policy.triggers.includes(fallbackTrigger)) {
|
|
128
|
+
return {
|
|
129
|
+
shouldFallback: false,
|
|
130
|
+
trigger: fallbackTrigger,
|
|
131
|
+
reason,
|
|
132
|
+
policy,
|
|
133
|
+
skippedReason: `Fallback trigger ${fallbackTrigger} is not enabled.`,
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
if (fallbackEventCount(run) >= policy.maxFallbacksPerRun) {
|
|
137
|
+
return {
|
|
138
|
+
shouldFallback: false,
|
|
139
|
+
trigger: fallbackTrigger,
|
|
140
|
+
reason,
|
|
141
|
+
policy,
|
|
142
|
+
skippedReason: `Fallback limit ${policy.maxFallbacksPerRun} reached.`,
|
|
143
|
+
};
|
|
144
|
+
}
|
|
145
|
+
if (policy.requireDifferentAgent && fallbackAgentInstanceId && fallbackAgentInstanceId === node.agentInstanceId) {
|
|
146
|
+
return {
|
|
147
|
+
shouldFallback: false,
|
|
148
|
+
trigger: fallbackTrigger,
|
|
149
|
+
reason,
|
|
150
|
+
policy,
|
|
151
|
+
skippedReason: 'Fallback agent must be different from the failed agent.',
|
|
152
|
+
};
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
return {
|
|
156
|
+
shouldFallback: true,
|
|
157
|
+
trigger: fallbackTrigger,
|
|
158
|
+
reason,
|
|
159
|
+
policy,
|
|
160
|
+
};
|
|
161
|
+
}
|