gsd-pi 2.75.0 → 2.76.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.
- package/README.md +186 -149
- package/dist/claude-cli-check.d.ts +10 -0
- package/dist/claude-cli-check.js +13 -3
- package/dist/headless-events.d.ts +1 -1
- package/dist/headless-events.js +5 -2
- package/dist/headless.js +5 -6
- package/dist/onboarding.d.ts +20 -1
- package/dist/onboarding.js +99 -39
- package/dist/resources/extensions/ask-user-questions.js +17 -5
- package/dist/resources/extensions/claude-code-cli/models.js +9 -0
- package/dist/resources/extensions/claude-code-cli/readiness.js +12 -2
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +76 -4
- package/dist/resources/extensions/gsd/auto/detect-stuck.js +9 -0
- package/dist/resources/extensions/gsd/auto/loop.js +67 -4
- package/dist/resources/extensions/gsd/auto/phases.js +72 -47
- package/dist/resources/extensions/gsd/auto/resolve.js +1 -1
- package/dist/resources/extensions/gsd/auto/run-unit.js +10 -1
- package/dist/resources/extensions/gsd/auto/session.js +5 -0
- package/dist/resources/extensions/gsd/auto-artifact-paths.js +20 -0
- package/dist/resources/extensions/gsd/auto-dashboard.js +37 -8
- package/dist/resources/extensions/gsd/auto-direct-dispatch.js +8 -2
- package/dist/resources/extensions/gsd/auto-dispatch.js +120 -14
- package/dist/resources/extensions/gsd/auto-loop.js +1 -1
- package/dist/resources/extensions/gsd/auto-model-selection.js +14 -4
- package/dist/resources/extensions/gsd/auto-post-unit.js +10 -8
- package/dist/resources/extensions/gsd/auto-prompts.js +190 -46
- package/dist/resources/extensions/gsd/auto-recovery.js +57 -0
- package/dist/resources/extensions/gsd/auto-start.js +5 -3
- package/dist/resources/extensions/gsd/auto-verification.js +3 -3
- package/dist/resources/extensions/gsd/auto-worktree.js +71 -2
- package/dist/resources/extensions/gsd/auto.js +57 -25
- package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +8 -21
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +45 -23
- package/dist/resources/extensions/gsd/bootstrap/memory-tools.js +128 -0
- package/dist/resources/extensions/gsd/bootstrap/query-tools.js +29 -0
- package/dist/resources/extensions/gsd/bootstrap/register-extension.js +2 -0
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +22 -0
- package/dist/resources/extensions/gsd/bootstrap/system-context.js +17 -4
- package/dist/resources/extensions/gsd/commands/catalog.js +81 -9
- package/dist/resources/extensions/gsd/commands/handlers/core.js +64 -24
- package/dist/resources/extensions/gsd/commands/handlers/escalate.js +171 -0
- package/dist/resources/extensions/gsd/commands/handlers/onboarding.js +159 -0
- package/dist/resources/extensions/gsd/commands/handlers/ops.js +21 -0
- package/dist/resources/extensions/gsd/commands/handlers/workflow.js +228 -29
- package/dist/resources/extensions/gsd/commands-cmux.js +5 -2
- package/dist/resources/extensions/gsd/commands-config.js +5 -0
- package/dist/resources/extensions/gsd/commands-debug.js +388 -0
- package/dist/resources/extensions/gsd/commands-do.js +1 -0
- package/dist/resources/extensions/gsd/commands-extract-learnings.js +233 -75
- package/dist/resources/extensions/gsd/commands-handlers.js +21 -2
- package/dist/resources/extensions/gsd/commands-memory.js +462 -0
- package/dist/resources/extensions/gsd/commands-prefs-wizard.js +40 -12
- package/dist/resources/extensions/gsd/commands-scan.js +94 -0
- package/dist/resources/extensions/gsd/commands-workflow-templates.js +101 -2
- package/dist/resources/extensions/gsd/custom-workflow-engine.js +74 -54
- package/dist/resources/extensions/gsd/db-writer.js +1 -0
- package/dist/resources/extensions/gsd/debug-session-store.js +238 -0
- package/dist/resources/extensions/gsd/definition-loader.js +7 -0
- package/dist/resources/extensions/gsd/docs/preferences-reference.md +9 -9
- package/dist/resources/extensions/gsd/doctor-environment.js +2 -1
- package/dist/resources/extensions/gsd/doctor-git-checks.js +27 -3
- package/dist/resources/extensions/gsd/doctor-proactive.js +4 -1
- package/dist/resources/extensions/gsd/doctor-providers.js +48 -20
- package/dist/resources/extensions/gsd/doctor-runtime-checks.js +22 -4
- package/dist/resources/extensions/gsd/doctor.js +7 -1
- package/dist/resources/extensions/gsd/error-classifier.js +6 -3
- package/dist/resources/extensions/gsd/escalation.js +321 -0
- package/dist/resources/extensions/gsd/forensics.js +26 -29
- package/dist/resources/extensions/gsd/git-service.js +0 -1
- package/dist/resources/extensions/gsd/graph.js +26 -2
- package/dist/resources/extensions/gsd/gsd-db.js +490 -32
- package/dist/resources/extensions/gsd/health-widget-core.js +42 -14
- package/dist/resources/extensions/gsd/health-widget.js +7 -4
- package/dist/resources/extensions/gsd/init-wizard.js +86 -45
- package/dist/resources/extensions/gsd/markdown-renderer.js +5 -5
- package/dist/resources/extensions/gsd/memory-embeddings.js +219 -0
- package/dist/resources/extensions/gsd/memory-extractor.js +78 -27
- package/dist/resources/extensions/gsd/memory-ingest.js +218 -0
- package/dist/resources/extensions/gsd/memory-relations.js +189 -0
- package/dist/resources/extensions/gsd/memory-source-store.js +113 -0
- package/dist/resources/extensions/gsd/memory-store.js +299 -6
- package/dist/resources/extensions/gsd/metrics.js +1 -0
- package/dist/resources/extensions/gsd/model-cost-table.js +3 -1
- package/dist/resources/extensions/gsd/model-router.js +16 -6
- package/dist/resources/extensions/gsd/native-git-bridge.js +137 -5
- package/dist/resources/extensions/gsd/notification-overlay.js +7 -22
- package/dist/resources/extensions/gsd/notification-widget.js +24 -39
- package/dist/resources/extensions/gsd/notifications.js +4 -0
- package/dist/resources/extensions/gsd/onboarding-state.js +133 -0
- package/dist/resources/extensions/gsd/post-execution-checks.js +27 -11
- package/dist/resources/extensions/gsd/pre-execution-checks.js +105 -8
- package/dist/resources/extensions/gsd/preferences-models.js +1 -0
- package/dist/resources/extensions/gsd/preferences-types.js +2 -1
- package/dist/resources/extensions/gsd/preferences-validation.js +42 -8
- package/dist/resources/extensions/gsd/preferences.js +10 -10
- package/dist/resources/extensions/gsd/prompts/add-tests.md +1 -0
- package/dist/resources/extensions/gsd/prompts/complete-milestone.md +4 -1
- package/dist/resources/extensions/gsd/prompts/debug-diagnose.md +25 -0
- package/dist/resources/extensions/gsd/prompts/debug-session-manager.md +80 -0
- package/dist/resources/extensions/gsd/prompts/execute-task.md +13 -1
- package/dist/resources/extensions/gsd/prompts/plan-milestone.md +12 -0
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -0
- package/dist/resources/extensions/gsd/prompts/refine-slice.md +69 -0
- package/dist/resources/extensions/gsd/prompts/scan.md +79 -0
- package/dist/resources/extensions/gsd/prompts/workflow-oneshot.md +26 -0
- package/dist/resources/extensions/gsd/python-resolver.js +70 -0
- package/dist/resources/extensions/gsd/run-manager.js +37 -17
- package/dist/resources/extensions/gsd/setup-catalog.js +75 -0
- package/dist/resources/extensions/gsd/state.js +47 -3
- package/dist/resources/extensions/gsd/templates/PREFERENCES.md +7 -7
- package/dist/resources/extensions/gsd/tools/complete-task.js +80 -0
- package/dist/resources/extensions/gsd/tools/memory-tools.js +306 -0
- package/dist/resources/extensions/gsd/tools/plan-milestone.js +37 -12
- package/dist/resources/extensions/gsd/tools/plan-slice.js +5 -2
- package/dist/resources/extensions/gsd/tools/skip-slice.js +78 -0
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +14 -0
- package/dist/resources/extensions/gsd/uok/flags.js +7 -7
- package/dist/resources/extensions/gsd/uok/kernel.js +8 -3
- package/dist/resources/extensions/gsd/verification-gate.js +2 -1
- package/dist/resources/extensions/gsd/workflow-dispatch.js +64 -0
- package/dist/resources/extensions/gsd/workflow-install.js +327 -0
- package/dist/resources/extensions/gsd/workflow-manifest.js +8 -0
- package/dist/resources/extensions/gsd/workflow-mcp.js +1 -6
- package/dist/resources/extensions/gsd/workflow-plugins.js +346 -0
- package/dist/resources/extensions/gsd/workflow-projections.js +17 -15
- package/dist/resources/extensions/gsd/workflow-templates/accessibility-audit.md +88 -0
- package/dist/resources/extensions/gsd/workflow-templates/api-breaking-change.md +117 -0
- package/dist/resources/extensions/gsd/workflow-templates/bugfix.md +1 -0
- package/dist/resources/extensions/gsd/workflow-templates/changelog-gen.md +82 -0
- package/dist/resources/extensions/gsd/workflow-templates/ci-bootstrap.md +144 -0
- package/dist/resources/extensions/gsd/workflow-templates/dead-code.md +81 -0
- package/dist/resources/extensions/gsd/workflow-templates/dep-upgrade.md +1 -0
- package/dist/resources/extensions/gsd/workflow-templates/docs-sync.yaml +76 -0
- package/dist/resources/extensions/gsd/workflow-templates/env-audit.yaml +88 -0
- package/dist/resources/extensions/gsd/workflow-templates/full-project.md +1 -0
- package/dist/resources/extensions/gsd/workflow-templates/hotfix.md +1 -0
- package/dist/resources/extensions/gsd/workflow-templates/issue-triage.md +84 -0
- package/dist/resources/extensions/gsd/workflow-templates/observability-setup.md +133 -0
- package/dist/resources/extensions/gsd/workflow-templates/onboarding-check.md +74 -0
- package/dist/resources/extensions/gsd/workflow-templates/performance-audit.md +125 -0
- package/dist/resources/extensions/gsd/workflow-templates/pr-review.md +67 -0
- package/dist/resources/extensions/gsd/workflow-templates/pr-triage.md +83 -0
- package/dist/resources/extensions/gsd/workflow-templates/refactor.md +1 -0
- package/dist/resources/extensions/gsd/workflow-templates/registry.json +184 -0
- package/dist/resources/extensions/gsd/workflow-templates/release.md +118 -0
- package/dist/resources/extensions/gsd/workflow-templates/rename-symbol.yaml +99 -0
- package/dist/resources/extensions/gsd/workflow-templates/security-audit.md +1 -0
- package/dist/resources/extensions/gsd/workflow-templates/small-feature.md +1 -0
- package/dist/resources/extensions/gsd/workflow-templates/spike.md +1 -0
- package/dist/resources/extensions/gsd/workflow-templates/test-backfill.yaml +73 -0
- package/dist/resources/extensions/gsd/worktree-resolver.js +42 -1
- package/dist/resources/extensions/remote-questions/commands.js +380 -0
- package/dist/resources/extensions/remote-questions/manager.js +39 -5
- package/dist/resources/extensions/remote-questions/telegram-adapter.js +79 -4
- package/dist/resources/extensions/search-the-web/command-search-provider.js +4 -1
- package/dist/resources/extensions/search-the-web/native-search.js +13 -2
- package/dist/resources/extensions/shared/interview-ui.js +189 -1
- package/dist/resources/extensions/shared/layout-utils.js +17 -0
- package/dist/resources/extensions/shared/rtk-shared.js +47 -0
- package/dist/resources/extensions/shared/rtk.js +3 -46
- package/dist/resources/skills/create-workflow/SKILL.md +33 -6
- package/dist/rtk-shared.d.ts +10 -0
- package/dist/rtk-shared.js +47 -0
- package/dist/rtk.d.ts +2 -6
- package/dist/rtk.js +3 -48
- package/dist/shared/workspace-types.d.ts +52 -0
- package/dist/shared/workspace-types.js +1 -0
- package/dist/tsconfig.extensions.tsbuildinfo +1 -1
- package/dist/update-check.d.ts +10 -0
- package/dist/update-check.js +24 -3
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +13 -13
- package/dist/web/standalone/.next/build-manifest.json +3 -3
- package/dist/web/standalone/.next/react-loadable-manifest.json +2 -2
- package/dist/web/standalone/.next/required-server-files.json +1 -1
- package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +2 -2
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/notifications/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +2 -2
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +3 -3
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +3 -3
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +13 -13
- package/dist/web/standalone/.next/server/chunks/6897.js +2 -2
- package/dist/web/standalone/.next/server/chunks/7461.js +1 -0
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-manifest.json +1 -1
- package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/static/chunks/2826.e59e8578e2e28639.js +9 -0
- package/dist/web/standalone/.next/static/chunks/{2008.71ee9230ad78df21.js → 3621.fc7480022c972438.js} +2 -2
- package/dist/web/standalone/.next/static/chunks/app/{page-7115e62689b5fd84.js → page-151349214571e2b6.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/{webpack-b868033a5834586d.js → webpack-5fc74f13a25fa1bb.js} +1 -1
- package/dist/web/standalone/.next/static/css/632cd626b1731d88.css +1 -0
- package/dist/web/standalone/server.js +1 -1
- package/dist/welcome-screen.js +48 -24
- package/dist/wizard.js +2 -2
- package/dist/worktree-cli.d.ts +6 -5
- package/dist/worktree-cli.js +23 -7
- package/package.json +3 -3
- package/packages/daemon/package.json +2 -2
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +12 -10
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/session-manager.d.ts.map +1 -1
- package/packages/mcp-server/dist/session-manager.js +8 -1
- package/packages/mcp-server/dist/session-manager.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.d.ts +1 -0
- package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +207 -71
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/package.json +2 -2
- package/packages/mcp-server/src/mcp-server.test.ts +40 -4
- package/packages/mcp-server/src/server.ts +12 -10
- package/packages/mcp-server/src/session-manager.ts +10 -3
- package/packages/mcp-server/src/workflow-tools.test.ts +346 -1
- package/packages/mcp-server/src/workflow-tools.ts +228 -75
- package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
- package/packages/native/package.json +1 -1
- package/packages/native/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-agent-core/package.json +1 -1
- package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-ai/dist/index.d.ts +1 -0
- package/packages/pi-ai/dist/index.d.ts.map +1 -1
- package/packages/pi-ai/dist/index.js +1 -0
- package/packages/pi-ai/dist/index.js.map +1 -1
- package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/capability-patches.js +3 -2
- package/packages/pi-ai/dist/models/capability-patches.js.map +1 -1
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts +68 -0
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.js +68 -0
- package/packages/pi-ai/dist/models/generated/amazon-bedrock.js.map +1 -1
- package/packages/pi-ai/dist/models/generated/anthropic.d.ts +17 -0
- package/packages/pi-ai/dist/models/generated/anthropic.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/generated/anthropic.js +17 -0
- package/packages/pi-ai/dist/models/generated/anthropic.js.map +1 -1
- package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts +17 -0
- package/packages/pi-ai/dist/models/generated/google-antigravity.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/generated/google-antigravity.js +17 -0
- package/packages/pi-ai/dist/models/generated/google-antigravity.js.map +1 -1
- package/packages/pi-ai/dist/models/generated/groq.d.ts +0 -153
- package/packages/pi-ai/dist/models/generated/groq.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/generated/groq.js +0 -153
- package/packages/pi-ai/dist/models/generated/groq.js.map +1 -1
- package/packages/pi-ai/dist/models/generated/index.d.ts +136 -153
- package/packages/pi-ai/dist/models/generated/index.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +17 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/generated/openai-codex.js +17 -0
- package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -1
- package/packages/pi-ai/dist/models/generated/openrouter.d.ts +17 -0
- package/packages/pi-ai/dist/models/generated/openrouter.d.ts.map +1 -1
- package/packages/pi-ai/dist/models/generated/openrouter.js +17 -0
- package/packages/pi-ai/dist/models/generated/openrouter.js.map +1 -1
- package/packages/pi-ai/dist/models.generated.test.js +17 -0
- package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
- package/packages/pi-ai/dist/providers/amazon-bedrock.d.ts +22 -1
- package/packages/pi-ai/dist/providers/amazon-bedrock.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/amazon-bedrock.js +40 -6
- package/packages/pi-ai/dist/providers/amazon-bedrock.js.map +1 -1
- package/packages/pi-ai/dist/providers/amazon-bedrock.test.d.ts +2 -0
- package/packages/pi-ai/dist/providers/amazon-bedrock.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/amazon-bedrock.test.js +106 -0
- package/packages/pi-ai/dist/providers/amazon-bedrock.test.js.map +1 -0
- package/packages/pi-ai/dist/providers/anthropic-auth.test.js +42 -1
- package/packages/pi-ai/dist/providers/anthropic-auth.test.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.d.ts +20 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.js +32 -2
- package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.test.js +12 -1
- package/packages/pi-ai/dist/providers/anthropic-shared.test.js.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.d.ts +11 -0
- package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
- package/packages/pi-ai/dist/providers/anthropic.js +18 -1
- package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
- package/packages/pi-ai/dist/providers/api-family.d.ts +27 -0
- package/packages/pi-ai/dist/providers/api-family.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/api-family.js +47 -0
- package/packages/pi-ai/dist/providers/api-family.js.map +1 -0
- package/packages/pi-ai/dist/providers/api-family.test.d.ts +2 -0
- package/packages/pi-ai/dist/providers/api-family.test.d.ts.map +1 -0
- package/packages/pi-ai/dist/providers/api-family.test.js +101 -0
- package/packages/pi-ai/dist/providers/api-family.test.js.map +1 -0
- package/packages/pi-ai/dist/utils/oauth/openai-codex.d.ts.map +1 -1
- package/packages/pi-ai/dist/utils/oauth/openai-codex.js +12 -0
- package/packages/pi-ai/dist/utils/oauth/openai-codex.js.map +1 -1
- package/packages/pi-ai/package.json +2 -2
- package/packages/pi-ai/scripts/generate-models.ts +50 -0
- package/packages/pi-ai/src/index.ts +1 -0
- package/packages/pi-ai/src/models/capability-patches.ts +5 -2
- package/packages/pi-ai/src/models/generated/amazon-bedrock.ts +68 -0
- package/packages/pi-ai/src/models/generated/anthropic.ts +17 -0
- package/packages/pi-ai/src/models/generated/google-antigravity.ts +17 -0
- package/packages/pi-ai/src/models/generated/groq.ts +0 -153
- package/packages/pi-ai/src/models/generated/openai-codex.ts +17 -0
- package/packages/pi-ai/src/models/generated/openrouter.ts +17 -0
- package/packages/pi-ai/src/models.generated.test.ts +17 -0
- package/packages/pi-ai/src/providers/amazon-bedrock.test.ts +164 -0
- package/packages/pi-ai/src/providers/amazon-bedrock.ts +41 -7
- package/packages/pi-ai/src/providers/anthropic-auth.test.ts +47 -1
- package/packages/pi-ai/src/providers/anthropic-shared.test.ts +15 -1
- package/packages/pi-ai/src/providers/anthropic-shared.ts +36 -3
- package/packages/pi-ai/src/providers/anthropic.ts +19 -1
- package/packages/pi-ai/src/providers/api-family.test.ts +129 -0
- package/packages/pi-ai/src/providers/api-family.ts +57 -0
- package/packages/pi-ai/src/utils/oauth/openai-codex.ts +15 -0
- package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-coding-agent/dist/cli/args.d.ts +6 -0
- package/packages/pi-coding-agent/dist/cli/args.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/cli/args.js +14 -4
- package/packages/pi-coding-agent/dist/cli/args.js.map +1 -1
- package/packages/pi-coding-agent/dist/cli/args.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/cli/args.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/cli/args.test.js +38 -0
- package/packages/pi-coding-agent/dist/cli/args.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +38 -0
- package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +14 -0
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.js +34 -0
- package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.test.js +74 -0
- package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +4 -1
- package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +1 -0
- package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +2 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +32 -0
- package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.js +4 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.d.ts +10 -0
- package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.js +39 -1
- package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/skill-tool.test.js +2 -2
- package/packages/pi-coding-agent/dist/core/skill-tool.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/main.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/main.js +3 -0
- package/packages/pi-coding-agent/dist/main.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +17 -7
- package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +48 -34
- package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts +4 -0
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +83 -33
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/theme/theme-schema.d.ts +70 -0
- package/packages/pi-coding-agent/dist/modes/interactive/theme/theme-schema.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/theme/theme-schema.js +77 -0
- package/packages/pi-coding-agent/dist/modes/interactive/theme/theme-schema.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.d.ts +1 -66
- package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.js +1 -75
- package/packages/pi-coding-agent/dist/modes/interactive/theme/theme.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.d.ts +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.js +192 -24
- package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.js.map +1 -1
- package/packages/pi-coding-agent/package.json +1 -1
- package/packages/pi-coding-agent/src/cli/args.test.ts +44 -0
- package/packages/pi-coding-agent/src/cli/args.ts +21 -6
- package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +56 -0
- package/packages/pi-coding-agent/src/core/auth-storage.test.ts +83 -0
- package/packages/pi-coding-agent/src/core/auth-storage.ts +35 -0
- package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +4 -1
- package/packages/pi-coding-agent/src/core/extensions/runner.ts +4 -1
- package/packages/pi-coding-agent/src/core/extensions/types.ts +2 -2
- package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +37 -1
- package/packages/pi-coding-agent/src/core/retry-handler.ts +4 -1
- package/packages/pi-coding-agent/src/core/sdk.ts +58 -1
- package/packages/pi-coding-agent/src/core/skill-tool.test.ts +2 -2
- package/packages/pi-coding-agent/src/main.ts +4 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +19 -7
- package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +53 -31
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +88 -36
- package/packages/pi-coding-agent/src/modes/interactive/theme/theme-schema.ts +83 -0
- package/packages/pi-coding-agent/src/modes/interactive/theme/theme.ts +2 -83
- package/packages/pi-coding-agent/src/modes/interactive/theme/themes.ts +208 -27
- package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
- package/packages/pi-tui/package.json +1 -1
- package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
- package/packages/rpc-client/package.json +1 -1
- package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
- package/pkg/dist/modes/interactive/theme/theme-schema.d.ts +70 -0
- package/pkg/dist/modes/interactive/theme/theme-schema.d.ts.map +1 -0
- package/pkg/dist/modes/interactive/theme/theme-schema.js +77 -0
- package/pkg/dist/modes/interactive/theme/theme-schema.js.map +1 -0
- package/pkg/dist/modes/interactive/theme/theme.d.ts +1 -66
- package/pkg/dist/modes/interactive/theme/theme.d.ts.map +1 -1
- package/pkg/dist/modes/interactive/theme/theme.js +1 -75
- package/pkg/dist/modes/interactive/theme/theme.js.map +1 -1
- package/pkg/dist/modes/interactive/theme/themes.d.ts +1 -1
- package/pkg/dist/modes/interactive/theme/themes.d.ts.map +1 -1
- package/pkg/dist/modes/interactive/theme/themes.js +192 -24
- package/pkg/dist/modes/interactive/theme/themes.js.map +1 -1
- package/pkg/package.json +1 -1
- package/src/resources/extensions/ask-user-questions.ts +24 -6
- package/src/resources/extensions/claude-code-cli/models.ts +9 -0
- package/src/resources/extensions/claude-code-cli/readiness.ts +13 -2
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +94 -4
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +84 -0
- package/src/resources/extensions/gsd/auto/detect-stuck.ts +10 -0
- package/src/resources/extensions/gsd/auto/loop-deps.ts +2 -9
- package/src/resources/extensions/gsd/auto/loop.ts +109 -3
- package/src/resources/extensions/gsd/auto/phases.ts +97 -60
- package/src/resources/extensions/gsd/auto/resolve.ts +1 -1
- package/src/resources/extensions/gsd/auto/run-unit.ts +11 -1
- package/src/resources/extensions/gsd/auto/session.ts +7 -0
- package/src/resources/extensions/gsd/auto-artifact-paths.ts +20 -0
- package/src/resources/extensions/gsd/auto-dashboard.ts +46 -5
- package/src/resources/extensions/gsd/auto-direct-dispatch.ts +15 -2
- package/src/resources/extensions/gsd/auto-dispatch.ts +141 -9
- package/src/resources/extensions/gsd/auto-loop.ts +1 -1
- package/src/resources/extensions/gsd/auto-model-selection.ts +17 -4
- package/src/resources/extensions/gsd/auto-post-unit.ts +10 -8
- package/src/resources/extensions/gsd/auto-prompts.ts +232 -47
- package/src/resources/extensions/gsd/auto-recovery.ts +63 -1
- package/src/resources/extensions/gsd/auto-start.ts +8 -6
- package/src/resources/extensions/gsd/auto-verification.ts +3 -3
- package/src/resources/extensions/gsd/auto-worktree.ts +81 -1
- package/src/resources/extensions/gsd/auto.ts +61 -28
- package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +8 -21
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +46 -24
- package/src/resources/extensions/gsd/bootstrap/memory-tools.ts +158 -0
- package/src/resources/extensions/gsd/bootstrap/query-tools.ts +31 -0
- package/src/resources/extensions/gsd/bootstrap/register-extension.ts +2 -0
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +22 -0
- package/src/resources/extensions/gsd/bootstrap/system-context.ts +20 -4
- package/src/resources/extensions/gsd/commands/catalog.ts +74 -9
- package/src/resources/extensions/gsd/commands/handlers/core.ts +69 -27
- package/src/resources/extensions/gsd/commands/handlers/escalate.ts +216 -0
- package/src/resources/extensions/gsd/commands/handlers/onboarding.ts +196 -0
- package/src/resources/extensions/gsd/commands/handlers/ops.ts +21 -0
- package/src/resources/extensions/gsd/commands/handlers/workflow.ts +279 -29
- package/src/resources/extensions/gsd/commands-cmux.ts +6 -2
- package/src/resources/extensions/gsd/commands-config.ts +10 -0
- package/src/resources/extensions/gsd/commands-debug.ts +484 -0
- package/src/resources/extensions/gsd/commands-do.ts +1 -0
- package/src/resources/extensions/gsd/commands-extract-learnings.ts +295 -76
- package/src/resources/extensions/gsd/commands-handlers.ts +19 -2
- package/src/resources/extensions/gsd/commands-memory.ts +551 -0
- package/src/resources/extensions/gsd/commands-prefs-wizard.ts +49 -12
- package/src/resources/extensions/gsd/commands-scan.ts +125 -0
- package/src/resources/extensions/gsd/commands-workflow-templates.ts +129 -2
- package/src/resources/extensions/gsd/custom-workflow-engine.ts +85 -60
- package/src/resources/extensions/gsd/db-writer.ts +3 -0
- package/src/resources/extensions/gsd/debug-session-store.ts +377 -0
- package/src/resources/extensions/gsd/definition-loader.ts +7 -0
- package/src/resources/extensions/gsd/docs/preferences-reference.md +9 -9
- package/src/resources/extensions/gsd/doctor-environment.ts +2 -1
- package/src/resources/extensions/gsd/doctor-git-checks.ts +28 -3
- package/src/resources/extensions/gsd/doctor-proactive.ts +4 -1
- package/src/resources/extensions/gsd/doctor-providers.ts +52 -22
- package/src/resources/extensions/gsd/doctor-runtime-checks.ts +23 -4
- package/src/resources/extensions/gsd/doctor-types.ts +1 -0
- package/src/resources/extensions/gsd/doctor.ts +7 -1
- package/src/resources/extensions/gsd/error-classifier.ts +6 -3
- package/src/resources/extensions/gsd/escalation.ts +367 -0
- package/src/resources/extensions/gsd/forensics.ts +25 -29
- package/src/resources/extensions/gsd/git-service.ts +0 -1
- package/src/resources/extensions/gsd/graph.ts +33 -3
- package/src/resources/extensions/gsd/gsd-db.ts +578 -32
- package/src/resources/extensions/gsd/health-widget-core.ts +43 -14
- package/src/resources/extensions/gsd/health-widget.ts +7 -3
- package/src/resources/extensions/gsd/init-wizard.ts +87 -54
- package/src/resources/extensions/gsd/markdown-renderer.ts +5 -5
- package/src/resources/extensions/gsd/memory-embeddings.ts +235 -0
- package/src/resources/extensions/gsd/memory-extractor.ts +100 -34
- package/src/resources/extensions/gsd/memory-ingest.ts +286 -0
- package/src/resources/extensions/gsd/memory-relations.ts +240 -0
- package/src/resources/extensions/gsd/memory-source-store.ts +138 -0
- package/src/resources/extensions/gsd/memory-store.ts +351 -7
- package/src/resources/extensions/gsd/metrics.ts +1 -0
- package/src/resources/extensions/gsd/model-cost-table.ts +3 -1
- package/src/resources/extensions/gsd/model-router.ts +25 -6
- package/src/resources/extensions/gsd/native-git-bridge.ts +134 -6
- package/src/resources/extensions/gsd/notification-overlay.ts +9 -19
- package/src/resources/extensions/gsd/notification-widget.ts +25 -43
- package/src/resources/extensions/gsd/notifications.ts +6 -0
- package/src/resources/extensions/gsd/onboarding-state.ts +146 -0
- package/src/resources/extensions/gsd/post-execution-checks.ts +37 -14
- package/src/resources/extensions/gsd/pre-execution-checks.ts +106 -12
- package/src/resources/extensions/gsd/preferences-models.ts +1 -0
- package/src/resources/extensions/gsd/preferences-types.ts +10 -2
- package/src/resources/extensions/gsd/preferences-validation.ts +33 -7
- package/src/resources/extensions/gsd/preferences.ts +10 -10
- package/src/resources/extensions/gsd/prompts/add-tests.md +1 -0
- package/src/resources/extensions/gsd/prompts/complete-milestone.md +4 -1
- package/src/resources/extensions/gsd/prompts/debug-diagnose.md +25 -0
- package/src/resources/extensions/gsd/prompts/debug-session-manager.md +80 -0
- package/src/resources/extensions/gsd/prompts/execute-task.md +13 -1
- package/src/resources/extensions/gsd/prompts/plan-milestone.md +12 -0
- package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -0
- package/src/resources/extensions/gsd/prompts/refine-slice.md +69 -0
- package/src/resources/extensions/gsd/prompts/scan.md +79 -0
- package/src/resources/extensions/gsd/prompts/workflow-oneshot.md +26 -0
- package/src/resources/extensions/gsd/python-resolver.ts +76 -0
- package/src/resources/extensions/gsd/run-manager.ts +53 -19
- package/src/resources/extensions/gsd/setup-catalog.ts +105 -0
- package/src/resources/extensions/gsd/state.ts +50 -2
- package/src/resources/extensions/gsd/templates/PREFERENCES.md +7 -7
- package/src/resources/extensions/gsd/tests/agent-end-retry.test.ts +1 -34
- package/src/resources/extensions/gsd/tests/artifact-corruption-2630.test.ts +7 -0
- package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +49 -0
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +45 -31
- package/src/resources/extensions/gsd/tests/auto-migrating-recovery.test.ts +63 -0
- package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +12 -0
- package/src/resources/extensions/gsd/tests/auto-prompts-fallback.test.ts +35 -0
- package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +123 -1
- package/src/resources/extensions/gsd/tests/auto-wrapup-inflight-guard.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/autocomplete-regressions-1675.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/commands-do.test.ts +48 -0
- package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +333 -21
- package/src/resources/extensions/gsd/tests/commands-scan.test.ts +351 -0
- package/src/resources/extensions/gsd/tests/commands-workflow-custom.test.ts +8 -6
- package/src/resources/extensions/gsd/tests/complete-slice.test.ts +6 -8
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/custom-workflow-engine.test.ts +63 -0
- package/src/resources/extensions/gsd/tests/debug-command-handler.test.ts +905 -0
- package/src/resources/extensions/gsd/tests/debug-command-lifecycle.integration.test.ts +1229 -0
- package/src/resources/extensions/gsd/tests/debug-session-store.test.ts +565 -0
- package/src/resources/extensions/gsd/tests/discuss-milestone-structured-questions.test.ts +64 -0
- package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +67 -0
- package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +62 -18
- package/src/resources/extensions/gsd/tests/enhanced-verification-integration.test.ts +5 -0
- package/src/resources/extensions/gsd/tests/escalation.test.ts +818 -0
- package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +29 -12
- package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +106 -0
- package/src/resources/extensions/gsd/tests/forensics-hook-key-parse.test.ts +74 -0
- package/src/resources/extensions/gsd/tests/graph-operations.test.ts +0 -4
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +44 -1
- package/src/resources/extensions/gsd/tests/health-widget.test.ts +8 -2
- package/src/resources/extensions/gsd/tests/init-prefs-routing.test.ts +190 -0
- package/src/resources/extensions/gsd/tests/integration/doctor-git-symlink-cwd.test.ts +79 -0
- package/src/resources/extensions/gsd/tests/integration/doctor-git.test.ts +39 -0
- package/src/resources/extensions/gsd/tests/integration/doctor-proactive.test.ts +44 -0
- package/src/resources/extensions/gsd/tests/integration/doctor-runtime.test.ts +68 -1
- package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +109 -11
- package/src/resources/extensions/gsd/tests/integration/idle-recovery.test.ts +51 -0
- package/src/resources/extensions/gsd/tests/integration/integration-proof.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/integration/test-isolation.ts +53 -0
- package/src/resources/extensions/gsd/tests/journal-integration.test.ts +171 -1
- package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/memory-embeddings.test.ts +213 -0
- package/src/resources/extensions/gsd/tests/memory-ingest.test.ts +153 -0
- package/src/resources/extensions/gsd/tests/memory-maintenance.test.ts +107 -0
- package/src/resources/extensions/gsd/tests/memory-relations.test.ts +175 -0
- package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/memory-tools.test.ts +295 -0
- package/src/resources/extensions/gsd/tests/milestone-status-tool.test.ts +3 -2
- package/src/resources/extensions/gsd/tests/model-cost-table.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/model-router.test.ts +51 -1
- package/src/resources/extensions/gsd/tests/model-unittype-mapping.test.ts +59 -0
- package/src/resources/extensions/gsd/tests/notification-overlay.test.ts +56 -37
- package/src/resources/extensions/gsd/tests/notification-widget.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/onboarding-state.test.ts +105 -0
- package/src/resources/extensions/gsd/tests/plan-milestone-boundary-map-preservation.test.ts +114 -0
- package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +4 -5
- package/src/resources/extensions/gsd/tests/plan-slice.test.ts +17 -0
- package/src/resources/extensions/gsd/tests/post-execution-checks.test.ts +105 -1
- package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +341 -6
- package/src/resources/extensions/gsd/tests/preferences.test.ts +69 -1
- package/src/resources/extensions/gsd/tests/progressive-planning.test.ts +539 -0
- package/src/resources/extensions/gsd/tests/projection-no-plan-overwrite.test.ts +11 -2
- package/src/resources/extensions/gsd/tests/projection-regression.test.ts +7 -0
- package/src/resources/extensions/gsd/tests/prompt-budget-enforcement.test.ts +159 -8
- package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +24 -0
- package/src/resources/extensions/gsd/tests/prompt-db.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/prompts-no-gitignored-test-refs.test.ts +56 -0
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +33 -0
- package/src/resources/extensions/gsd/tests/python-resolver.test.ts +131 -0
- package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +67 -0
- package/src/resources/extensions/gsd/tests/remote-notification-from-desktop.test.ts +107 -0
- package/src/resources/extensions/gsd/tests/requirements.test.ts +9 -0
- package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +153 -0
- package/src/resources/extensions/gsd/tests/skip-slice-cascades-tasks.test.ts +125 -0
- package/src/resources/extensions/gsd/tests/slice-context-injection.test.ts +16 -4
- package/src/resources/extensions/gsd/tests/stash-queued-context-files.test.ts +54 -0
- package/src/resources/extensions/gsd/tests/state-corruption-2945.test.ts +3 -0
- package/src/resources/extensions/gsd/tests/stuck-detection-coverage.test.ts +15 -0
- package/src/resources/extensions/gsd/tests/summary-render-parity.test.ts +5 -0
- package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/uok-flags.test.ts +31 -1
- package/src/resources/extensions/gsd/tests/uok-kernel-path.test.ts +166 -0
- package/src/resources/extensions/gsd/tests/verification-gate.test.ts +35 -0
- package/src/resources/extensions/gsd/tests/workflow-install.test.ts +113 -0
- package/src/resources/extensions/gsd/tests/workflow-logger-wiring.test.ts +15 -6
- package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/workflow-plugins.test.ts +310 -0
- package/src/resources/extensions/gsd/tests/workflow-projections.test.ts +7 -0
- package/src/resources/extensions/gsd/tests/workflow-templates.test.ts +8 -2
- package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +97 -0
- package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +77 -2
- package/src/resources/extensions/gsd/tools/complete-task.ts +87 -0
- package/src/resources/extensions/gsd/tools/memory-tools.ts +380 -0
- package/src/resources/extensions/gsd/tools/plan-milestone.ts +42 -8
- package/src/resources/extensions/gsd/tools/plan-slice.ts +6 -1
- package/src/resources/extensions/gsd/tools/skip-slice.ts +133 -0
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +14 -0
- package/src/resources/extensions/gsd/types.ts +62 -0
- package/src/resources/extensions/gsd/unit-runtime.ts +1 -0
- package/src/resources/extensions/gsd/uok/contracts.ts +2 -1
- package/src/resources/extensions/gsd/uok/flags.ts +7 -7
- package/src/resources/extensions/gsd/uok/kernel.ts +16 -4
- package/src/resources/extensions/gsd/verification-gate.ts +2 -1
- package/src/resources/extensions/gsd/workflow-dispatch.ts +106 -0
- package/src/resources/extensions/gsd/workflow-install.ts +423 -0
- package/src/resources/extensions/gsd/workflow-logger.ts +3 -1
- package/src/resources/extensions/gsd/workflow-manifest.ts +8 -0
- package/src/resources/extensions/gsd/workflow-mcp.ts +1 -6
- package/src/resources/extensions/gsd/workflow-plugins.ts +403 -0
- package/src/resources/extensions/gsd/workflow-projections.ts +18 -16
- package/src/resources/extensions/gsd/workflow-templates/accessibility-audit.md +88 -0
- package/src/resources/extensions/gsd/workflow-templates/api-breaking-change.md +117 -0
- package/src/resources/extensions/gsd/workflow-templates/bugfix.md +1 -0
- package/src/resources/extensions/gsd/workflow-templates/changelog-gen.md +82 -0
- package/src/resources/extensions/gsd/workflow-templates/ci-bootstrap.md +144 -0
- package/src/resources/extensions/gsd/workflow-templates/dead-code.md +81 -0
- package/src/resources/extensions/gsd/workflow-templates/dep-upgrade.md +1 -0
- package/src/resources/extensions/gsd/workflow-templates/docs-sync.yaml +76 -0
- package/src/resources/extensions/gsd/workflow-templates/env-audit.yaml +88 -0
- package/src/resources/extensions/gsd/workflow-templates/full-project.md +1 -0
- package/src/resources/extensions/gsd/workflow-templates/hotfix.md +1 -0
- package/src/resources/extensions/gsd/workflow-templates/issue-triage.md +84 -0
- package/src/resources/extensions/gsd/workflow-templates/observability-setup.md +133 -0
- package/src/resources/extensions/gsd/workflow-templates/onboarding-check.md +74 -0
- package/src/resources/extensions/gsd/workflow-templates/performance-audit.md +125 -0
- package/src/resources/extensions/gsd/workflow-templates/pr-review.md +67 -0
- package/src/resources/extensions/gsd/workflow-templates/pr-triage.md +83 -0
- package/src/resources/extensions/gsd/workflow-templates/refactor.md +1 -0
- package/src/resources/extensions/gsd/workflow-templates/registry.json +184 -0
- package/src/resources/extensions/gsd/workflow-templates/release.md +118 -0
- package/src/resources/extensions/gsd/workflow-templates/rename-symbol.yaml +99 -0
- package/src/resources/extensions/gsd/workflow-templates/security-audit.md +1 -0
- package/src/resources/extensions/gsd/workflow-templates/small-feature.md +1 -0
- package/src/resources/extensions/gsd/workflow-templates/spike.md +1 -0
- package/src/resources/extensions/gsd/workflow-templates/test-backfill.yaml +73 -0
- package/src/resources/extensions/gsd/workflow-templates.ts +7 -0
- package/src/resources/extensions/gsd/workspace-index.ts +9 -4
- package/src/resources/extensions/gsd/worktree-resolver.ts +47 -1
- package/src/resources/extensions/remote-questions/commands.ts +480 -0
- package/src/resources/extensions/remote-questions/manager.ts +49 -4
- package/src/resources/extensions/remote-questions/telegram-adapter.ts +86 -4
- package/src/resources/extensions/remote-questions/tests/command-polling.test.ts +246 -0
- package/src/resources/extensions/remote-questions/tests/remote-answer-normalization.test.ts +92 -0
- package/src/resources/extensions/remote-questions/tests/telegram-commands.test.ts +267 -0
- package/src/resources/extensions/search-the-web/command-search-provider.ts +4 -1
- package/src/resources/extensions/search-the-web/native-search.ts +13 -3
- package/src/resources/extensions/shared/interview-ui.ts +195 -1
- package/src/resources/extensions/shared/layout-utils.ts +26 -0
- package/src/resources/extensions/shared/rtk-shared.ts +58 -0
- package/src/resources/extensions/shared/rtk.ts +12 -52
- package/src/resources/extensions/shared/tests/interview-preview.test.ts +177 -0
- package/src/resources/extensions/shared/tests/preview-layout.test.ts +120 -0
- package/src/resources/skills/create-workflow/SKILL.md +33 -6
- package/dist/web/standalone/.next/static/chunks/2826.dd3dc8bbd3025fa5.js +0 -9
- package/dist/web/standalone/.next/static/css/f6e8833d46e738d8.css +0 -1
- package/packages/native/dist/ps/types.d.ts +0 -5
- package/packages/native/dist/ps/types.js +0 -2
- package/packages/native/src/ps/types.ts +0 -5
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/LICENSE +0 -201
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/README.md +0 -9
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-cjs/index.js +0 -762
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/build-abort-error.js +0 -19
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/constants.js +0 -1
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/get-transformed-headers.js +0 -9
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/index.js +0 -3
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/node-http-handler.js +0 -230
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/node-http2-connection-manager.js +0 -87
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/node-http2-connection-pool.js +0 -32
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/node-http2-handler.js +0 -169
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/readable.mock.js +0 -21
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/server.mock.js +0 -88
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/set-connection-timeout.js +0 -36
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/set-request-timeout.js +0 -21
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/set-socket-keep-alive.js +0 -22
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/set-socket-timeout.js +0 -23
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/stream-collector/collector.js +0 -8
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/stream-collector/index.js +0 -41
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/stream-collector/readable.mock.js +0 -21
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/timing.js +0 -4
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-es/write-request-body.js +0 -63
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/build-abort-error.d.ts +0 -10
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/constants.d.ts +0 -5
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/get-transformed-headers.d.ts +0 -4
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/index.d.ts +0 -3
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/node-http-handler.d.ts +0 -46
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/node-http2-connection-manager.d.ts +0 -24
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/node-http2-connection-pool.d.ts +0 -12
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/node-http2-handler.d.ts +0 -63
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/readable.mock.d.ts +0 -13
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/server.mock.d.ts +0 -12
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/set-connection-timeout.d.ts +0 -2
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/set-request-timeout.d.ts +0 -6
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/set-socket-keep-alive.d.ts +0 -6
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/set-socket-timeout.d.ts +0 -2
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/stream-collector/collector.d.ts +0 -5
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/stream-collector/index.d.ts +0 -6
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/stream-collector/readable.mock.d.ts +0 -13
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/timing.d.ts +0 -8
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/build-abort-error.d.ts +0 -10
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/constants.d.ts +0 -5
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/get-transformed-headers.d.ts +0 -4
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/index.d.ts +0 -3
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/node-http-handler.d.ts +0 -46
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/node-http2-connection-manager.d.ts +0 -24
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/node-http2-connection-pool.d.ts +0 -12
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/node-http2-handler.d.ts +0 -63
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/readable.mock.d.ts +0 -13
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/server.mock.d.ts +0 -12
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/set-connection-timeout.d.ts +0 -2
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/set-request-timeout.d.ts +0 -6
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/set-socket-keep-alive.d.ts +0 -6
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/set-socket-timeout.d.ts +0 -2
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/stream-collector/collector.d.ts +0 -5
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/stream-collector/index.d.ts +0 -6
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/stream-collector/readable.mock.d.ts +0 -13
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/timing.d.ts +0 -8
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/ts3.4/write-request-body.d.ts +0 -12
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/dist-types/write-request-body.d.ts +0 -12
- package/packages/pi-ai/node_modules/@smithy/node-http-handler/package.json +0 -68
- package/packages/pi-ai/oauth.d.ts +0 -1
- package/packages/pi-ai/oauth.js +0 -1
- /package/dist/web/standalone/.next/static/{prkokVQFxWtUVIku57_0z → ssX7BLv3Dw9Fb4CtrCGeR}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{prkokVQFxWtUVIku57_0z → ssX7BLv3Dw9Fb4CtrCGeR}/_ssgManifest.js +0 -0
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/core/extensions/types.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AA21BH,MAAM,UAAU,qBAAqB,CAAC,QAAgB,EAAE,KAAsB;IAC7E,OAAO,KAAK,CAAC,QAAQ,KAAK,QAAQ,CAAC;AACpC,CAAC;AAiCD,MAAM,UAAU,mBAAmB,CAAC,QAAgB,EAAE,KAAoB;IACzE,OAAO,KAAK,CAAC,QAAQ,KAAK,QAAQ,CAAC;AACpC,CAAC","sourcesContent":["/**\n * Extension system types.\n *\n * Extensions are TypeScript modules that can:\n * - Subscribe to agent lifecycle events\n * - Register LLM-callable tools\n * - Register commands, keyboard shortcuts, and CLI flags\n * - Interact with the user via UI primitives\n */\n\nimport type {\n\tAgentMessage,\n\tAgentToolResult,\n\tAgentToolUpdateCallback,\n\tThinkingLevel,\n} from \"@gsd/pi-agent-core\";\nimport type {\n\tApi,\n\tAssistantMessageEvent,\n\tAssistantMessageEventStream,\n\tContext,\n\tImageContent,\n\tModel,\n\tOAuthCredentials,\n\tOAuthLoginCallbacks,\n\tSimpleStreamOptions,\n\tTextContent,\n\tToolResultMessage,\n} from \"@gsd/pi-ai\";\nimport type {\n\tAutocompleteItem,\n\tComponent,\n\tEditorComponent,\n\tEditorTheme,\n\tKeyId,\n\tOverlayHandle,\n\tOverlayOptions,\n\tTUI,\n} from \"@gsd/pi-tui\";\nimport type { Static, TSchema } from \"@sinclair/typebox\";\nimport type { Theme } from \"../../modes/interactive/theme/theme.js\";\nimport type { BashResult } from \"../bash-executor.js\";\nimport type { CompactionPreparation, CompactionResult } from \"../compaction/index.js\";\nimport type { EventBus } from \"../event-bus.js\";\nimport type { ExecOptions, ExecResult } from \"../exec.js\";\nimport type { ReadonlyFooterDataProvider } from \"../footer-data-provider.js\";\nimport type { KeybindingsManager } from \"../keybindings.js\";\nimport type { CustomMessage } from \"../messages.js\";\nimport type { ModelRegistry } from \"../model-registry.js\";\nimport type {\n\tBranchSummaryEntry,\n\tCompactionEntry,\n\tReadonlySessionManager,\n\tSessionEntry,\n\tSessionManager,\n} from \"../session-manager.js\";\nimport type { SlashCommandInfo } from \"../slash-commands.js\";\nimport type { BashOperations } from \"../tools/bash.js\";\nimport type { EditToolDetails } from \"../tools/edit.js\";\nimport type {\n\tBashToolDetails,\n\tBashToolInput,\n\tEditToolInput,\n\tFindToolDetails,\n\tFindToolInput,\n\tGrepToolDetails,\n\tGrepToolInput,\n\tLsToolDetails,\n\tLsToolInput,\n\tReadToolDetails,\n\tReadToolInput,\n\tWriteToolInput,\n} from \"../tools/index.js\";\n\nexport type { ExecOptions, ExecResult } from \"../exec.js\";\nexport type { AgentToolResult, AgentToolUpdateCallback };\nexport type { AppAction, KeybindingsManager } from \"../keybindings.js\";\n\n// ============================================================================\n// UI Context\n// ============================================================================\n\n/** Options for extension UI dialogs. */\nexport interface ExtensionUIDialogOptions {\n\t/** AbortSignal to programmatically dismiss the dialog. */\n\tsignal?: AbortSignal;\n\t/** Timeout in milliseconds. Dialog auto-dismisses with live countdown display. */\n\ttimeout?: number;\n\t/** When true, the user can select multiple options. The return type becomes `string[]`. */\n\tallowMultiple?: boolean;\n\t/** When true, text input dialogs should hide typed characters if supported by the client surface. */\n\tsecure?: boolean;\n}\n\n/** Placement for extension widgets. */\nexport type WidgetPlacement = \"aboveEditor\" | \"belowEditor\";\n\n/** Options for extension widgets. */\nexport interface ExtensionWidgetOptions {\n\t/** Where the widget is rendered. Defaults to \"aboveEditor\". */\n\tplacement?: WidgetPlacement;\n}\n\n/** Raw terminal input listener for extensions. */\nexport type TerminalInputHandler = (data: string) => { consume?: boolean; data?: string } | undefined;\n\n/**\n * UI context for extensions to request interactive UI.\n * Each mode (interactive, RPC, print) provides its own implementation.\n */\nexport interface ExtensionUIContext {\n\t/** Show a selector and return the user's choice. When `opts.allowMultiple` is true, returns an array. */\n\tselect(title: string, options: string[], opts?: ExtensionUIDialogOptions): Promise<string | string[] | undefined>;\n\n\t/** Show a confirmation dialog. */\n\tconfirm(title: string, message: string, opts?: ExtensionUIDialogOptions): Promise<boolean>;\n\n\t/** Show a text input dialog. */\n\tinput(title: string, placeholder?: string, opts?: ExtensionUIDialogOptions): Promise<string | undefined>;\n\n\t/** Show a notification to the user. */\n\tnotify(message: string, type?: \"info\" | \"warning\" | \"error\" | \"success\"): void;\n\n\t/** Listen to raw terminal input (interactive mode only). Returns an unsubscribe function. */\n\tonTerminalInput(handler: TerminalInputHandler): () => void;\n\n\t/** Set status text in the footer/status bar. Pass undefined to clear. */\n\tsetStatus(key: string, text: string | undefined): void;\n\n\t/** Set the working/loading message shown during streaming. Call with no argument to restore default. */\n\tsetWorkingMessage(message?: string): void;\n\n\t/** Set a widget to display above or below the editor. Accepts string array or component factory. */\n\tsetWidget(key: string, content: string[] | undefined, options?: ExtensionWidgetOptions): void;\n\tsetWidget(\n\t\tkey: string,\n\t\tcontent: ((tui: TUI, theme: Theme) => Component & { dispose?(): void }) | undefined,\n\t\toptions?: ExtensionWidgetOptions,\n\t): void;\n\n\t/** Set a custom footer component, or undefined to restore the built-in footer.\n\t *\n\t * The factory receives a FooterDataProvider for data not otherwise accessible:\n\t * git branch and extension statuses from setStatus(). Token stats, model info,\n\t * etc. are available via ctx.sessionManager and ctx.model.\n\t */\n\tsetFooter(\n\t\tfactory:\n\t\t\t| ((tui: TUI, theme: Theme, footerData: ReadonlyFooterDataProvider) => Component & { dispose?(): void })\n\t\t\t| undefined,\n\t): void;\n\n\t/** Set a custom header component (shown at startup, above chat), or undefined to restore the built-in header. */\n\tsetHeader(factory: ((tui: TUI, theme: Theme) => Component & { dispose?(): void }) | undefined): void;\n\n\t/** Set the terminal window/tab title. */\n\tsetTitle(title: string): void;\n\n\t/** Show a custom component with keyboard focus. */\n\tcustom<T>(\n\t\tfactory: (\n\t\t\ttui: TUI,\n\t\t\ttheme: Theme,\n\t\t\tkeybindings: KeybindingsManager,\n\t\t\tdone: (result: T) => void,\n\t\t) => (Component & { dispose?(): void }) | Promise<Component & { dispose?(): void }>,\n\t\toptions?: {\n\t\t\toverlay?: boolean;\n\t\t\t/** Overlay positioning/sizing options. Can be static or a function for dynamic updates. */\n\t\t\toverlayOptions?: OverlayOptions | (() => OverlayOptions);\n\t\t\t/** Called with the overlay handle after the overlay is shown. Use to control visibility. */\n\t\t\tonHandle?: (handle: OverlayHandle) => void;\n\t\t},\n\t): Promise<T>;\n\n\t/** Paste text into the editor, triggering paste handling (collapse for large content). */\n\tpasteToEditor(text: string): void;\n\n\t/** Set the text in the core input editor. */\n\tsetEditorText(text: string): void;\n\n\t/** Get the current text from the core input editor. */\n\tgetEditorText(): string;\n\n\t/** Show a multi-line editor for text editing. */\n\teditor(title: string, prefill?: string): Promise<string | undefined>;\n\n\t/**\n\t * Set a custom editor component via factory function.\n\t * Pass undefined to restore the default editor.\n\t *\n\t * The factory receives:\n\t * - `theme`: EditorTheme for styling borders and autocomplete\n\t * - `keybindings`: KeybindingsManager for app-level keybindings\n\t *\n\t * For full app keybinding support (escape, ctrl+d, model switching, etc.),\n\t * extend `CustomEditor` from `@gsd/pi-coding-agent` and call\n\t * `super.handleInput(data)` for keys you don't handle.\n\t *\n\t * @example\n\t * ```ts\n\t * import { CustomEditor } from \"@gsd/pi-coding-agent\";\n\t *\n\t * class VimEditor extends CustomEditor {\n\t * private mode: \"normal\" | \"insert\" = \"insert\";\n\t *\n\t * handleInput(data: string): void {\n\t * if (this.mode === \"normal\") {\n\t * // Handle vim normal mode keys...\n\t * if (data === \"i\") { this.mode = \"insert\"; return; }\n\t * }\n\t * super.handleInput(data); // App keybindings + text editing\n\t * }\n\t * }\n\t *\n\t * ctx.ui.setEditorComponent((tui, theme, keybindings) =>\n\t * new VimEditor(tui, theme, keybindings)\n\t * );\n\t * ```\n\t */\n\tsetEditorComponent(\n\t\tfactory: ((tui: TUI, theme: EditorTheme, keybindings: KeybindingsManager) => EditorComponent) | undefined,\n\t): void;\n\n\t/** Get the current theme for styling. */\n\treadonly theme: Theme;\n\n\t/** Get all available themes with their names and file paths. */\n\tgetAllThemes(): { name: string; path: string | undefined }[];\n\n\t/** Load a theme by name without switching to it. Returns undefined if not found. */\n\tgetTheme(name: string): Theme | undefined;\n\n\t/** Set the current theme by name or Theme object. */\n\tsetTheme(theme: string | Theme): { success: boolean; error?: string };\n\n\t/** Get current tool output expansion state. */\n\tgetToolsExpanded(): boolean;\n\n\t/** Set tool output expansion state. */\n\tsetToolsExpanded(expanded: boolean): void;\n}\n\n// ============================================================================\n// Extension Context\n// ============================================================================\n\nexport interface ContextUsage {\n\t/** Estimated context tokens, or null if unknown (e.g. right after compaction, before next LLM response). */\n\ttokens: number | null;\n\tcontextWindow: number;\n\t/** Context usage as percentage of context window, or null if tokens is unknown. */\n\tpercent: number | null;\n}\n\nexport interface CompactOptions {\n\tcustomInstructions?: string;\n\tonComplete?: (result: CompactionResult) => void;\n\tonError?: (error: Error) => void;\n}\n\n/**\n * Context passed to extension event handlers.\n */\nexport interface ExtensionContext {\n\t/** UI methods for user interaction */\n\tui: ExtensionUIContext;\n\t/** Whether UI is available (false in print/RPC mode) */\n\thasUI: boolean;\n\t/** Current working directory */\n\tcwd: string;\n\t/** Session manager (read-only) */\n\tsessionManager: ReadonlySessionManager;\n\t/** Model registry for API key resolution */\n\tmodelRegistry: ModelRegistry;\n\t/** Current model (may be undefined) */\n\tmodel: Model<any> | undefined;\n\t/** Whether the agent is idle (not streaming) */\n\tisIdle(): boolean;\n\t/** Abort the current agent operation */\n\tabort(): void;\n\t/** Whether there are queued messages waiting */\n\thasPendingMessages(): boolean;\n\t/** Gracefully shutdown pi and exit. Available in all contexts. */\n\tshutdown(): void;\n\t/** Get current context usage for the active model. */\n\tgetContextUsage(): ContextUsage | undefined;\n\t/** Trigger compaction without awaiting completion. */\n\tcompact(options?: CompactOptions): void;\n\t/** Get the current effective system prompt. */\n\tgetSystemPrompt(): string;\n}\n\n/**\n * Extended context for command handlers.\n * Includes session control methods only safe in user-initiated commands.\n */\nexport interface ExtensionCommandContext extends ExtensionContext {\n\t/** Wait for the agent to finish streaming */\n\twaitForIdle(): Promise<void>;\n\n\t/** Start a new session, optionally with initialization. */\n\tnewSession(options?: {\n\t\tparentSession?: string;\n\t\tsetup?: (sessionManager: SessionManager) => Promise<void>;\n\t}): Promise<{ cancelled: boolean }>;\n\n\t/** Fork from a specific entry, creating a new session file. */\n\tfork(entryId: string): Promise<{ cancelled: boolean }>;\n\n\t/** Navigate to a different point in the session tree. */\n\tnavigateTree(\n\t\ttargetId: string,\n\t\toptions?: { summarize?: boolean; customInstructions?: string; replaceInstructions?: boolean; label?: string },\n\t): Promise<{ cancelled: boolean }>;\n\n\t/** Switch to a different session file. */\n\tswitchSession(sessionPath: string): Promise<{ cancelled: boolean }>;\n\n\t/** Reload extensions, skills, prompts, and themes. */\n\treload(): Promise<void>;\n}\n\n// ============================================================================\n// Tool Types\n// ============================================================================\n\n/** Rendering options for tool results */\nexport interface ToolRenderResultOptions {\n\t/** Whether the result view is expanded */\n\texpanded: boolean;\n\t/** Whether this is a partial/streaming result */\n\tisPartial: boolean;\n}\n\n/**\n * Tool compatibility metadata for provider-aware tool filtering (ADR-005 Phase 2).\n * Tools without compatibility metadata are assumed universally compatible.\n */\nexport interface ToolCompatibility {\n\t/** Tool produces image content in results (filtered for providers without imageToolResults) */\n\tproducesImages?: boolean;\n\t/** Tool requires schema features that some providers don't support (e.g., [\"patternProperties\"]) */\n\tschemaFeatures?: string[];\n\t/** Tool is effective only with models above a minimum capability threshold */\n\tminCapabilityTier?: \"light\" | \"standard\" | \"heavy\";\n}\n\n/**\n * Tool definition for registerTool().\n */\nexport interface ToolDefinition<TParams extends TSchema = TSchema, TDetails = unknown> {\n\t/** Tool name (used in LLM tool calls) */\n\tname: string;\n\t/** Human-readable label for UI */\n\tlabel: string;\n\t/** Description for LLM */\n\tdescription: string;\n\t/** Optional one-line snippet for the Available tools section in the default system prompt. Falls back to description when omitted. */\n\tpromptSnippet?: string;\n\t/** Optional guideline bullets appended to the default system prompt Guidelines section when this tool is active. */\n\tpromptGuidelines?: string[];\n\t/** Parameter schema (TypeBox) */\n\tparameters: TParams;\n\t/** Provider compatibility metadata (ADR-005). Omit for universally compatible tools. */\n\tcompatibility?: ToolCompatibility;\n\n\t/** Execute the tool. */\n\texecute(\n\t\ttoolCallId: string,\n\t\tparams: Static<TParams>,\n\t\tsignal: AbortSignal | undefined,\n\t\tonUpdate: AgentToolUpdateCallback<TDetails> | undefined,\n\t\tctx: ExtensionContext,\n\t): Promise<AgentToolResult<TDetails>>;\n\n\t/** Custom rendering for tool call display */\n\trenderCall?: (args: Static<TParams>, theme: Theme) => Component | undefined;\n\n\t/** Custom rendering for tool result display */\n\trenderResult?: (\n\t\tresult: AgentToolResult<TDetails>,\n\t\toptions: ToolRenderResultOptions,\n\t\ttheme: Theme,\n\t) => Component | undefined;\n}\n\n// ============================================================================\n// Resource Events\n// ============================================================================\n\n/** Fired after session_start to allow extensions to provide additional resource paths. */\nexport interface ResourcesDiscoverEvent {\n\ttype: \"resources_discover\";\n\tcwd: string;\n\treason: \"startup\" | \"reload\";\n}\n\n/** Result from resources_discover event handler */\nexport interface ResourcesDiscoverResult {\n\tskillPaths?: string[];\n\tpromptPaths?: string[];\n\tthemePaths?: string[];\n}\n\n// ============================================================================\n// Session Events\n// ============================================================================\n\n/** Fired before session manager creation to allow custom session directory resolution */\nexport interface SessionDirectoryEvent {\n\ttype: \"session_directory\";\n\tcwd: string;\n}\n\n/** Fired on initial session load */\nexport interface SessionStartEvent {\n\ttype: \"session_start\";\n}\n\n/** Fired before switching to another session (can be cancelled) */\nexport interface SessionBeforeSwitchEvent {\n\ttype: \"session_before_switch\";\n\treason: \"new\" | \"resume\";\n\ttargetSessionFile?: string;\n}\n\n/** Fired after switching to another session */\nexport interface SessionSwitchEvent {\n\ttype: \"session_switch\";\n\treason: \"new\" | \"resume\";\n\tpreviousSessionFile: string | undefined;\n}\n\n/** Fired before forking a session (can be cancelled) */\nexport interface SessionBeforeForkEvent {\n\ttype: \"session_before_fork\";\n\tentryId: string;\n}\n\n/** Fired after forking a session */\nexport interface SessionForkEvent {\n\ttype: \"session_fork\";\n\tpreviousSessionFile: string | undefined;\n}\n\n/** Fired before context compaction (can be cancelled or customized) */\nexport interface SessionBeforeCompactEvent {\n\ttype: \"session_before_compact\";\n\tpreparation: CompactionPreparation;\n\tbranchEntries: SessionEntry[];\n\tcustomInstructions?: string;\n\tsignal: AbortSignal;\n}\n\n/** Fired after context compaction */\nexport interface SessionCompactEvent {\n\ttype: \"session_compact\";\n\tcompactionEntry: CompactionEntry;\n\tfromExtension: boolean;\n}\n\n/** Fired on process exit */\nexport interface SessionShutdownEvent {\n\ttype: \"session_shutdown\";\n}\n\n/** Preparation data for tree navigation */\nexport interface TreePreparation {\n\ttargetId: string;\n\toldLeafId: string | null;\n\tcommonAncestorId: string | null;\n\tentriesToSummarize: SessionEntry[];\n\tuserWantsSummary: boolean;\n\t/** Custom instructions for summarization */\n\tcustomInstructions?: string;\n\t/** If true, customInstructions replaces the default prompt instead of being appended */\n\treplaceInstructions?: boolean;\n\t/** Label to attach to the branch summary entry */\n\tlabel?: string;\n}\n\n/** Fired before navigating in the session tree (can be cancelled) */\nexport interface SessionBeforeTreeEvent {\n\ttype: \"session_before_tree\";\n\tpreparation: TreePreparation;\n\tsignal: AbortSignal;\n}\n\n/** Fired after navigating in the session tree */\nexport interface SessionTreeEvent {\n\ttype: \"session_tree\";\n\tnewLeafId: string | null;\n\toldLeafId: string | null;\n\tsummaryEntry?: BranchSummaryEntry;\n\tfromExtension?: boolean;\n}\n\nexport type SessionEvent =\n\t| SessionDirectoryEvent\n\t| SessionStartEvent\n\t| SessionBeforeSwitchEvent\n\t| SessionSwitchEvent\n\t| SessionBeforeForkEvent\n\t| SessionForkEvent\n\t| SessionBeforeCompactEvent\n\t| SessionCompactEvent\n\t| SessionShutdownEvent\n\t| SessionBeforeTreeEvent\n\t| SessionTreeEvent;\n\n// ============================================================================\n// Agent Events\n// ============================================================================\n\n/** Fired before each LLM call. Can modify messages. */\nexport interface ContextEvent {\n\ttype: \"context\";\n\tmessages: AgentMessage[];\n}\n\n/** Fired before a provider request is sent. Can replace the payload. */\nexport interface BeforeProviderRequestEvent {\n\ttype: \"before_provider_request\";\n\tpayload: unknown;\n\t/** The resolved model for this request (provider, id, etc.) */\n\tmodel?: { provider: string; id: string };\n}\n\n/** Fired after user submits prompt but before agent loop. */\nexport interface BeforeAgentStartEvent {\n\ttype: \"before_agent_start\";\n\tprompt: string;\n\timages?: ImageContent[];\n\tsystemPrompt: string;\n}\n\n/** Fired when an agent loop starts */\nexport interface AgentStartEvent {\n\ttype: \"agent_start\";\n}\n\n/** Fired when an agent loop ends */\nexport interface AgentEndEvent {\n\ttype: \"agent_end\";\n\tmessages: AgentMessage[];\n}\n\n/** Fired at the start of each turn */\nexport interface TurnStartEvent {\n\ttype: \"turn_start\";\n\tturnIndex: number;\n\ttimestamp: number;\n}\n\n/** Fired at the end of each turn */\nexport interface TurnEndEvent {\n\ttype: \"turn_end\";\n\tturnIndex: number;\n\tmessage: AgentMessage;\n\ttoolResults: ToolResultMessage[];\n}\n\n/** Fired when a message starts (user, assistant, or toolResult) */\nexport interface MessageStartEvent {\n\ttype: \"message_start\";\n\tmessage: AgentMessage;\n}\n\n/** Fired during assistant message streaming with token-by-token updates */\nexport interface MessageUpdateEvent {\n\ttype: \"message_update\";\n\tmessage: AgentMessage;\n\tassistantMessageEvent: AssistantMessageEvent;\n}\n\n/** Fired when a message ends */\nexport interface MessageEndEvent {\n\ttype: \"message_end\";\n\tmessage: AgentMessage;\n}\n\n/** Fired when a tool starts executing */\nexport interface ToolExecutionStartEvent {\n\ttype: \"tool_execution_start\";\n\ttoolCallId: string;\n\ttoolName: string;\n\targs: any;\n}\n\n/** Fired during tool execution with partial/streaming output */\nexport interface ToolExecutionUpdateEvent {\n\ttype: \"tool_execution_update\";\n\ttoolCallId: string;\n\ttoolName: string;\n\targs: any;\n\tpartialResult: any;\n}\n\n/** Fired when a tool finishes executing */\nexport interface ToolExecutionEndEvent {\n\ttype: \"tool_execution_end\";\n\ttoolCallId: string;\n\ttoolName: string;\n\tresult: any;\n\tisError: boolean;\n}\n\n// ============================================================================\n// Model Events\n// ============================================================================\n\nexport type ModelSelectSource = \"set\" | \"cycle\" | \"restore\";\n\n/** Fired when a new model is selected */\nexport interface ModelSelectEvent {\n\ttype: \"model_select\";\n\tmodel: Model<any>;\n\tpreviousModel: Model<any> | undefined;\n\tsource: ModelSelectSource;\n}\n\n/** Fired before model selection runs capability scoring. Extensions can override the selected model. */\nexport interface BeforeModelSelectEvent {\n\ttype: \"before_model_select\";\n\tunitType: string;\n\tunitId: string;\n\tclassification: { tier: string; reason: string; downgraded: boolean };\n\ttaskMetadata?: Record<string, unknown>;\n\teligibleModels: string[];\n\tphaseConfig?: { primary: string; fallbacks: string[] };\n}\n\n/** Result from before_model_select event handler. Return { modelId } to override selection. */\nexport interface BeforeModelSelectResult {\n\tmodelId: string;\n}\n\n/**\n * Fired after model selection to allow extensions to adjust the active tool set (ADR-005 Phase 4).\n * Extensions can add, remove, or reorder tools based on the selected model's provider capabilities.\n */\nexport interface AdjustToolSetEvent {\n\ttype: \"adjust_tool_set\";\n\t/** The selected model's API type */\n\tselectedModelApi: string;\n\t/** The selected model's provider */\n\tselectedModelProvider: string;\n\t/** The selected model ID */\n\tselectedModelId: string;\n\t/** Current active tool names */\n\tactiveToolNames: string[];\n\t/** Tools already filtered by provider compatibility */\n\tfilteredTools: string[];\n}\n\n/** Result from adjust_tool_set event handler. Return { toolNames } to override tool set. */\nexport interface AdjustToolSetResult {\n\t/** Replacement tool names. If omitted, the default filtering is used. */\n\ttoolNames?: string[];\n}\n\n// ============================================================================\n// User Bash Events\n// ============================================================================\n\n/**\n * Fired before the bash tool executes a shell command.\n * Extensions can return a transformed command string.\n * All registered handlers are called in order; each receives the output of the previous.\n */\nexport interface BashTransformEvent {\n\ttype: \"bash_transform\";\n\t/** The command string about to be executed */\n\tcommand: string;\n\t/** Current working directory */\n\tcwd: string;\n}\n\n/** Result from bash_transform event handler */\nexport interface BashTransformEventResult {\n\t/** Replacement command string. If omitted or empty, the original command is used. */\n\tcommand?: string;\n}\n\n/** Fired when user executes a bash command via ! or !! prefix */\nexport interface UserBashEvent {\n\ttype: \"user_bash\";\n\t/** The command to execute */\n\tcommand: string;\n\t/** True if !! prefix was used (excluded from LLM context) */\n\texcludeFromContext: boolean;\n\t/** Current working directory */\n\tcwd: string;\n}\n\n// ============================================================================\n// Input Events\n// ============================================================================\n\n/** Source of user input */\nexport type InputSource = \"interactive\" | \"rpc\" | \"extension\";\n\n/** Fired when user input is received, before agent processing */\nexport interface InputEvent {\n\ttype: \"input\";\n\t/** The input text */\n\ttext: string;\n\t/** Attached images, if any */\n\timages?: ImageContent[];\n\t/** Where the input came from */\n\tsource: InputSource;\n}\n\n/** Result from input event handler */\nexport type InputEventResult =\n\t| { action: \"continue\" }\n\t| { action: \"transform\"; text: string; images?: ImageContent[] }\n\t| { action: \"handled\" };\n\n// ============================================================================\n// Tool Events\n// ============================================================================\n\ninterface ToolCallEventBase {\n\ttype: \"tool_call\";\n\ttoolCallId: string;\n}\n\nexport interface BashToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"bash\";\n\tinput: BashToolInput;\n}\n\nexport interface ReadToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"read\";\n\tinput: ReadToolInput;\n}\n\nexport interface EditToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"edit\";\n\tinput: EditToolInput;\n}\n\nexport interface WriteToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"write\";\n\tinput: WriteToolInput;\n}\n\nexport interface GrepToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"grep\";\n\tinput: GrepToolInput;\n}\n\nexport interface FindToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"find\";\n\tinput: FindToolInput;\n}\n\nexport interface LsToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"ls\";\n\tinput: LsToolInput;\n}\n\nexport interface CustomToolCallEvent extends ToolCallEventBase {\n\ttoolName: string;\n\tinput: Record<string, unknown>;\n}\n\n/** Fired before a tool executes. Can block. */\nexport type ToolCallEvent =\n\t| BashToolCallEvent\n\t| ReadToolCallEvent\n\t| EditToolCallEvent\n\t| WriteToolCallEvent\n\t| GrepToolCallEvent\n\t| FindToolCallEvent\n\t| LsToolCallEvent\n\t| CustomToolCallEvent;\n\ninterface ToolResultEventBase {\n\ttype: \"tool_result\";\n\ttoolCallId: string;\n\tinput: Record<string, unknown>;\n\tcontent: (TextContent | ImageContent)[];\n\tisError: boolean;\n}\n\nexport interface BashToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"bash\";\n\tdetails: BashToolDetails | undefined;\n}\n\nexport interface ReadToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"read\";\n\tdetails: ReadToolDetails | undefined;\n}\n\nexport interface EditToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"edit\";\n\tdetails: EditToolDetails | undefined;\n}\n\nexport interface WriteToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"write\";\n\tdetails: undefined;\n}\n\nexport interface GrepToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"grep\";\n\tdetails: GrepToolDetails | undefined;\n}\n\nexport interface FindToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"find\";\n\tdetails: FindToolDetails | undefined;\n}\n\nexport interface LsToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"ls\";\n\tdetails: LsToolDetails | undefined;\n}\n\nexport interface CustomToolResultEvent extends ToolResultEventBase {\n\ttoolName: string;\n\tdetails: unknown;\n}\n\n/** Fired after a tool executes. Can modify result. */\nexport type ToolResultEvent =\n\t| BashToolResultEvent\n\t| ReadToolResultEvent\n\t| EditToolResultEvent\n\t| WriteToolResultEvent\n\t| GrepToolResultEvent\n\t| FindToolResultEvent\n\t| LsToolResultEvent\n\t| CustomToolResultEvent;\n\n/**\n * Type guard for narrowing ToolResultEvent by tool name.\n *\n * Built-in tools narrow automatically (no type params needed):\n * ```ts\n * if (isToolResultEventType(\"bash\", event)) {\n * event.details; // BashToolDetails | undefined\n * }\n * ```\n *\n * Custom tools require explicit type parameters:\n * ```ts\n * if (isToolResultEventType<\"my_tool\", MyDetails>(\"my_tool\", event)) {\n * event.details; // typed\n * }\n * ```\n */\nexport function isToolResultEventType(toolName: \"bash\", event: ToolResultEvent): event is BashToolResultEvent;\nexport function isToolResultEventType(toolName: \"read\", event: ToolResultEvent): event is ReadToolResultEvent;\nexport function isToolResultEventType(toolName: \"edit\", event: ToolResultEvent): event is EditToolResultEvent;\nexport function isToolResultEventType(toolName: \"write\", event: ToolResultEvent): event is WriteToolResultEvent;\nexport function isToolResultEventType(toolName: \"grep\", event: ToolResultEvent): event is GrepToolResultEvent;\nexport function isToolResultEventType(toolName: \"find\", event: ToolResultEvent): event is FindToolResultEvent;\nexport function isToolResultEventType(toolName: \"ls\", event: ToolResultEvent): event is LsToolResultEvent;\nexport function isToolResultEventType<TName extends string, TDetails = unknown>(\n\ttoolName: TName,\n\tevent: ToolResultEvent,\n): event is ToolResultEvent & { toolName: TName; details: TDetails };\nexport function isToolResultEventType(toolName: string, event: ToolResultEvent): boolean {\n\treturn event.toolName === toolName;\n}\n\n/**\n * Type guard for narrowing ToolCallEvent by tool name.\n *\n * Built-in tools narrow automatically (no type params needed):\n * ```ts\n * if (isToolCallEventType(\"bash\", event)) {\n * event.input.command; // string\n * }\n * ```\n *\n * Custom tools require explicit type parameters:\n * ```ts\n * if (isToolCallEventType<\"my_tool\", MyToolInput>(\"my_tool\", event)) {\n * event.input.action; // typed\n * }\n * ```\n *\n * Note: Direct narrowing via `event.toolName === \"bash\"` doesn't work because\n * CustomToolCallEvent.toolName is `string` which overlaps with all literals.\n */\nexport function isToolCallEventType(toolName: \"bash\", event: ToolCallEvent): event is BashToolCallEvent;\nexport function isToolCallEventType(toolName: \"read\", event: ToolCallEvent): event is ReadToolCallEvent;\nexport function isToolCallEventType(toolName: \"edit\", event: ToolCallEvent): event is EditToolCallEvent;\nexport function isToolCallEventType(toolName: \"write\", event: ToolCallEvent): event is WriteToolCallEvent;\nexport function isToolCallEventType(toolName: \"grep\", event: ToolCallEvent): event is GrepToolCallEvent;\nexport function isToolCallEventType(toolName: \"find\", event: ToolCallEvent): event is FindToolCallEvent;\nexport function isToolCallEventType(toolName: \"ls\", event: ToolCallEvent): event is LsToolCallEvent;\nexport function isToolCallEventType<TName extends string, TInput extends Record<string, unknown>>(\n\ttoolName: TName,\n\tevent: ToolCallEvent,\n): event is ToolCallEvent & { toolName: TName; input: TInput };\nexport function isToolCallEventType(toolName: string, event: ToolCallEvent): boolean {\n\treturn event.toolName === toolName;\n}\n\n/** Union of all event types */\nexport type ExtensionEvent =\n\t| ResourcesDiscoverEvent\n\t| SessionEvent\n\t| ContextEvent\n\t| BeforeProviderRequestEvent\n\t| BeforeAgentStartEvent\n\t| AgentStartEvent\n\t| AgentEndEvent\n\t| TurnStartEvent\n\t| TurnEndEvent\n\t| MessageStartEvent\n\t| MessageUpdateEvent\n\t| MessageEndEvent\n\t| ToolExecutionStartEvent\n\t| ToolExecutionUpdateEvent\n\t| ToolExecutionEndEvent\n\t| ModelSelectEvent\n\t| BashTransformEvent\n\t| UserBashEvent\n\t| InputEvent\n\t| ToolCallEvent\n\t| ToolResultEvent;\n\n// ============================================================================\n// Event Results\n// ============================================================================\n\nexport interface ContextEventResult {\n\tmessages?: AgentMessage[];\n}\n\nexport type BeforeProviderRequestEventResult = unknown;\n\nexport interface ToolCallEventResult {\n\tblock?: boolean;\n\treason?: string;\n}\n\n/** Result from user_bash event handler */\nexport interface UserBashEventResult {\n\t/** Custom operations to use for execution */\n\toperations?: BashOperations;\n\t/** Full replacement: extension handled execution, use this result */\n\tresult?: BashResult;\n}\n\nexport interface ToolResultEventResult {\n\tcontent?: (TextContent | ImageContent)[];\n\tdetails?: unknown;\n\tisError?: boolean;\n}\n\nexport interface BeforeAgentStartEventResult {\n\tmessage?: Pick<CustomMessage, \"customType\" | \"content\" | \"display\" | \"details\">;\n\t/** Replace the system prompt for this turn. If multiple extensions return this, they are chained. */\n\tsystemPrompt?: string;\n}\n\nexport interface SessionDirectoryResult {\n\t/** Custom session directory path. If multiple extensions return this, the last one wins. */\n\tsessionDir?: string;\n}\n\n/** Special startup-only handler. Unlike other events, this receives no ExtensionContext. */\nexport type SessionDirectoryHandler = (\n\tevent: SessionDirectoryEvent,\n) => Promise<SessionDirectoryResult | undefined> | SessionDirectoryResult | undefined;\n\nexport interface SessionBeforeSwitchResult {\n\tcancel?: boolean;\n}\n\nexport interface SessionBeforeForkResult {\n\tcancel?: boolean;\n\tskipConversationRestore?: boolean;\n}\n\nexport interface SessionBeforeCompactResult {\n\tcancel?: boolean;\n\tcompaction?: CompactionResult;\n}\n\nexport interface SessionBeforeTreeResult {\n\tcancel?: boolean;\n\tsummary?: {\n\t\tsummary: string;\n\t\tdetails?: unknown;\n\t};\n\t/** Override custom instructions for summarization */\n\tcustomInstructions?: string;\n\t/** Override whether customInstructions replaces the default prompt */\n\treplaceInstructions?: boolean;\n\t/** Override label to attach to the branch summary entry */\n\tlabel?: string;\n}\n\n// ============================================================================\n// Message Rendering\n// ============================================================================\n\nexport interface MessageRenderOptions {\n\texpanded: boolean;\n}\n\nexport type MessageRenderer<T = unknown> = (\n\tmessage: CustomMessage<T>,\n\toptions: MessageRenderOptions,\n\ttheme: Theme,\n) => Component | undefined;\n\n// ============================================================================\n// Command Registration\n// ============================================================================\n\nexport interface RegisteredCommand {\n\tname: string;\n\tdescription?: string;\n\tgetArgumentCompletions?: (argumentPrefix: string) => AutocompleteItem[] | null;\n\thandler: (args: string, ctx: ExtensionCommandContext) => Promise<void>;\n}\n\nexport type LifecycleHookScope = \"user\" | \"project\";\nexport type LifecycleHookPhase = \"beforeInstall\" | \"afterInstall\" | \"beforeRemove\" | \"afterRemove\";\n\nexport interface LifecycleHookContext {\n\t/** Lifecycle phase currently being executed. */\n\tphase: LifecycleHookPhase;\n\t/** Package source string passed to install (npm:, git:, https://, local path). */\n\tsource: string;\n\t/** Resolved installed package path (or resolved local path), when available for this phase. */\n\tinstalledPath?: string;\n\t/** Where the package was installed. */\n\tscope: LifecycleHookScope;\n\t/** Current working directory for the install invocation. */\n\tcwd: string;\n\t/** Whether install is running in an interactive TTY. */\n\tinteractive: boolean;\n\t/** Info-level logging sink for install output. */\n\tlog(message: string): void;\n\t/** Warning-level logging sink for install output. */\n\twarn(message: string): void;\n\t/** Error-level logging sink for install output. */\n\terror(message: string): void;\n}\n\nexport type LifecycleHookHandler = (ctx: LifecycleHookContext) => Promise<void> | void;\nexport type LifecycleHookMap = Record<LifecycleHookPhase, LifecycleHookHandler[]>;\n\n// ============================================================================\n// Extension API\n// ============================================================================\n\n/** Handler function type for events */\n// biome-ignore lint/suspicious/noConfusingVoidType: void allows bare return statements\nexport type ExtensionHandler<E, R = undefined> = (event: E, ctx: ExtensionContext) => Promise<R | void> | R | void;\n\n/**\n * ExtensionAPI passed to extension factory functions.\n */\nexport interface ExtensionAPI {\n\t// =========================================================================\n\t// Event Subscription\n\t// =========================================================================\n\n\ton(event: \"resources_discover\", handler: ExtensionHandler<ResourcesDiscoverEvent, ResourcesDiscoverResult>): void;\n\ton(event: \"session_directory\", handler: SessionDirectoryHandler): void;\n\ton(event: \"session_start\", handler: ExtensionHandler<SessionStartEvent>): void;\n\ton(\n\t\tevent: \"session_before_switch\",\n\t\thandler: ExtensionHandler<SessionBeforeSwitchEvent, SessionBeforeSwitchResult>,\n\t): void;\n\ton(event: \"session_switch\", handler: ExtensionHandler<SessionSwitchEvent>): void;\n\ton(event: \"session_before_fork\", handler: ExtensionHandler<SessionBeforeForkEvent, SessionBeforeForkResult>): void;\n\ton(event: \"session_fork\", handler: ExtensionHandler<SessionForkEvent>): void;\n\ton(\n\t\tevent: \"session_before_compact\",\n\t\thandler: ExtensionHandler<SessionBeforeCompactEvent, SessionBeforeCompactResult>,\n\t): void;\n\ton(event: \"session_compact\", handler: ExtensionHandler<SessionCompactEvent>): void;\n\ton(event: \"session_shutdown\", handler: ExtensionHandler<SessionShutdownEvent>): void;\n\ton(event: \"session_before_tree\", handler: ExtensionHandler<SessionBeforeTreeEvent, SessionBeforeTreeResult>): void;\n\ton(event: \"session_tree\", handler: ExtensionHandler<SessionTreeEvent>): void;\n\ton(event: \"context\", handler: ExtensionHandler<ContextEvent, ContextEventResult>): void;\n\ton(\n\t\tevent: \"before_provider_request\",\n\t\thandler: ExtensionHandler<BeforeProviderRequestEvent, BeforeProviderRequestEventResult>,\n\t): void;\n\ton(event: \"before_agent_start\", handler: ExtensionHandler<BeforeAgentStartEvent, BeforeAgentStartEventResult>): void;\n\ton(event: \"agent_start\", handler: ExtensionHandler<AgentStartEvent>): void;\n\ton(event: \"agent_end\", handler: ExtensionHandler<AgentEndEvent>): void;\n\ton(event: \"turn_start\", handler: ExtensionHandler<TurnStartEvent>): void;\n\ton(event: \"turn_end\", handler: ExtensionHandler<TurnEndEvent>): void;\n\ton(event: \"message_start\", handler: ExtensionHandler<MessageStartEvent>): void;\n\ton(event: \"message_update\", handler: ExtensionHandler<MessageUpdateEvent>): void;\n\ton(event: \"message_end\", handler: ExtensionHandler<MessageEndEvent>): void;\n\ton(event: \"tool_execution_start\", handler: ExtensionHandler<ToolExecutionStartEvent>): void;\n\ton(event: \"tool_execution_update\", handler: ExtensionHandler<ToolExecutionUpdateEvent>): void;\n\ton(event: \"tool_execution_end\", handler: ExtensionHandler<ToolExecutionEndEvent>): void;\n\ton(event: \"model_select\", handler: ExtensionHandler<ModelSelectEvent>): void;\n\ton(event: \"bash_transform\", handler: ExtensionHandler<BashTransformEvent, BashTransformEventResult>): void;\n\ton(event: \"tool_call\", handler: ExtensionHandler<ToolCallEvent, ToolCallEventResult>): void;\n\ton(event: \"tool_result\", handler: ExtensionHandler<ToolResultEvent, ToolResultEventResult>): void;\n\ton(event: \"user_bash\", handler: ExtensionHandler<UserBashEvent, UserBashEventResult>): void;\n\ton(event: \"input\", handler: ExtensionHandler<InputEvent, InputEventResult>): void;\n\ton(event: \"before_model_select\", handler: ExtensionHandler<BeforeModelSelectEvent, BeforeModelSelectResult>): void;\n\ton(event: \"adjust_tool_set\", handler: ExtensionHandler<AdjustToolSetEvent, AdjustToolSetResult>): void;\n\n\t// =========================================================================\n\t// Event Emission (for host extensions that orchestrate model selection)\n\t// =========================================================================\n\n\t/** Emit before_model_select event. Returns override model ID or undefined. */\n\temitBeforeModelSelect(event: Omit<BeforeModelSelectEvent, \"type\">): Promise<BeforeModelSelectResult | undefined>;\n\n\t/** Emit adjust_tool_set event (ADR-005). Returns override tool names or undefined. */\n\temitAdjustToolSet(event: Omit<AdjustToolSetEvent, \"type\">): Promise<AdjustToolSetResult | undefined>;\n\n\t// =========================================================================\n\t// Tool Registration\n\t// =========================================================================\n\n\t/** Register a tool that the LLM can call. */\n\tregisterTool<TParams extends TSchema = TSchema, TDetails = unknown>(tool: ToolDefinition<TParams, TDetails>): void;\n\n\t// =========================================================================\n\t// Command, Shortcut, Flag Registration\n\t// =========================================================================\n\n\t/** Register a custom command. */\n\tregisterCommand(name: string, options: Omit<RegisteredCommand, \"name\">): void;\n\n\t/** Register a lifecycle hook run before package installation starts. */\n\tregisterBeforeInstall(handler: LifecycleHookHandler): void;\n\n\t/** Register a lifecycle hook run after package installation completes. */\n\tregisterAfterInstall(handler: LifecycleHookHandler): void;\n\n\t/** Register a lifecycle hook run before package removal starts. */\n\tregisterBeforeRemove(handler: LifecycleHookHandler): void;\n\n\t/** Register a lifecycle hook run after package removal completes. */\n\tregisterAfterRemove(handler: LifecycleHookHandler): void;\n\n\t/** Register a keyboard shortcut. */\n\tregisterShortcut(\n\t\tshortcut: KeyId,\n\t\toptions: {\n\t\t\tdescription?: string;\n\t\t\thandler: (ctx: ExtensionContext) => Promise<void> | void;\n\t\t},\n\t): void;\n\n\t/** Register a CLI flag. */\n\tregisterFlag(\n\t\tname: string,\n\t\toptions: {\n\t\t\tdescription?: string;\n\t\t\ttype: \"boolean\" | \"string\";\n\t\t\tdefault?: boolean | string;\n\t\t},\n\t): void;\n\n\t/** Get the value of a registered CLI flag. */\n\tgetFlag(name: string): boolean | string | undefined;\n\n\t// =========================================================================\n\t// Message Rendering\n\t// =========================================================================\n\n\t/** Register a custom renderer for CustomMessageEntry. */\n\tregisterMessageRenderer<T = unknown>(customType: string, renderer: MessageRenderer<T>): void;\n\n\t// =========================================================================\n\t// Actions\n\t// =========================================================================\n\n\t/** Send a custom message to the session. */\n\tsendMessage<T = unknown>(\n\t\tmessage: Pick<CustomMessage<T>, \"customType\" | \"content\" | \"display\" | \"details\">,\n\t\toptions?: { triggerTurn?: boolean; deliverAs?: \"steer\" | \"followUp\" | \"nextTurn\" },\n\t): void;\n\n\t/**\n\t * Send a user message to the agent. Always triggers a turn.\n\t * When the agent is streaming, use deliverAs to specify how to queue the message.\n\t */\n\tsendUserMessage(\n\t\tcontent: string | (TextContent | ImageContent)[],\n\t\toptions?: { deliverAs?: \"steer\" | \"followUp\" },\n\t): void;\n\n\t/**\n\t * Retry the last turn by removing the failed assistant response and\n\t * re-running the agent from the last user message. No-op if the last\n\t * message is not an assistant error.\n\t */\n\tretryLastTurn(): void;\n\n\t/** Append a custom entry to the session for state persistence (not sent to LLM). */\n\tappendEntry<T = unknown>(customType: string, data?: T): void;\n\n\t// =========================================================================\n\t// Session Metadata\n\t// =========================================================================\n\n\t/** Set the session display name (shown in session selector). */\n\tsetSessionName(name: string): void;\n\n\t/** Get the current session name, if set. */\n\tgetSessionName(): string | undefined;\n\n\t/** Set or clear a label on an entry. Labels are user-defined markers for bookmarking/navigation. */\n\tsetLabel(entryId: string, label: string | undefined): void;\n\n\t/** Execute a shell command. */\n\texec(command: string, args: string[], options?: ExecOptions): Promise<ExecResult>;\n\n\t/** Get the list of currently active tool names. */\n\tgetActiveTools(): string[];\n\n\t/** Get all configured tools with name and description. */\n\tgetAllTools(): ToolInfo[];\n\n\t/** Set the active tools by name. */\n\tsetActiveTools(toolNames: string[]): void;\n\n\t/** Get available slash commands in the current session. */\n\tgetCommands(): SlashCommandInfo[];\n\n\t// =========================================================================\n\t// Model and Thinking Level\n\t// =========================================================================\n\n\t/** Set the current model. Returns false if no API key available. */\n\tsetModel(model: Model<any>, options?: { persist?: boolean }): Promise<boolean>;\n\n\t/** Get current thinking level. */\n\tgetThinkingLevel(): ThinkingLevel;\n\n\t/** Set thinking level (clamped to model capabilities). */\n\tsetThinkingLevel(level: ThinkingLevel): void;\n\n\t// =========================================================================\n\t// Provider Registration\n\t// =========================================================================\n\n\t/**\n\t * Register or override a model provider.\n\t *\n\t * If `models` is provided: replaces all existing models for this provider.\n\t * If only `baseUrl` is provided: overrides the URL for existing models.\n\t * If `oauth` is provided: registers OAuth provider for /login support.\n\t * If `streamSimple` is provided: registers a custom API stream handler.\n\t *\n\t * During initial extension load this call is queued and applied once the\n\t * runner has bound its context. After that it takes effect immediately, so\n\t * it is safe to call from command handlers or event callbacks without\n\t * requiring a `/reload`.\n\t *\n\t * @example\n\t * // Register a new provider with custom models\n\t * pi.registerProvider(\"my-proxy\", {\n\t * baseUrl: \"https://proxy.example.com\",\n\t * apiKey: \"PROXY_API_KEY\",\n\t * api: \"anthropic-messages\",\n\t * models: [\n\t * {\n\t * id: \"claude-sonnet-4-20250514\",\n\t * name: \"Claude 4 Sonnet (proxy)\",\n\t * reasoning: false,\n\t * input: [\"text\", \"image\"],\n\t * cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },\n\t * contextWindow: 200000,\n\t * maxTokens: 16384\n\t * }\n\t * ]\n\t * });\n\t *\n\t * @example\n\t * // Override baseUrl for an existing provider\n\t * pi.registerProvider(\"anthropic\", {\n\t * baseUrl: \"https://proxy.example.com\"\n\t * });\n\t *\n\t * @example\n\t * // Register provider with OAuth support\n\t * pi.registerProvider(\"corporate-ai\", {\n\t * baseUrl: \"https://ai.corp.com\",\n\t * api: \"openai-responses\",\n\t * models: [...],\n\t * oauth: {\n\t * name: \"Corporate AI (SSO)\",\n\t * async login(callbacks) { ... },\n\t * async refreshToken(credentials) { ... },\n\t * getApiKey(credentials) { return credentials.access; }\n\t * }\n\t * });\n\t */\n\tregisterProvider(name: string, config: ProviderConfig): void;\n\n\t/**\n\t * Unregister a previously registered provider.\n\t *\n\t * Removes all models belonging to the named provider and restores any\n\t * built-in models that were overridden by it. Has no effect if the provider\n\t * is not currently registered.\n\t *\n\t * Like `registerProvider`, this takes effect immediately when called after\n\t * the initial load phase.\n\t *\n\t * @example\n\t * pi.unregisterProvider(\"my-proxy\");\n\t */\n\tunregisterProvider(name: string): void;\n\n\t/** Shared event bus for extension communication. */\n\tevents: EventBus;\n}\n\n// ============================================================================\n// Provider Registration Types\n// ============================================================================\n\n/** Configuration for registering a provider via pi.registerProvider(). */\nexport interface ProviderConfig {\n\t/** Auth behavior for provider availability and request key handling. Defaults to \"apiKey\". */\n\tauthMode?: \"apiKey\" | \"oauth\" | \"externalCli\" | \"none\";\n\t/** Optional readiness check. Return false if the provider cannot accept requests (e.g., CLI not authenticated, API key invalid).\n\t * Called before default auth checks. Trusted at the same level as extension code — extensions already have arbitrary code execution. */\n\tisReady?: () => boolean;\n\t/** Base URL for the API endpoint. Required when defining models. */\n\tbaseUrl?: string;\n\t/** API key or environment variable name. Required when defining models (unless oauth provided). */\n\tapiKey?: string;\n\t/** API type. Required at provider or model level when defining models. */\n\tapi?: Api;\n\t/** Optional streamSimple handler for custom APIs. */\n\tstreamSimple?: (model: Model<Api>, context: Context, options?: SimpleStreamOptions) => AssistantMessageEventStream;\n\t/** Custom headers to include in requests. */\n\theaders?: Record<string, string>;\n\t/** If true, adds Authorization: Bearer header with the resolved API key. */\n\tauthHeader?: boolean;\n\t/** Models to register. If provided, replaces all existing models for this provider. */\n\tmodels?: ProviderModelConfig[];\n\t/** OAuth provider for /login support. The `id` is set automatically from the provider name. */\n\toauth?: {\n\t\t/** Display name for the provider in login UI. */\n\t\tname: string;\n\t\t/** Run the login flow, return credentials to persist. */\n\t\tlogin(callbacks: OAuthLoginCallbacks): Promise<OAuthCredentials>;\n\t\t/** Refresh expired credentials, return updated credentials to persist. */\n\t\trefreshToken(credentials: OAuthCredentials): Promise<OAuthCredentials>;\n\t\t/** Convert credentials to API key string for the provider. */\n\t\tgetApiKey(credentials: OAuthCredentials): string;\n\t\t/** Optional: modify models for this provider (e.g., update baseUrl based on credentials). */\n\t\tmodifyModels?(models: Model<Api>[], credentials: OAuthCredentials): Model<Api>[];\n\t};\n}\n\n/** Configuration for a model within a provider. */\nexport interface ProviderModelConfig {\n\t/** Model ID (e.g., \"claude-sonnet-4-20250514\"). */\n\tid: string;\n\t/** Display name (e.g., \"Claude 4 Sonnet\"). */\n\tname: string;\n\t/** API type override for this model. */\n\tapi?: Api;\n\t/** Whether the model supports extended thinking. */\n\treasoning: boolean;\n\t/** Supported input types. */\n\tinput: (\"text\" | \"image\")[];\n\t/** Cost per token (for tracking, can be 0). */\n\tcost: { input: number; output: number; cacheRead: number; cacheWrite: number };\n\t/** Maximum context window size in tokens. */\n\tcontextWindow: number;\n\t/** Maximum output tokens. */\n\tmaxTokens: number;\n\t/** Custom headers for this model. */\n\theaders?: Record<string, string>;\n\t/** OpenAI compatibility settings. */\n\tcompat?: Model<Api>[\"compat\"];\n\t/** Opaque provider-specific options (e.g. Ollama keep_alive, num_gpu). */\n\tproviderOptions?: Record<string, unknown>;\n}\n\n/** Extension factory function type. Supports both sync and async initialization. */\nexport type ExtensionFactory = (pi: ExtensionAPI) => void | Promise<void>;\n\n// ============================================================================\n// Loaded Extension Types\n// ============================================================================\n\nexport interface RegisteredTool {\n\tdefinition: ToolDefinition;\n\textensionPath: string;\n}\n\nexport interface ExtensionFlag {\n\tname: string;\n\tdescription?: string;\n\ttype: \"boolean\" | \"string\";\n\tdefault?: boolean | string;\n\textensionPath: string;\n}\n\nexport interface ExtensionShortcut {\n\tshortcut: KeyId;\n\tdescription?: string;\n\thandler: (ctx: ExtensionContext) => Promise<void> | void;\n\textensionPath: string;\n}\n\ntype HandlerFn = (...args: unknown[]) => Promise<unknown>;\n\n/** Tool info with name, description, and parameter schema */\nexport type ToolInfo = Pick<ToolDefinition, \"name\" | \"description\" | \"parameters\">;\n\n/**\n * Shared state created by loader, used during registration and runtime.\n * Contains flag values (defaults set during registration, CLI values set after).\n */\nexport interface ExtensionRuntimeState {\n\tflagValues: Map<string, boolean | string>;\n\t/** Provider registrations queued during extension loading, processed when runner binds */\n\tpendingProviderRegistrations: Array<{ name: string; config: ProviderConfig }>;\n\t/**\n\t * Register or unregister a provider.\n\t *\n\t * Before bindCore(): queues registrations / removes from queue.\n\t * After bindCore(): calls ModelRegistry directly for immediate effect.\n\t */\n\tregisterProvider: (name: string, config: ProviderConfig) => void;\n\tunregisterProvider: (name: string) => void;\n\t/** Emit before_model_select event to all registered handlers. Bound by ExtensionRunner. */\n\temitBeforeModelSelect: (event: Omit<BeforeModelSelectEvent, \"type\">) => Promise<BeforeModelSelectResult | undefined>;\n\t/** Emit adjust_tool_set event to all registered handlers. Bound by ExtensionRunner (ADR-005). */\n\temitAdjustToolSet: (event: Omit<AdjustToolSetEvent, \"type\">) => Promise<AdjustToolSetResult | undefined>;\n}\n\n/**\n * Action implementations for pi.* API methods.\n * Provided to runner.initialize(), copied into the shared runtime.\n */\nexport interface ExtensionActions {\n\tsendMessage: <T = unknown>(\n\t\tmessage: Pick<CustomMessage<T>, \"customType\" | \"content\" | \"display\" | \"details\">,\n\t\toptions?: { triggerTurn?: boolean; deliverAs?: \"steer\" | \"followUp\" | \"nextTurn\" },\n\t) => void;\n\tsendUserMessage: (\n\t\tcontent: string | (TextContent | ImageContent)[],\n\t\toptions?: { deliverAs?: \"steer\" | \"followUp\" },\n\t) => void;\n\tretryLastTurn: () => void;\n\tappendEntry: <T = unknown>(customType: string, data?: T) => void;\n\tsetSessionName: (name: string) => void;\n\tgetSessionName: () => string | undefined;\n\tsetLabel: (entryId: string, label: string | undefined) => void;\n\tgetActiveTools: () => string[];\n\tgetAllTools: () => ToolInfo[];\n\tsetActiveTools: (toolNames: string[]) => void;\n\trefreshTools: () => void;\n\tgetCommands: () => SlashCommandInfo[];\n\tsetModel: (model: Model<any>, options?: { persist?: boolean }) => Promise<boolean>;\n\tgetThinkingLevel: () => ThinkingLevel;\n\tsetThinkingLevel: (level: ThinkingLevel) => void;\n}\n\n/**\n * Actions for ExtensionContext (ctx.* in event handlers).\n * Required by all modes.\n */\nexport interface ExtensionContextActions {\n\tgetModel: () => Model<any> | undefined;\n\tisIdle: () => boolean;\n\tabort: () => void;\n\thasPendingMessages: () => boolean;\n\tshutdown: () => void;\n\tgetContextUsage: () => ContextUsage | undefined;\n\tcompact: (options?: CompactOptions) => void;\n\tgetSystemPrompt: () => string;\n}\n\n/**\n * Actions for ExtensionCommandContext (ctx.* in command handlers).\n * Only needed for interactive mode where extension commands are invokable.\n */\nexport interface ExtensionCommandContextActions {\n\twaitForIdle: () => Promise<void>;\n\tnewSession: (options?: {\n\t\tparentSession?: string;\n\t\tsetup?: (sessionManager: SessionManager) => Promise<void>;\n\t}) => Promise<{ cancelled: boolean }>;\n\tfork: (entryId: string) => Promise<{ cancelled: boolean }>;\n\tnavigateTree: (\n\t\ttargetId: string,\n\t\toptions?: { summarize?: boolean; customInstructions?: string; replaceInstructions?: boolean; label?: string },\n\t) => Promise<{ cancelled: boolean }>;\n\tswitchSession: (sessionPath: string) => Promise<{ cancelled: boolean }>;\n\treload: () => Promise<void>;\n}\n\n/**\n * Full runtime = state + actions.\n * Created by loader with throwing action stubs, completed by runner.initialize().\n */\nexport interface ExtensionRuntime extends ExtensionRuntimeState, ExtensionActions {}\n\n/** Loaded extension with all registered items. */\nexport interface Extension {\n\tpath: string;\n\tresolvedPath: string;\n\thandlers: Map<string, HandlerFn[]>;\n\ttools: Map<string, RegisteredTool>;\n\tmessageRenderers: Map<string, MessageRenderer>;\n\tcommands: Map<string, RegisteredCommand>;\n\tflags: Map<string, ExtensionFlag>;\n\tshortcuts: Map<KeyId, ExtensionShortcut>;\n\tlifecycleHooks: LifecycleHookMap;\n}\n\n/** Result of loading extensions. */\nexport interface LoadExtensionsResult {\n\textensions: Extension[];\n\terrors: Array<{ path: string; error: string }>;\n\t/** Shared runtime - actions are throwing stubs until runner.initialize() */\n\truntime: ExtensionRuntime;\n}\n\n// ============================================================================\n// Extension Error\n// ============================================================================\n\nexport interface ExtensionError {\n\textensionPath: string;\n\tevent: string;\n\terror: string;\n\tstack?: string;\n}\n"]}
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/core/extensions/types.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AA21BH,MAAM,UAAU,qBAAqB,CAAC,QAAgB,EAAE,KAAsB;IAC7E,OAAO,KAAK,CAAC,QAAQ,KAAK,QAAQ,CAAC;AACpC,CAAC;AAiCD,MAAM,UAAU,mBAAmB,CAAC,QAAgB,EAAE,KAAoB;IACzE,OAAO,KAAK,CAAC,QAAQ,KAAK,QAAQ,CAAC;AACpC,CAAC","sourcesContent":["/**\n * Extension system types.\n *\n * Extensions are TypeScript modules that can:\n * - Subscribe to agent lifecycle events\n * - Register LLM-callable tools\n * - Register commands, keyboard shortcuts, and CLI flags\n * - Interact with the user via UI primitives\n */\n\nimport type {\n\tAgentMessage,\n\tAgentToolResult,\n\tAgentToolUpdateCallback,\n\tThinkingLevel,\n} from \"@gsd/pi-agent-core\";\nimport type {\n\tApi,\n\tAssistantMessageEvent,\n\tAssistantMessageEventStream,\n\tContext,\n\tImageContent,\n\tModel,\n\tOAuthCredentials,\n\tOAuthLoginCallbacks,\n\tSimpleStreamOptions,\n\tTextContent,\n\tToolResultMessage,\n} from \"@gsd/pi-ai\";\nimport type {\n\tAutocompleteItem,\n\tComponent,\n\tEditorComponent,\n\tEditorTheme,\n\tKeyId,\n\tOverlayHandle,\n\tOverlayOptions,\n\tTUI,\n} from \"@gsd/pi-tui\";\nimport type { Static, TSchema } from \"@sinclair/typebox\";\nimport type { Theme } from \"../../modes/interactive/theme/theme.js\";\nimport type { BashResult } from \"../bash-executor.js\";\nimport type { CompactionPreparation, CompactionResult } from \"../compaction/index.js\";\nimport type { EventBus } from \"../event-bus.js\";\nimport type { ExecOptions, ExecResult } from \"../exec.js\";\nimport type { ReadonlyFooterDataProvider } from \"../footer-data-provider.js\";\nimport type { KeybindingsManager } from \"../keybindings.js\";\nimport type { CustomMessage } from \"../messages.js\";\nimport type { ModelRegistry } from \"../model-registry.js\";\nimport type {\n\tBranchSummaryEntry,\n\tCompactionEntry,\n\tReadonlySessionManager,\n\tSessionEntry,\n\tSessionManager,\n} from \"../session-manager.js\";\nimport type { SlashCommandInfo } from \"../slash-commands.js\";\nimport type { BashOperations } from \"../tools/bash.js\";\nimport type { EditToolDetails } from \"../tools/edit.js\";\nimport type {\n\tBashToolDetails,\n\tBashToolInput,\n\tEditToolInput,\n\tFindToolDetails,\n\tFindToolInput,\n\tGrepToolDetails,\n\tGrepToolInput,\n\tLsToolDetails,\n\tLsToolInput,\n\tReadToolDetails,\n\tReadToolInput,\n\tWriteToolInput,\n} from \"../tools/index.js\";\n\nexport type { ExecOptions, ExecResult } from \"../exec.js\";\nexport type { AgentToolResult, AgentToolUpdateCallback };\nexport type { AppAction, KeybindingsManager } from \"../keybindings.js\";\n\n// ============================================================================\n// UI Context\n// ============================================================================\n\n/** Options for extension UI dialogs. */\nexport interface ExtensionUIDialogOptions {\n\t/** AbortSignal to programmatically dismiss the dialog. */\n\tsignal?: AbortSignal;\n\t/** Timeout in milliseconds. Dialog auto-dismisses with live countdown display. */\n\ttimeout?: number;\n\t/** When true, the user can select multiple options. The return type becomes `string[]`. */\n\tallowMultiple?: boolean;\n\t/** When true, text input dialogs should hide typed characters if supported by the client surface. */\n\tsecure?: boolean;\n}\n\n/** Placement for extension widgets. */\nexport type WidgetPlacement = \"aboveEditor\" | \"belowEditor\";\n\n/** Options for extension widgets. */\nexport interface ExtensionWidgetOptions {\n\t/** Where the widget is rendered. Defaults to \"aboveEditor\". */\n\tplacement?: WidgetPlacement;\n}\n\n/** Raw terminal input listener for extensions. */\nexport type TerminalInputHandler = (data: string) => { consume?: boolean; data?: string } | undefined;\n\n/**\n * UI context for extensions to request interactive UI.\n * Each mode (interactive, RPC, print) provides its own implementation.\n */\nexport interface ExtensionUIContext {\n\t/** Show a selector and return the user's choice. When `opts.allowMultiple` is true, returns an array. */\n\tselect(title: string, options: string[], opts?: ExtensionUIDialogOptions): Promise<string | string[] | undefined>;\n\n\t/** Show a confirmation dialog. */\n\tconfirm(title: string, message: string, opts?: ExtensionUIDialogOptions): Promise<boolean>;\n\n\t/** Show a text input dialog. */\n\tinput(title: string, placeholder?: string, opts?: ExtensionUIDialogOptions): Promise<string | undefined>;\n\n\t/** Show a notification to the user. */\n\tnotify(message: string, type?: \"info\" | \"warning\" | \"error\" | \"success\"): void;\n\n\t/** Listen to raw terminal input (interactive mode only). Returns an unsubscribe function. */\n\tonTerminalInput(handler: TerminalInputHandler): () => void;\n\n\t/** Set status text in the footer/status bar. Pass undefined to clear. */\n\tsetStatus(key: string, text: string | undefined): void;\n\n\t/** Set the working/loading message shown during streaming. Call with no argument to restore default. */\n\tsetWorkingMessage(message?: string): void;\n\n\t/** Set a widget to display above or below the editor. Accepts string array or component factory. */\n\tsetWidget(key: string, content: string[] | undefined, options?: ExtensionWidgetOptions): void;\n\tsetWidget(\n\t\tkey: string,\n\t\tcontent: ((tui: TUI, theme: Theme) => Component & { dispose?(): void }) | undefined,\n\t\toptions?: ExtensionWidgetOptions,\n\t): void;\n\n\t/** Set a custom footer component, or undefined to restore the built-in footer.\n\t *\n\t * The factory receives a FooterDataProvider for data not otherwise accessible:\n\t * git branch and extension statuses from setStatus(). Token stats, model info,\n\t * etc. are available via ctx.sessionManager and ctx.model.\n\t */\n\tsetFooter(\n\t\tfactory:\n\t\t\t| ((tui: TUI, theme: Theme, footerData: ReadonlyFooterDataProvider) => Component & { dispose?(): void })\n\t\t\t| undefined,\n\t): void;\n\n\t/** Set a custom header component (shown at startup, above chat), or undefined to restore the built-in header. */\n\tsetHeader(factory: ((tui: TUI, theme: Theme) => Component & { dispose?(): void }) | undefined): void;\n\n\t/** Set the terminal window/tab title. */\n\tsetTitle(title: string): void;\n\n\t/** Show a custom component with keyboard focus. */\n\tcustom<T>(\n\t\tfactory: (\n\t\t\ttui: TUI,\n\t\t\ttheme: Theme,\n\t\t\tkeybindings: KeybindingsManager,\n\t\t\tdone: (result: T) => void,\n\t\t) => (Component & { dispose?(): void }) | Promise<Component & { dispose?(): void }>,\n\t\toptions?: {\n\t\t\toverlay?: boolean;\n\t\t\t/** Overlay positioning/sizing options. Can be static or a function for dynamic updates. */\n\t\t\toverlayOptions?: OverlayOptions | (() => OverlayOptions);\n\t\t\t/** Called with the overlay handle after the overlay is shown. Use to control visibility. */\n\t\t\tonHandle?: (handle: OverlayHandle) => void;\n\t\t},\n\t): Promise<T>;\n\n\t/** Paste text into the editor, triggering paste handling (collapse for large content). */\n\tpasteToEditor(text: string): void;\n\n\t/** Set the text in the core input editor. */\n\tsetEditorText(text: string): void;\n\n\t/** Get the current text from the core input editor. */\n\tgetEditorText(): string;\n\n\t/** Show a multi-line editor for text editing. */\n\teditor(title: string, prefill?: string): Promise<string | undefined>;\n\n\t/**\n\t * Set a custom editor component via factory function.\n\t * Pass undefined to restore the default editor.\n\t *\n\t * The factory receives:\n\t * - `theme`: EditorTheme for styling borders and autocomplete\n\t * - `keybindings`: KeybindingsManager for app-level keybindings\n\t *\n\t * For full app keybinding support (escape, ctrl+d, model switching, etc.),\n\t * extend `CustomEditor` from `@gsd/pi-coding-agent` and call\n\t * `super.handleInput(data)` for keys you don't handle.\n\t *\n\t * @example\n\t * ```ts\n\t * import { CustomEditor } from \"@gsd/pi-coding-agent\";\n\t *\n\t * class VimEditor extends CustomEditor {\n\t * private mode: \"normal\" | \"insert\" = \"insert\";\n\t *\n\t * handleInput(data: string): void {\n\t * if (this.mode === \"normal\") {\n\t * // Handle vim normal mode keys...\n\t * if (data === \"i\") { this.mode = \"insert\"; return; }\n\t * }\n\t * super.handleInput(data); // App keybindings + text editing\n\t * }\n\t * }\n\t *\n\t * ctx.ui.setEditorComponent((tui, theme, keybindings) =>\n\t * new VimEditor(tui, theme, keybindings)\n\t * );\n\t * ```\n\t */\n\tsetEditorComponent(\n\t\tfactory: ((tui: TUI, theme: EditorTheme, keybindings: KeybindingsManager) => EditorComponent) | undefined,\n\t): void;\n\n\t/** Get the current theme for styling. */\n\treadonly theme: Theme;\n\n\t/** Get all available themes with their names and file paths. */\n\tgetAllThemes(): { name: string; path: string | undefined }[];\n\n\t/** Load a theme by name without switching to it. Returns undefined if not found. */\n\tgetTheme(name: string): Theme | undefined;\n\n\t/** Set the current theme by name or Theme object. */\n\tsetTheme(theme: string | Theme): { success: boolean; error?: string };\n\n\t/** Get current tool output expansion state. */\n\tgetToolsExpanded(): boolean;\n\n\t/** Set tool output expansion state. */\n\tsetToolsExpanded(expanded: boolean): void;\n}\n\n// ============================================================================\n// Extension Context\n// ============================================================================\n\nexport interface ContextUsage {\n\t/** Estimated context tokens, or null if unknown (e.g. right after compaction, before next LLM response). */\n\ttokens: number | null;\n\tcontextWindow: number;\n\t/** Context usage as percentage of context window, or null if tokens is unknown. */\n\tpercent: number | null;\n}\n\nexport interface CompactOptions {\n\tcustomInstructions?: string;\n\tonComplete?: (result: CompactionResult) => void;\n\tonError?: (error: Error) => void;\n}\n\n/**\n * Context passed to extension event handlers.\n */\nexport interface ExtensionContext {\n\t/** UI methods for user interaction */\n\tui: ExtensionUIContext;\n\t/** Whether UI is available (false in print/RPC mode) */\n\thasUI: boolean;\n\t/** Current working directory */\n\tcwd: string;\n\t/** Session manager (read-only) */\n\tsessionManager: ReadonlySessionManager;\n\t/** Model registry for API key resolution */\n\tmodelRegistry: ModelRegistry;\n\t/** Current model (may be undefined) */\n\tmodel: Model<any> | undefined;\n\t/** Whether the agent is idle (not streaming) */\n\tisIdle(): boolean;\n\t/** Abort the current agent operation */\n\tabort(): void;\n\t/** Whether there are queued messages waiting */\n\thasPendingMessages(): boolean;\n\t/** Gracefully shutdown pi and exit. Available in all contexts. */\n\tshutdown(): void;\n\t/** Get current context usage for the active model. */\n\tgetContextUsage(): ContextUsage | undefined;\n\t/** Trigger compaction without awaiting completion. */\n\tcompact(options?: CompactOptions): void;\n\t/** Get the current effective system prompt. */\n\tgetSystemPrompt(): string;\n}\n\n/**\n * Extended context for command handlers.\n * Includes session control methods only safe in user-initiated commands.\n */\nexport interface ExtensionCommandContext extends ExtensionContext {\n\t/** Wait for the agent to finish streaming */\n\twaitForIdle(): Promise<void>;\n\n\t/** Start a new session, optionally with initialization. */\n\tnewSession(options?: {\n\t\tparentSession?: string;\n\t\tsetup?: (sessionManager: SessionManager) => Promise<void>;\n\t}): Promise<{ cancelled: boolean }>;\n\n\t/** Fork from a specific entry, creating a new session file. */\n\tfork(entryId: string): Promise<{ cancelled: boolean }>;\n\n\t/** Navigate to a different point in the session tree. */\n\tnavigateTree(\n\t\ttargetId: string,\n\t\toptions?: { summarize?: boolean; customInstructions?: string; replaceInstructions?: boolean; label?: string },\n\t): Promise<{ cancelled: boolean }>;\n\n\t/** Switch to a different session file. */\n\tswitchSession(sessionPath: string): Promise<{ cancelled: boolean }>;\n\n\t/** Reload extensions, skills, prompts, and themes. */\n\treload(): Promise<void>;\n}\n\n// ============================================================================\n// Tool Types\n// ============================================================================\n\n/** Rendering options for tool results */\nexport interface ToolRenderResultOptions {\n\t/** Whether the result view is expanded */\n\texpanded: boolean;\n\t/** Whether this is a partial/streaming result */\n\tisPartial: boolean;\n}\n\n/**\n * Tool compatibility metadata for provider-aware tool filtering (ADR-005 Phase 2).\n * Tools without compatibility metadata are assumed universally compatible.\n */\nexport interface ToolCompatibility {\n\t/** Tool produces image content in results (filtered for providers without imageToolResults) */\n\tproducesImages?: boolean;\n\t/** Tool requires schema features that some providers don't support (e.g., [\"patternProperties\"]) */\n\tschemaFeatures?: string[];\n\t/** Tool is effective only with models above a minimum capability threshold */\n\tminCapabilityTier?: \"light\" | \"standard\" | \"heavy\";\n}\n\n/**\n * Tool definition for registerTool().\n */\nexport interface ToolDefinition<TParams extends TSchema = TSchema, TDetails = unknown> {\n\t/** Tool name (used in LLM tool calls) */\n\tname: string;\n\t/** Human-readable label for UI */\n\tlabel: string;\n\t/** Description for LLM */\n\tdescription: string;\n\t/** Optional one-line snippet for the Available tools section in the default system prompt. Falls back to description when omitted. */\n\tpromptSnippet?: string;\n\t/** Optional guideline bullets appended to the default system prompt Guidelines section when this tool is active. */\n\tpromptGuidelines?: string[];\n\t/** Parameter schema (TypeBox) */\n\tparameters: TParams;\n\t/** Provider compatibility metadata (ADR-005). Omit for universally compatible tools. */\n\tcompatibility?: ToolCompatibility;\n\n\t/** Execute the tool. */\n\texecute(\n\t\ttoolCallId: string,\n\t\tparams: Static<TParams>,\n\t\tsignal: AbortSignal | undefined,\n\t\tonUpdate: AgentToolUpdateCallback<TDetails> | undefined,\n\t\tctx: ExtensionContext,\n\t): Promise<AgentToolResult<TDetails>>;\n\n\t/** Custom rendering for tool call display */\n\trenderCall?: (args: Static<TParams>, theme: Theme) => Component | undefined;\n\n\t/** Custom rendering for tool result display */\n\trenderResult?: (\n\t\tresult: AgentToolResult<TDetails>,\n\t\toptions: ToolRenderResultOptions,\n\t\ttheme: Theme,\n\t) => Component | undefined;\n}\n\n// ============================================================================\n// Resource Events\n// ============================================================================\n\n/** Fired after session_start to allow extensions to provide additional resource paths. */\nexport interface ResourcesDiscoverEvent {\n\ttype: \"resources_discover\";\n\tcwd: string;\n\treason: \"startup\" | \"reload\";\n}\n\n/** Result from resources_discover event handler */\nexport interface ResourcesDiscoverResult {\n\tskillPaths?: string[];\n\tpromptPaths?: string[];\n\tthemePaths?: string[];\n}\n\n// ============================================================================\n// Session Events\n// ============================================================================\n\n/** Fired before session manager creation to allow custom session directory resolution */\nexport interface SessionDirectoryEvent {\n\ttype: \"session_directory\";\n\tcwd: string;\n}\n\n/** Fired on initial session load */\nexport interface SessionStartEvent {\n\ttype: \"session_start\";\n}\n\n/** Fired before switching to another session (can be cancelled) */\nexport interface SessionBeforeSwitchEvent {\n\ttype: \"session_before_switch\";\n\treason: \"new\" | \"resume\";\n\ttargetSessionFile?: string;\n}\n\n/** Fired after switching to another session */\nexport interface SessionSwitchEvent {\n\ttype: \"session_switch\";\n\treason: \"new\" | \"resume\";\n\tpreviousSessionFile: string | undefined;\n}\n\n/** Fired before forking a session (can be cancelled) */\nexport interface SessionBeforeForkEvent {\n\ttype: \"session_before_fork\";\n\tentryId: string;\n}\n\n/** Fired after forking a session */\nexport interface SessionForkEvent {\n\ttype: \"session_fork\";\n\tpreviousSessionFile: string | undefined;\n}\n\n/** Fired before context compaction (can be cancelled or customized) */\nexport interface SessionBeforeCompactEvent {\n\ttype: \"session_before_compact\";\n\tpreparation: CompactionPreparation;\n\tbranchEntries: SessionEntry[];\n\tcustomInstructions?: string;\n\tsignal: AbortSignal;\n}\n\n/** Fired after context compaction */\nexport interface SessionCompactEvent {\n\ttype: \"session_compact\";\n\tcompactionEntry: CompactionEntry;\n\tfromExtension: boolean;\n}\n\n/** Fired on process exit */\nexport interface SessionShutdownEvent {\n\ttype: \"session_shutdown\";\n}\n\n/** Preparation data for tree navigation */\nexport interface TreePreparation {\n\ttargetId: string;\n\toldLeafId: string | null;\n\tcommonAncestorId: string | null;\n\tentriesToSummarize: SessionEntry[];\n\tuserWantsSummary: boolean;\n\t/** Custom instructions for summarization */\n\tcustomInstructions?: string;\n\t/** If true, customInstructions replaces the default prompt instead of being appended */\n\treplaceInstructions?: boolean;\n\t/** Label to attach to the branch summary entry */\n\tlabel?: string;\n}\n\n/** Fired before navigating in the session tree (can be cancelled) */\nexport interface SessionBeforeTreeEvent {\n\ttype: \"session_before_tree\";\n\tpreparation: TreePreparation;\n\tsignal: AbortSignal;\n}\n\n/** Fired after navigating in the session tree */\nexport interface SessionTreeEvent {\n\ttype: \"session_tree\";\n\tnewLeafId: string | null;\n\toldLeafId: string | null;\n\tsummaryEntry?: BranchSummaryEntry;\n\tfromExtension?: boolean;\n}\n\nexport type SessionEvent =\n\t| SessionDirectoryEvent\n\t| SessionStartEvent\n\t| SessionBeforeSwitchEvent\n\t| SessionSwitchEvent\n\t| SessionBeforeForkEvent\n\t| SessionForkEvent\n\t| SessionBeforeCompactEvent\n\t| SessionCompactEvent\n\t| SessionShutdownEvent\n\t| SessionBeforeTreeEvent\n\t| SessionTreeEvent;\n\n// ============================================================================\n// Agent Events\n// ============================================================================\n\n/** Fired before each LLM call. Can modify messages. */\nexport interface ContextEvent {\n\ttype: \"context\";\n\tmessages: AgentMessage[];\n}\n\n/** Fired before a provider request is sent. Can replace the payload. */\nexport interface BeforeProviderRequestEvent {\n\ttype: \"before_provider_request\";\n\tpayload: unknown;\n\t/** The resolved model for this request (provider, id, api, etc.) */\n\tmodel?: { provider: string; id: string; api?: string };\n}\n\n/** Fired after user submits prompt but before agent loop. */\nexport interface BeforeAgentStartEvent {\n\ttype: \"before_agent_start\";\n\tprompt: string;\n\timages?: ImageContent[];\n\tsystemPrompt: string;\n}\n\n/** Fired when an agent loop starts */\nexport interface AgentStartEvent {\n\ttype: \"agent_start\";\n}\n\n/** Fired when an agent loop ends */\nexport interface AgentEndEvent {\n\ttype: \"agent_end\";\n\tmessages: AgentMessage[];\n}\n\n/** Fired at the start of each turn */\nexport interface TurnStartEvent {\n\ttype: \"turn_start\";\n\tturnIndex: number;\n\ttimestamp: number;\n}\n\n/** Fired at the end of each turn */\nexport interface TurnEndEvent {\n\ttype: \"turn_end\";\n\tturnIndex: number;\n\tmessage: AgentMessage;\n\ttoolResults: ToolResultMessage[];\n}\n\n/** Fired when a message starts (user, assistant, or toolResult) */\nexport interface MessageStartEvent {\n\ttype: \"message_start\";\n\tmessage: AgentMessage;\n}\n\n/** Fired during assistant message streaming with token-by-token updates */\nexport interface MessageUpdateEvent {\n\ttype: \"message_update\";\n\tmessage: AgentMessage;\n\tassistantMessageEvent: AssistantMessageEvent;\n}\n\n/** Fired when a message ends */\nexport interface MessageEndEvent {\n\ttype: \"message_end\";\n\tmessage: AgentMessage;\n}\n\n/** Fired when a tool starts executing */\nexport interface ToolExecutionStartEvent {\n\ttype: \"tool_execution_start\";\n\ttoolCallId: string;\n\ttoolName: string;\n\targs: any;\n}\n\n/** Fired during tool execution with partial/streaming output */\nexport interface ToolExecutionUpdateEvent {\n\ttype: \"tool_execution_update\";\n\ttoolCallId: string;\n\ttoolName: string;\n\targs: any;\n\tpartialResult: any;\n}\n\n/** Fired when a tool finishes executing */\nexport interface ToolExecutionEndEvent {\n\ttype: \"tool_execution_end\";\n\ttoolCallId: string;\n\ttoolName: string;\n\tresult: any;\n\tisError: boolean;\n}\n\n// ============================================================================\n// Model Events\n// ============================================================================\n\nexport type ModelSelectSource = \"set\" | \"cycle\" | \"restore\";\n\n/** Fired when a new model is selected */\nexport interface ModelSelectEvent {\n\ttype: \"model_select\";\n\tmodel: Model<any>;\n\tpreviousModel: Model<any> | undefined;\n\tsource: ModelSelectSource;\n}\n\n/** Fired before model selection runs capability scoring. Extensions can override the selected model. */\nexport interface BeforeModelSelectEvent {\n\ttype: \"before_model_select\";\n\tunitType: string;\n\tunitId: string;\n\tclassification: { tier: string; reason: string; downgraded: boolean };\n\ttaskMetadata?: Record<string, unknown>;\n\teligibleModels: string[];\n\tphaseConfig?: { primary: string; fallbacks: string[] };\n}\n\n/** Result from before_model_select event handler. Return { modelId } to override selection. */\nexport interface BeforeModelSelectResult {\n\tmodelId: string;\n}\n\n/**\n * Fired after model selection to allow extensions to adjust the active tool set (ADR-005 Phase 4).\n * Extensions can add, remove, or reorder tools based on the selected model's provider capabilities.\n */\nexport interface AdjustToolSetEvent {\n\ttype: \"adjust_tool_set\";\n\t/** The selected model's API type */\n\tselectedModelApi: string;\n\t/** The selected model's provider */\n\tselectedModelProvider: string;\n\t/** The selected model ID */\n\tselectedModelId: string;\n\t/** Current active tool names */\n\tactiveToolNames: string[];\n\t/** Tools already filtered by provider compatibility */\n\tfilteredTools: string[];\n}\n\n/** Result from adjust_tool_set event handler. Return { toolNames } to override tool set. */\nexport interface AdjustToolSetResult {\n\t/** Replacement tool names. If omitted, the default filtering is used. */\n\ttoolNames?: string[];\n}\n\n// ============================================================================\n// User Bash Events\n// ============================================================================\n\n/**\n * Fired before the bash tool executes a shell command.\n * Extensions can return a transformed command string.\n * All registered handlers are called in order; each receives the output of the previous.\n */\nexport interface BashTransformEvent {\n\ttype: \"bash_transform\";\n\t/** The command string about to be executed */\n\tcommand: string;\n\t/** Current working directory */\n\tcwd: string;\n}\n\n/** Result from bash_transform event handler */\nexport interface BashTransformEventResult {\n\t/** Replacement command string. If omitted or empty, the original command is used. */\n\tcommand?: string;\n}\n\n/** Fired when user executes a bash command via ! or !! prefix */\nexport interface UserBashEvent {\n\ttype: \"user_bash\";\n\t/** The command to execute */\n\tcommand: string;\n\t/** True if !! prefix was used (excluded from LLM context) */\n\texcludeFromContext: boolean;\n\t/** Current working directory */\n\tcwd: string;\n}\n\n// ============================================================================\n// Input Events\n// ============================================================================\n\n/** Source of user input */\nexport type InputSource = \"interactive\" | \"rpc\" | \"extension\";\n\n/** Fired when user input is received, before agent processing */\nexport interface InputEvent {\n\ttype: \"input\";\n\t/** The input text */\n\ttext: string;\n\t/** Attached images, if any */\n\timages?: ImageContent[];\n\t/** Where the input came from */\n\tsource: InputSource;\n}\n\n/** Result from input event handler */\nexport type InputEventResult =\n\t| { action: \"continue\" }\n\t| { action: \"transform\"; text: string; images?: ImageContent[] }\n\t| { action: \"handled\" };\n\n// ============================================================================\n// Tool Events\n// ============================================================================\n\ninterface ToolCallEventBase {\n\ttype: \"tool_call\";\n\ttoolCallId: string;\n}\n\nexport interface BashToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"bash\";\n\tinput: BashToolInput;\n}\n\nexport interface ReadToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"read\";\n\tinput: ReadToolInput;\n}\n\nexport interface EditToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"edit\";\n\tinput: EditToolInput;\n}\n\nexport interface WriteToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"write\";\n\tinput: WriteToolInput;\n}\n\nexport interface GrepToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"grep\";\n\tinput: GrepToolInput;\n}\n\nexport interface FindToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"find\";\n\tinput: FindToolInput;\n}\n\nexport interface LsToolCallEvent extends ToolCallEventBase {\n\ttoolName: \"ls\";\n\tinput: LsToolInput;\n}\n\nexport interface CustomToolCallEvent extends ToolCallEventBase {\n\ttoolName: string;\n\tinput: Record<string, unknown>;\n}\n\n/** Fired before a tool executes. Can block. */\nexport type ToolCallEvent =\n\t| BashToolCallEvent\n\t| ReadToolCallEvent\n\t| EditToolCallEvent\n\t| WriteToolCallEvent\n\t| GrepToolCallEvent\n\t| FindToolCallEvent\n\t| LsToolCallEvent\n\t| CustomToolCallEvent;\n\ninterface ToolResultEventBase {\n\ttype: \"tool_result\";\n\ttoolCallId: string;\n\tinput: Record<string, unknown>;\n\tcontent: (TextContent | ImageContent)[];\n\tisError: boolean;\n}\n\nexport interface BashToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"bash\";\n\tdetails: BashToolDetails | undefined;\n}\n\nexport interface ReadToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"read\";\n\tdetails: ReadToolDetails | undefined;\n}\n\nexport interface EditToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"edit\";\n\tdetails: EditToolDetails | undefined;\n}\n\nexport interface WriteToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"write\";\n\tdetails: undefined;\n}\n\nexport interface GrepToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"grep\";\n\tdetails: GrepToolDetails | undefined;\n}\n\nexport interface FindToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"find\";\n\tdetails: FindToolDetails | undefined;\n}\n\nexport interface LsToolResultEvent extends ToolResultEventBase {\n\ttoolName: \"ls\";\n\tdetails: LsToolDetails | undefined;\n}\n\nexport interface CustomToolResultEvent extends ToolResultEventBase {\n\ttoolName: string;\n\tdetails: unknown;\n}\n\n/** Fired after a tool executes. Can modify result. */\nexport type ToolResultEvent =\n\t| BashToolResultEvent\n\t| ReadToolResultEvent\n\t| EditToolResultEvent\n\t| WriteToolResultEvent\n\t| GrepToolResultEvent\n\t| FindToolResultEvent\n\t| LsToolResultEvent\n\t| CustomToolResultEvent;\n\n/**\n * Type guard for narrowing ToolResultEvent by tool name.\n *\n * Built-in tools narrow automatically (no type params needed):\n * ```ts\n * if (isToolResultEventType(\"bash\", event)) {\n * event.details; // BashToolDetails | undefined\n * }\n * ```\n *\n * Custom tools require explicit type parameters:\n * ```ts\n * if (isToolResultEventType<\"my_tool\", MyDetails>(\"my_tool\", event)) {\n * event.details; // typed\n * }\n * ```\n */\nexport function isToolResultEventType(toolName: \"bash\", event: ToolResultEvent): event is BashToolResultEvent;\nexport function isToolResultEventType(toolName: \"read\", event: ToolResultEvent): event is ReadToolResultEvent;\nexport function isToolResultEventType(toolName: \"edit\", event: ToolResultEvent): event is EditToolResultEvent;\nexport function isToolResultEventType(toolName: \"write\", event: ToolResultEvent): event is WriteToolResultEvent;\nexport function isToolResultEventType(toolName: \"grep\", event: ToolResultEvent): event is GrepToolResultEvent;\nexport function isToolResultEventType(toolName: \"find\", event: ToolResultEvent): event is FindToolResultEvent;\nexport function isToolResultEventType(toolName: \"ls\", event: ToolResultEvent): event is LsToolResultEvent;\nexport function isToolResultEventType<TName extends string, TDetails = unknown>(\n\ttoolName: TName,\n\tevent: ToolResultEvent,\n): event is ToolResultEvent & { toolName: TName; details: TDetails };\nexport function isToolResultEventType(toolName: string, event: ToolResultEvent): boolean {\n\treturn event.toolName === toolName;\n}\n\n/**\n * Type guard for narrowing ToolCallEvent by tool name.\n *\n * Built-in tools narrow automatically (no type params needed):\n * ```ts\n * if (isToolCallEventType(\"bash\", event)) {\n * event.input.command; // string\n * }\n * ```\n *\n * Custom tools require explicit type parameters:\n * ```ts\n * if (isToolCallEventType<\"my_tool\", MyToolInput>(\"my_tool\", event)) {\n * event.input.action; // typed\n * }\n * ```\n *\n * Note: Direct narrowing via `event.toolName === \"bash\"` doesn't work because\n * CustomToolCallEvent.toolName is `string` which overlaps with all literals.\n */\nexport function isToolCallEventType(toolName: \"bash\", event: ToolCallEvent): event is BashToolCallEvent;\nexport function isToolCallEventType(toolName: \"read\", event: ToolCallEvent): event is ReadToolCallEvent;\nexport function isToolCallEventType(toolName: \"edit\", event: ToolCallEvent): event is EditToolCallEvent;\nexport function isToolCallEventType(toolName: \"write\", event: ToolCallEvent): event is WriteToolCallEvent;\nexport function isToolCallEventType(toolName: \"grep\", event: ToolCallEvent): event is GrepToolCallEvent;\nexport function isToolCallEventType(toolName: \"find\", event: ToolCallEvent): event is FindToolCallEvent;\nexport function isToolCallEventType(toolName: \"ls\", event: ToolCallEvent): event is LsToolCallEvent;\nexport function isToolCallEventType<TName extends string, TInput extends Record<string, unknown>>(\n\ttoolName: TName,\n\tevent: ToolCallEvent,\n): event is ToolCallEvent & { toolName: TName; input: TInput };\nexport function isToolCallEventType(toolName: string, event: ToolCallEvent): boolean {\n\treturn event.toolName === toolName;\n}\n\n/** Union of all event types */\nexport type ExtensionEvent =\n\t| ResourcesDiscoverEvent\n\t| SessionEvent\n\t| ContextEvent\n\t| BeforeProviderRequestEvent\n\t| BeforeAgentStartEvent\n\t| AgentStartEvent\n\t| AgentEndEvent\n\t| TurnStartEvent\n\t| TurnEndEvent\n\t| MessageStartEvent\n\t| MessageUpdateEvent\n\t| MessageEndEvent\n\t| ToolExecutionStartEvent\n\t| ToolExecutionUpdateEvent\n\t| ToolExecutionEndEvent\n\t| ModelSelectEvent\n\t| BashTransformEvent\n\t| UserBashEvent\n\t| InputEvent\n\t| ToolCallEvent\n\t| ToolResultEvent;\n\n// ============================================================================\n// Event Results\n// ============================================================================\n\nexport interface ContextEventResult {\n\tmessages?: AgentMessage[];\n}\n\nexport type BeforeProviderRequestEventResult = unknown;\n\nexport interface ToolCallEventResult {\n\tblock?: boolean;\n\treason?: string;\n}\n\n/** Result from user_bash event handler */\nexport interface UserBashEventResult {\n\t/** Custom operations to use for execution */\n\toperations?: BashOperations;\n\t/** Full replacement: extension handled execution, use this result */\n\tresult?: BashResult;\n}\n\nexport interface ToolResultEventResult {\n\tcontent?: (TextContent | ImageContent)[];\n\tdetails?: unknown;\n\tisError?: boolean;\n}\n\nexport interface BeforeAgentStartEventResult {\n\tmessage?: Pick<CustomMessage, \"customType\" | \"content\" | \"display\" | \"details\">;\n\t/** Replace the system prompt for this turn. If multiple extensions return this, they are chained. */\n\tsystemPrompt?: string;\n}\n\nexport interface SessionDirectoryResult {\n\t/** Custom session directory path. If multiple extensions return this, the last one wins. */\n\tsessionDir?: string;\n}\n\n/** Special startup-only handler. Unlike other events, this receives no ExtensionContext. */\nexport type SessionDirectoryHandler = (\n\tevent: SessionDirectoryEvent,\n) => Promise<SessionDirectoryResult | undefined> | SessionDirectoryResult | undefined;\n\nexport interface SessionBeforeSwitchResult {\n\tcancel?: boolean;\n}\n\nexport interface SessionBeforeForkResult {\n\tcancel?: boolean;\n\tskipConversationRestore?: boolean;\n}\n\nexport interface SessionBeforeCompactResult {\n\tcancel?: boolean;\n\tcompaction?: CompactionResult;\n}\n\nexport interface SessionBeforeTreeResult {\n\tcancel?: boolean;\n\tsummary?: {\n\t\tsummary: string;\n\t\tdetails?: unknown;\n\t};\n\t/** Override custom instructions for summarization */\n\tcustomInstructions?: string;\n\t/** Override whether customInstructions replaces the default prompt */\n\treplaceInstructions?: boolean;\n\t/** Override label to attach to the branch summary entry */\n\tlabel?: string;\n}\n\n// ============================================================================\n// Message Rendering\n// ============================================================================\n\nexport interface MessageRenderOptions {\n\texpanded: boolean;\n}\n\nexport type MessageRenderer<T = unknown> = (\n\tmessage: CustomMessage<T>,\n\toptions: MessageRenderOptions,\n\ttheme: Theme,\n) => Component | undefined;\n\n// ============================================================================\n// Command Registration\n// ============================================================================\n\nexport interface RegisteredCommand {\n\tname: string;\n\tdescription?: string;\n\tgetArgumentCompletions?: (argumentPrefix: string) => AutocompleteItem[] | null;\n\thandler: (args: string, ctx: ExtensionCommandContext) => Promise<void>;\n}\n\nexport type LifecycleHookScope = \"user\" | \"project\";\nexport type LifecycleHookPhase = \"beforeInstall\" | \"afterInstall\" | \"beforeRemove\" | \"afterRemove\";\n\nexport interface LifecycleHookContext {\n\t/** Lifecycle phase currently being executed. */\n\tphase: LifecycleHookPhase;\n\t/** Package source string passed to install (npm:, git:, https://, local path). */\n\tsource: string;\n\t/** Resolved installed package path (or resolved local path), when available for this phase. */\n\tinstalledPath?: string;\n\t/** Where the package was installed. */\n\tscope: LifecycleHookScope;\n\t/** Current working directory for the install invocation. */\n\tcwd: string;\n\t/** Whether install is running in an interactive TTY. */\n\tinteractive: boolean;\n\t/** Info-level logging sink for install output. */\n\tlog(message: string): void;\n\t/** Warning-level logging sink for install output. */\n\twarn(message: string): void;\n\t/** Error-level logging sink for install output. */\n\terror(message: string): void;\n}\n\nexport type LifecycleHookHandler = (ctx: LifecycleHookContext) => Promise<void> | void;\nexport type LifecycleHookMap = Record<LifecycleHookPhase, LifecycleHookHandler[]>;\n\n// ============================================================================\n// Extension API\n// ============================================================================\n\n/** Handler function type for events */\n// biome-ignore lint/suspicious/noConfusingVoidType: void allows bare return statements\nexport type ExtensionHandler<E, R = undefined> = (event: E, ctx: ExtensionContext) => Promise<R | void> | R | void;\n\n/**\n * ExtensionAPI passed to extension factory functions.\n */\nexport interface ExtensionAPI {\n\t// =========================================================================\n\t// Event Subscription\n\t// =========================================================================\n\n\ton(event: \"resources_discover\", handler: ExtensionHandler<ResourcesDiscoverEvent, ResourcesDiscoverResult>): void;\n\ton(event: \"session_directory\", handler: SessionDirectoryHandler): void;\n\ton(event: \"session_start\", handler: ExtensionHandler<SessionStartEvent>): void;\n\ton(\n\t\tevent: \"session_before_switch\",\n\t\thandler: ExtensionHandler<SessionBeforeSwitchEvent, SessionBeforeSwitchResult>,\n\t): void;\n\ton(event: \"session_switch\", handler: ExtensionHandler<SessionSwitchEvent>): void;\n\ton(event: \"session_before_fork\", handler: ExtensionHandler<SessionBeforeForkEvent, SessionBeforeForkResult>): void;\n\ton(event: \"session_fork\", handler: ExtensionHandler<SessionForkEvent>): void;\n\ton(\n\t\tevent: \"session_before_compact\",\n\t\thandler: ExtensionHandler<SessionBeforeCompactEvent, SessionBeforeCompactResult>,\n\t): void;\n\ton(event: \"session_compact\", handler: ExtensionHandler<SessionCompactEvent>): void;\n\ton(event: \"session_shutdown\", handler: ExtensionHandler<SessionShutdownEvent>): void;\n\ton(event: \"session_before_tree\", handler: ExtensionHandler<SessionBeforeTreeEvent, SessionBeforeTreeResult>): void;\n\ton(event: \"session_tree\", handler: ExtensionHandler<SessionTreeEvent>): void;\n\ton(event: \"context\", handler: ExtensionHandler<ContextEvent, ContextEventResult>): void;\n\ton(\n\t\tevent: \"before_provider_request\",\n\t\thandler: ExtensionHandler<BeforeProviderRequestEvent, BeforeProviderRequestEventResult>,\n\t): void;\n\ton(event: \"before_agent_start\", handler: ExtensionHandler<BeforeAgentStartEvent, BeforeAgentStartEventResult>): void;\n\ton(event: \"agent_start\", handler: ExtensionHandler<AgentStartEvent>): void;\n\ton(event: \"agent_end\", handler: ExtensionHandler<AgentEndEvent>): void;\n\ton(event: \"turn_start\", handler: ExtensionHandler<TurnStartEvent>): void;\n\ton(event: \"turn_end\", handler: ExtensionHandler<TurnEndEvent>): void;\n\ton(event: \"message_start\", handler: ExtensionHandler<MessageStartEvent>): void;\n\ton(event: \"message_update\", handler: ExtensionHandler<MessageUpdateEvent>): void;\n\ton(event: \"message_end\", handler: ExtensionHandler<MessageEndEvent>): void;\n\ton(event: \"tool_execution_start\", handler: ExtensionHandler<ToolExecutionStartEvent>): void;\n\ton(event: \"tool_execution_update\", handler: ExtensionHandler<ToolExecutionUpdateEvent>): void;\n\ton(event: \"tool_execution_end\", handler: ExtensionHandler<ToolExecutionEndEvent>): void;\n\ton(event: \"model_select\", handler: ExtensionHandler<ModelSelectEvent>): void;\n\ton(event: \"bash_transform\", handler: ExtensionHandler<BashTransformEvent, BashTransformEventResult>): void;\n\ton(event: \"tool_call\", handler: ExtensionHandler<ToolCallEvent, ToolCallEventResult>): void;\n\ton(event: \"tool_result\", handler: ExtensionHandler<ToolResultEvent, ToolResultEventResult>): void;\n\ton(event: \"user_bash\", handler: ExtensionHandler<UserBashEvent, UserBashEventResult>): void;\n\ton(event: \"input\", handler: ExtensionHandler<InputEvent, InputEventResult>): void;\n\ton(event: \"before_model_select\", handler: ExtensionHandler<BeforeModelSelectEvent, BeforeModelSelectResult>): void;\n\ton(event: \"adjust_tool_set\", handler: ExtensionHandler<AdjustToolSetEvent, AdjustToolSetResult>): void;\n\n\t// =========================================================================\n\t// Event Emission (for host extensions that orchestrate model selection)\n\t// =========================================================================\n\n\t/** Emit before_model_select event. Returns override model ID or undefined. */\n\temitBeforeModelSelect(event: Omit<BeforeModelSelectEvent, \"type\">): Promise<BeforeModelSelectResult | undefined>;\n\n\t/** Emit adjust_tool_set event (ADR-005). Returns override tool names or undefined. */\n\temitAdjustToolSet(event: Omit<AdjustToolSetEvent, \"type\">): Promise<AdjustToolSetResult | undefined>;\n\n\t// =========================================================================\n\t// Tool Registration\n\t// =========================================================================\n\n\t/** Register a tool that the LLM can call. */\n\tregisterTool<TParams extends TSchema = TSchema, TDetails = unknown>(tool: ToolDefinition<TParams, TDetails>): void;\n\n\t// =========================================================================\n\t// Command, Shortcut, Flag Registration\n\t// =========================================================================\n\n\t/** Register a custom command. */\n\tregisterCommand(name: string, options: Omit<RegisteredCommand, \"name\">): void;\n\n\t/** Register a lifecycle hook run before package installation starts. */\n\tregisterBeforeInstall(handler: LifecycleHookHandler): void;\n\n\t/** Register a lifecycle hook run after package installation completes. */\n\tregisterAfterInstall(handler: LifecycleHookHandler): void;\n\n\t/** Register a lifecycle hook run before package removal starts. */\n\tregisterBeforeRemove(handler: LifecycleHookHandler): void;\n\n\t/** Register a lifecycle hook run after package removal completes. */\n\tregisterAfterRemove(handler: LifecycleHookHandler): void;\n\n\t/** Register a keyboard shortcut. */\n\tregisterShortcut(\n\t\tshortcut: KeyId,\n\t\toptions: {\n\t\t\tdescription?: string;\n\t\t\thandler: (ctx: ExtensionContext) => Promise<void> | void;\n\t\t},\n\t): void;\n\n\t/** Register a CLI flag. */\n\tregisterFlag(\n\t\tname: string,\n\t\toptions: {\n\t\t\tdescription?: string;\n\t\t\ttype: \"boolean\" | \"string\";\n\t\t\tdefault?: boolean | string;\n\t\t},\n\t): void;\n\n\t/** Get the value of a registered CLI flag. */\n\tgetFlag(name: string): boolean | string | undefined;\n\n\t// =========================================================================\n\t// Message Rendering\n\t// =========================================================================\n\n\t/** Register a custom renderer for CustomMessageEntry. */\n\tregisterMessageRenderer<T = unknown>(customType: string, renderer: MessageRenderer<T>): void;\n\n\t// =========================================================================\n\t// Actions\n\t// =========================================================================\n\n\t/** Send a custom message to the session. */\n\tsendMessage<T = unknown>(\n\t\tmessage: Pick<CustomMessage<T>, \"customType\" | \"content\" | \"display\" | \"details\">,\n\t\toptions?: { triggerTurn?: boolean; deliverAs?: \"steer\" | \"followUp\" | \"nextTurn\" },\n\t): void;\n\n\t/**\n\t * Send a user message to the agent. Always triggers a turn.\n\t * When the agent is streaming, use deliverAs to specify how to queue the message.\n\t */\n\tsendUserMessage(\n\t\tcontent: string | (TextContent | ImageContent)[],\n\t\toptions?: { deliverAs?: \"steer\" | \"followUp\" },\n\t): void;\n\n\t/**\n\t * Retry the last turn by removing the failed assistant response and\n\t * re-running the agent from the last user message. No-op if the last\n\t * message is not an assistant error.\n\t */\n\tretryLastTurn(): void;\n\n\t/** Append a custom entry to the session for state persistence (not sent to LLM). */\n\tappendEntry<T = unknown>(customType: string, data?: T): void;\n\n\t// =========================================================================\n\t// Session Metadata\n\t// =========================================================================\n\n\t/** Set the session display name (shown in session selector). */\n\tsetSessionName(name: string): void;\n\n\t/** Get the current session name, if set. */\n\tgetSessionName(): string | undefined;\n\n\t/** Set or clear a label on an entry. Labels are user-defined markers for bookmarking/navigation. */\n\tsetLabel(entryId: string, label: string | undefined): void;\n\n\t/** Execute a shell command. */\n\texec(command: string, args: string[], options?: ExecOptions): Promise<ExecResult>;\n\n\t/** Get the list of currently active tool names. */\n\tgetActiveTools(): string[];\n\n\t/** Get all configured tools with name and description. */\n\tgetAllTools(): ToolInfo[];\n\n\t/** Set the active tools by name. */\n\tsetActiveTools(toolNames: string[]): void;\n\n\t/** Get available slash commands in the current session. */\n\tgetCommands(): SlashCommandInfo[];\n\n\t// =========================================================================\n\t// Model and Thinking Level\n\t// =========================================================================\n\n\t/** Set the current model. Returns false if no API key available. */\n\tsetModel(model: Model<any>, options?: { persist?: boolean }): Promise<boolean>;\n\n\t/** Get current thinking level. */\n\tgetThinkingLevel(): ThinkingLevel;\n\n\t/** Set thinking level (clamped to model capabilities). */\n\tsetThinkingLevel(level: ThinkingLevel): void;\n\n\t// =========================================================================\n\t// Provider Registration\n\t// =========================================================================\n\n\t/**\n\t * Register or override a model provider.\n\t *\n\t * If `models` is provided: replaces all existing models for this provider.\n\t * If only `baseUrl` is provided: overrides the URL for existing models.\n\t * If `oauth` is provided: registers OAuth provider for /login support.\n\t * If `streamSimple` is provided: registers a custom API stream handler.\n\t *\n\t * During initial extension load this call is queued and applied once the\n\t * runner has bound its context. After that it takes effect immediately, so\n\t * it is safe to call from command handlers or event callbacks without\n\t * requiring a `/reload`.\n\t *\n\t * @example\n\t * // Register a new provider with custom models\n\t * pi.registerProvider(\"my-proxy\", {\n\t * baseUrl: \"https://proxy.example.com\",\n\t * apiKey: \"PROXY_API_KEY\",\n\t * api: \"anthropic-messages\",\n\t * models: [\n\t * {\n\t * id: \"claude-sonnet-4-20250514\",\n\t * name: \"Claude 4 Sonnet (proxy)\",\n\t * reasoning: false,\n\t * input: [\"text\", \"image\"],\n\t * cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },\n\t * contextWindow: 200000,\n\t * maxTokens: 16384\n\t * }\n\t * ]\n\t * });\n\t *\n\t * @example\n\t * // Override baseUrl for an existing provider\n\t * pi.registerProvider(\"anthropic\", {\n\t * baseUrl: \"https://proxy.example.com\"\n\t * });\n\t *\n\t * @example\n\t * // Register provider with OAuth support\n\t * pi.registerProvider(\"corporate-ai\", {\n\t * baseUrl: \"https://ai.corp.com\",\n\t * api: \"openai-responses\",\n\t * models: [...],\n\t * oauth: {\n\t * name: \"Corporate AI (SSO)\",\n\t * async login(callbacks) { ... },\n\t * async refreshToken(credentials) { ... },\n\t * getApiKey(credentials) { return credentials.access; }\n\t * }\n\t * });\n\t */\n\tregisterProvider(name: string, config: ProviderConfig): void;\n\n\t/**\n\t * Unregister a previously registered provider.\n\t *\n\t * Removes all models belonging to the named provider and restores any\n\t * built-in models that were overridden by it. Has no effect if the provider\n\t * is not currently registered.\n\t *\n\t * Like `registerProvider`, this takes effect immediately when called after\n\t * the initial load phase.\n\t *\n\t * @example\n\t * pi.unregisterProvider(\"my-proxy\");\n\t */\n\tunregisterProvider(name: string): void;\n\n\t/** Shared event bus for extension communication. */\n\tevents: EventBus;\n}\n\n// ============================================================================\n// Provider Registration Types\n// ============================================================================\n\n/** Configuration for registering a provider via pi.registerProvider(). */\nexport interface ProviderConfig {\n\t/** Auth behavior for provider availability and request key handling. Defaults to \"apiKey\". */\n\tauthMode?: \"apiKey\" | \"oauth\" | \"externalCli\" | \"none\";\n\t/** Optional readiness check. Return false if the provider cannot accept requests (e.g., CLI not authenticated, API key invalid).\n\t * Called before default auth checks. Trusted at the same level as extension code — extensions already have arbitrary code execution. */\n\tisReady?: () => boolean;\n\t/** Base URL for the API endpoint. Required when defining models. */\n\tbaseUrl?: string;\n\t/** API key or environment variable name. Required when defining models (unless oauth provided). */\n\tapiKey?: string;\n\t/** API type. Required at provider or model level when defining models. */\n\tapi?: Api;\n\t/** Optional streamSimple handler for custom APIs. */\n\tstreamSimple?: (model: Model<Api>, context: Context, options?: SimpleStreamOptions) => AssistantMessageEventStream;\n\t/** Custom headers to include in requests. */\n\theaders?: Record<string, string>;\n\t/** If true, adds Authorization: Bearer header with the resolved API key. */\n\tauthHeader?: boolean;\n\t/** Models to register. If provided, replaces all existing models for this provider. */\n\tmodels?: ProviderModelConfig[];\n\t/** OAuth provider for /login support. The `id` is set automatically from the provider name. */\n\toauth?: {\n\t\t/** Display name for the provider in login UI. */\n\t\tname: string;\n\t\t/** Run the login flow, return credentials to persist. */\n\t\tlogin(callbacks: OAuthLoginCallbacks): Promise<OAuthCredentials>;\n\t\t/** Refresh expired credentials, return updated credentials to persist. */\n\t\trefreshToken(credentials: OAuthCredentials): Promise<OAuthCredentials>;\n\t\t/** Convert credentials to API key string for the provider. */\n\t\tgetApiKey(credentials: OAuthCredentials): string;\n\t\t/** Optional: modify models for this provider (e.g., update baseUrl based on credentials). */\n\t\tmodifyModels?(models: Model<Api>[], credentials: OAuthCredentials): Model<Api>[];\n\t};\n}\n\n/** Configuration for a model within a provider. */\nexport interface ProviderModelConfig {\n\t/** Model ID (e.g., \"claude-sonnet-4-20250514\"). */\n\tid: string;\n\t/** Display name (e.g., \"Claude 4 Sonnet\"). */\n\tname: string;\n\t/** API type override for this model. */\n\tapi?: Api;\n\t/** Whether the model supports extended thinking. */\n\treasoning: boolean;\n\t/** Supported input types. */\n\tinput: (\"text\" | \"image\")[];\n\t/** Cost per token (for tracking, can be 0). */\n\tcost: { input: number; output: number; cacheRead: number; cacheWrite: number };\n\t/** Maximum context window size in tokens. */\n\tcontextWindow: number;\n\t/** Maximum output tokens. */\n\tmaxTokens: number;\n\t/** Custom headers for this model. */\n\theaders?: Record<string, string>;\n\t/** OpenAI compatibility settings. */\n\tcompat?: Model<Api>[\"compat\"];\n\t/** Opaque provider-specific options (e.g. Ollama keep_alive, num_gpu). */\n\tproviderOptions?: Record<string, unknown>;\n}\n\n/** Extension factory function type. Supports both sync and async initialization. */\nexport type ExtensionFactory = (pi: ExtensionAPI) => void | Promise<void>;\n\n// ============================================================================\n// Loaded Extension Types\n// ============================================================================\n\nexport interface RegisteredTool {\n\tdefinition: ToolDefinition;\n\textensionPath: string;\n}\n\nexport interface ExtensionFlag {\n\tname: string;\n\tdescription?: string;\n\ttype: \"boolean\" | \"string\";\n\tdefault?: boolean | string;\n\textensionPath: string;\n}\n\nexport interface ExtensionShortcut {\n\tshortcut: KeyId;\n\tdescription?: string;\n\thandler: (ctx: ExtensionContext) => Promise<void> | void;\n\textensionPath: string;\n}\n\ntype HandlerFn = (...args: unknown[]) => Promise<unknown>;\n\n/** Tool info with name, description, and parameter schema */\nexport type ToolInfo = Pick<ToolDefinition, \"name\" | \"description\" | \"parameters\">;\n\n/**\n * Shared state created by loader, used during registration and runtime.\n * Contains flag values (defaults set during registration, CLI values set after).\n */\nexport interface ExtensionRuntimeState {\n\tflagValues: Map<string, boolean | string>;\n\t/** Provider registrations queued during extension loading, processed when runner binds */\n\tpendingProviderRegistrations: Array<{ name: string; config: ProviderConfig }>;\n\t/**\n\t * Register or unregister a provider.\n\t *\n\t * Before bindCore(): queues registrations / removes from queue.\n\t * After bindCore(): calls ModelRegistry directly for immediate effect.\n\t */\n\tregisterProvider: (name: string, config: ProviderConfig) => void;\n\tunregisterProvider: (name: string) => void;\n\t/** Emit before_model_select event to all registered handlers. Bound by ExtensionRunner. */\n\temitBeforeModelSelect: (event: Omit<BeforeModelSelectEvent, \"type\">) => Promise<BeforeModelSelectResult | undefined>;\n\t/** Emit adjust_tool_set event to all registered handlers. Bound by ExtensionRunner (ADR-005). */\n\temitAdjustToolSet: (event: Omit<AdjustToolSetEvent, \"type\">) => Promise<AdjustToolSetResult | undefined>;\n}\n\n/**\n * Action implementations for pi.* API methods.\n * Provided to runner.initialize(), copied into the shared runtime.\n */\nexport interface ExtensionActions {\n\tsendMessage: <T = unknown>(\n\t\tmessage: Pick<CustomMessage<T>, \"customType\" | \"content\" | \"display\" | \"details\">,\n\t\toptions?: { triggerTurn?: boolean; deliverAs?: \"steer\" | \"followUp\" | \"nextTurn\" },\n\t) => void;\n\tsendUserMessage: (\n\t\tcontent: string | (TextContent | ImageContent)[],\n\t\toptions?: { deliverAs?: \"steer\" | \"followUp\" },\n\t) => void;\n\tretryLastTurn: () => void;\n\tappendEntry: <T = unknown>(customType: string, data?: T) => void;\n\tsetSessionName: (name: string) => void;\n\tgetSessionName: () => string | undefined;\n\tsetLabel: (entryId: string, label: string | undefined) => void;\n\tgetActiveTools: () => string[];\n\tgetAllTools: () => ToolInfo[];\n\tsetActiveTools: (toolNames: string[]) => void;\n\trefreshTools: () => void;\n\tgetCommands: () => SlashCommandInfo[];\n\tsetModel: (model: Model<any>, options?: { persist?: boolean }) => Promise<boolean>;\n\tgetThinkingLevel: () => ThinkingLevel;\n\tsetThinkingLevel: (level: ThinkingLevel) => void;\n}\n\n/**\n * Actions for ExtensionContext (ctx.* in event handlers).\n * Required by all modes.\n */\nexport interface ExtensionContextActions {\n\tgetModel: () => Model<any> | undefined;\n\tisIdle: () => boolean;\n\tabort: () => void;\n\thasPendingMessages: () => boolean;\n\tshutdown: () => void;\n\tgetContextUsage: () => ContextUsage | undefined;\n\tcompact: (options?: CompactOptions) => void;\n\tgetSystemPrompt: () => string;\n}\n\n/**\n * Actions for ExtensionCommandContext (ctx.* in command handlers).\n * Only needed for interactive mode where extension commands are invokable.\n */\nexport interface ExtensionCommandContextActions {\n\twaitForIdle: () => Promise<void>;\n\tnewSession: (options?: {\n\t\tparentSession?: string;\n\t\tsetup?: (sessionManager: SessionManager) => Promise<void>;\n\t}) => Promise<{ cancelled: boolean }>;\n\tfork: (entryId: string) => Promise<{ cancelled: boolean }>;\n\tnavigateTree: (\n\t\ttargetId: string,\n\t\toptions?: { summarize?: boolean; customInstructions?: string; replaceInstructions?: boolean; label?: string },\n\t) => Promise<{ cancelled: boolean }>;\n\tswitchSession: (sessionPath: string) => Promise<{ cancelled: boolean }>;\n\treload: () => Promise<void>;\n}\n\n/**\n * Full runtime = state + actions.\n * Created by loader with throwing action stubs, completed by runner.initialize().\n */\nexport interface ExtensionRuntime extends ExtensionRuntimeState, ExtensionActions {}\n\n/** Loaded extension with all registered items. */\nexport interface Extension {\n\tpath: string;\n\tresolvedPath: string;\n\thandlers: Map<string, HandlerFn[]>;\n\ttools: Map<string, RegisteredTool>;\n\tmessageRenderers: Map<string, MessageRenderer>;\n\tcommands: Map<string, RegisteredCommand>;\n\tflags: Map<string, ExtensionFlag>;\n\tshortcuts: Map<KeyId, ExtensionShortcut>;\n\tlifecycleHooks: LifecycleHookMap;\n}\n\n/** Result of loading extensions. */\nexport interface LoadExtensionsResult {\n\textensions: Extension[];\n\terrors: Array<{ path: string; error: string }>;\n\t/** Shared runtime - actions are throwing stubs until runner.initialize() */\n\truntime: ExtensionRuntime;\n}\n\n// ============================================================================\n// Extension Error\n// ============================================================================\n\nexport interface ExtensionError {\n\textensionPath: string;\n\tevent: string;\n\terror: string;\n\tstack?: string;\n}\n"]}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import assert from "node:assert/strict";
|
|
2
2
|
import { describe, it } from "node:test";
|
|
3
3
|
import { getApiProvider } from "@gsd/pi-ai";
|
|
4
|
+
import { AuthStorage } from "./auth-storage.js";
|
|
4
5
|
import { ModelRegistry } from "./model-registry.js";
|
|
5
6
|
function createRegistry(hasAuthFn) {
|
|
6
7
|
const authStorage = {
|
|
@@ -13,6 +14,9 @@ function createRegistry(hasAuthFn) {
|
|
|
13
14
|
};
|
|
14
15
|
return new ModelRegistry(authStorage, undefined);
|
|
15
16
|
}
|
|
17
|
+
function createInMemoryRegistry(data = {}) {
|
|
18
|
+
return new ModelRegistry(AuthStorage.inMemory(data), undefined);
|
|
19
|
+
}
|
|
16
20
|
function createProviderModel(id, api) {
|
|
17
21
|
return {
|
|
18
22
|
id,
|
|
@@ -348,6 +352,34 @@ describe("ModelRegistry authMode — getAvailable", () => {
|
|
|
348
352
|
const available = registry.getAvailable();
|
|
349
353
|
assert.equal(available.length, 0);
|
|
350
354
|
});
|
|
355
|
+
it("prunes Codex models removed from ChatGPT-backed openai-codex OAuth", () => {
|
|
356
|
+
const registry = createInMemoryRegistry({
|
|
357
|
+
"openai-codex": {
|
|
358
|
+
type: "oauth",
|
|
359
|
+
access: "oauth-access",
|
|
360
|
+
refresh: "oauth-refresh",
|
|
361
|
+
expires: Date.now() + 60_000,
|
|
362
|
+
accountId: "acct_123",
|
|
363
|
+
},
|
|
364
|
+
});
|
|
365
|
+
assert.equal(registry.find("openai-codex", "gpt-5.1-codex-max"), undefined);
|
|
366
|
+
assert.equal(registry.find("openai-codex", "gpt-5.1"), undefined);
|
|
367
|
+
assert.equal(findModel(registry, "openai-codex", "gpt-5.2-codex"), undefined);
|
|
368
|
+
assert.ok(registry.find("openai-codex", "gpt-5.4"));
|
|
369
|
+
assert.ok(findModel(registry, "openai-codex", "gpt-5.4"));
|
|
370
|
+
assert.ok(registry.find("openai-codex", "gpt-5.4-mini"));
|
|
371
|
+
assert.ok(findModel(registry, "openai-codex", "gpt-5.4-mini"));
|
|
372
|
+
});
|
|
373
|
+
it("keeps API-backed OpenAI Codex-capable models available", () => {
|
|
374
|
+
const registry = createInMemoryRegistry({
|
|
375
|
+
openai: {
|
|
376
|
+
type: "api_key",
|
|
377
|
+
key: "sk-test",
|
|
378
|
+
},
|
|
379
|
+
});
|
|
380
|
+
assert.ok(registry.find("openai", "gpt-5.2-codex"));
|
|
381
|
+
assert.ok(findModel(registry, "openai", "gpt-5.2-codex"));
|
|
382
|
+
});
|
|
351
383
|
});
|
|
352
384
|
// ─── getApiKey ────────────────────────────────────────────────────────────────
|
|
353
385
|
describe("ModelRegistry authMode — getApiKey", () => {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"model-registry-auth-mode.test.js","sourceRoot":"","sources":["../../src/core/model-registry-auth-mode.test.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from \"node:assert/strict\";\nimport { describe, it } from \"node:test\";\nimport type { Api, Model, SimpleStreamOptions, Context, AssistantMessageEventStream } from \"@gsd/pi-ai\";\nimport { getApiProvider } from \"@gsd/pi-ai\";\nimport type { AuthStorage } from \"./auth-storage.js\";\nimport { ModelRegistry } from \"./model-registry.js\";\n\nfunction createRegistry(hasAuthFn?: (provider: string) => boolean): ModelRegistry {\n\tconst authStorage = {\n\t\tsetFallbackResolver: () => {},\n\t\tonCredentialChange: () => {},\n\t\tgetOAuthProviders: () => [],\n\t\tget: () => undefined,\n\t\thasAuth: hasAuthFn ?? (() => false),\n\t\tgetApiKey: async () => undefined,\n\t} as unknown as AuthStorage;\n\n\treturn new ModelRegistry(authStorage, undefined);\n}\n\nfunction createProviderModel(id: string, api?: string): NonNullable<Parameters<ModelRegistry[\"registerProvider\"]>[1][\"models\"]>[number] {\n\treturn {\n\t\tid,\n\t\tname: id,\n\t\tapi: (api ?? \"openai-completions\") as Api,\n\t\treasoning: false,\n\t\tinput: [\"text\"],\n\t\tcost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },\n\t\tcontextWindow: 128000,\n\t\tmaxTokens: 16384,\n\t};\n}\n\nfunction findModel(registry: ModelRegistry, provider: string, id: string): Model<Api> | undefined {\n\treturn registry.getAvailable().find((m) => m.provider === provider && m.id === id);\n}\n\nfunction makeModel(provider: string, id: string, api: string): Model<Api> {\n\treturn {\n\t\tid,\n\t\tname: id,\n\t\tapi: api as Api,\n\t\tprovider,\n\t\tbaseUrl: `${provider}:`,\n\t\treasoning: false,\n\t\tinput: [\"text\"],\n\t\tcost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },\n\t\tcontextWindow: 128000,\n\t\tmaxTokens: 16384,\n\t};\n}\n\nfunction makeContext(): Context {\n\treturn {\n\t\tsystemPrompt: \"test\",\n\t\tmessages: [{ role: \"user\", content: \"hello\", timestamp: Date.now() }],\n\t};\n}\n\n/** No-op streamSimple for tests that need one to pass validation but don't inspect it. */\nconst noopStreamSimple = (_model: Model<Api>, _context: Context, _options?: SimpleStreamOptions) => {\n\treturn {\n\t\t[Symbol.asyncIterator]() { return { next: async () => ({ value: undefined, done: true as const }) }; },\n\t\tresult: () => Promise.resolve({ role: \"assistant\" as const, content: [], api: \"test\" as Api, provider: \"test\", model: \"test\", usage: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, totalTokens: 0, cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 } }, stopReason: \"stop\" as const, timestamp: Date.now() }),\n\t\tpush: () => {},\n\t\tend: () => {},\n\t} as unknown as AssistantMessageEventStream;\n};\n\n/** Create a spy streamSimple that captures the options it receives and returns a stub stream. */\nfunction createStreamSpy(): {\n\tstreamSimple: (model: Model<Api>, context: Context, options?: SimpleStreamOptions) => AssistantMessageEventStream;\n\tgetCapturedOptions: () => SimpleStreamOptions | undefined;\n} {\n\tlet capturedOptions: SimpleStreamOptions | undefined;\n\tconst streamSimple = (_model: Model<Api>, _context: Context, options?: SimpleStreamOptions) => {\n\t\tcapturedOptions = options;\n\t\t// Return a minimal stub that satisfies AssistantMessageEventStream\n\t\treturn {\n\t\t\t[Symbol.asyncIterator]() { return { next: async () => ({ value: undefined, done: true as const }) }; },\n\t\t\tresult: () => Promise.resolve({ role: \"assistant\" as const, content: [], api: \"test\" as Api, provider: \"test\", model: \"test\", usage: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, totalTokens: 0, cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 } }, stopReason: \"stop\" as const, timestamp: Date.now() }),\n\t\t\tpush: () => {},\n\t\t\tend: () => {},\n\t\t} as unknown as AssistantMessageEventStream;\n\t};\n\treturn { streamSimple, getCapturedOptions: () => capturedOptions };\n}\n\n// ─── Registration ─────────────────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — registration\", () => {\n\tit(\"registers externalCli provider with streamSimple and without apiKey/oauth\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tassert.doesNotThrow(() => {\n\t\t\tregistry.registerProvider(\"cli-provider\", {\n\t\t\t\tauthMode: \"externalCli\",\n\t\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tstreamSimple: spy.streamSimple,\n\t\t\t\tmodels: [createProviderModel(\"cli-model\")],\n\t\t\t});\n\t\t});\n\t});\n\n\tit(\"registers none provider with streamSimple and without apiKey/oauth\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tassert.doesNotThrow(() => {\n\t\t\tregistry.registerProvider(\"none-provider\", {\n\t\t\t\tauthMode: \"none\",\n\t\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tstreamSimple: spy.streamSimple,\n\t\t\t\tmodels: [createProviderModel(\"local-model\")],\n\t\t\t});\n\t\t});\n\t});\n\n\tit(\"rejects apiKey provider without apiKey or oauth — message mentions authMode\", () => {\n\t\tconst registry = createRegistry();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"apikey-provider\", {\n\t\t\t\tauthMode: \"apiKey\",\n\t\t\t\tbaseUrl: \"https://api.local\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"authMode\"), \"error message must mention authMode\");\n\t\t\tassert.ok(err.message.includes(\"externalCli\"), \"error message must suggest externalCli\");\n\t\t\treturn true;\n\t\t});\n\t});\n\n\tit(\"rejects provider with no authMode and no apiKey/oauth (defaults to apiKey)\", () => {\n\t\tconst registry = createRegistry();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"bare-provider\", {\n\t\t\t\tbaseUrl: \"https://api.local\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"authMode\"), \"error message must mention authMode\");\n\t\t\treturn true;\n\t\t});\n\t});\n\n\tit(\"rejects externalCli provider without streamSimple\", () => {\n\t\tconst registry = createRegistry();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"cli-no-stream\", {\n\t\t\t\tauthMode: \"externalCli\",\n\t\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"streamSimple\"), \"error message must mention streamSimple\");\n\t\t\tassert.ok(err.message.includes(\"externalCli\"), \"error message must mention authMode\");\n\t\t\treturn true;\n\t\t});\n\t});\n\n\tit(\"rejects none provider without streamSimple\", () => {\n\t\tconst registry = createRegistry();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"none-no-stream\", {\n\t\t\t\tauthMode: \"none\",\n\t\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"streamSimple\"), \"error message must mention streamSimple\");\n\t\t\tassert.ok(err.message.includes(\"none\"), \"error message must mention authMode\");\n\t\t\treturn true;\n\t\t});\n\t});\n\n\tit(\"rejects externalCli provider that also sets apiKey\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"cli-with-key\", {\n\t\t\t\tauthMode: \"externalCli\",\n\t\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tapiKey: \"SHOULD_NOT_EXIST\",\n\t\t\t\tstreamSimple: spy.streamSimple,\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"apiKey\"), \"error message must mention apiKey\");\n\t\t\tassert.ok(err.message.includes(\"externalCli\"), \"error message must mention authMode\");\n\t\t\treturn true;\n\t\t});\n\t});\n\n\tit(\"rejects none provider that also sets apiKey\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"none-with-key\", {\n\t\t\t\tauthMode: \"none\",\n\t\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tapiKey: \"SHOULD_NOT_EXIST\",\n\t\t\t\tstreamSimple: spy.streamSimple,\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"apiKey\"), \"error message must mention apiKey\");\n\t\t\tassert.ok(err.message.includes(\"none\"), \"error message must mention authMode\");\n\t\t\treturn true;\n\t\t});\n\t});\n});\n\n// ─── getProviderAuthMode ──────────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — getProviderAuthMode\", () => {\n\tit(\"returns apiKey for unregistered (built-in) providers\", () => {\n\t\tconst registry = createRegistry();\n\t\tassert.equal(registry.getProviderAuthMode(\"anthropic\"), \"apiKey\");\n\t});\n\n\tit(\"returns explicit authMode when set\", () => {\n\t\tconst registry = createRegistry();\n\t\tregistry.registerProvider(\"cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.getProviderAuthMode(\"cli\"), \"externalCli\");\n\t});\n\n\tit(\"returns none when authMode is none\", () => {\n\t\tconst registry = createRegistry();\n\t\tregistry.registerProvider(\"local\", {\n\t\t\tauthMode: \"none\",\n\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.getProviderAuthMode(\"local\"), \"none\");\n\t});\n});\n\n// ─── isProviderRequestReady ───────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — isProviderRequestReady\", () => {\n\tit(\"returns true for externalCli without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.isProviderRequestReady(\"cli\"), true);\n\t});\n\n\tit(\"returns true for none without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"local\", {\n\t\t\tauthMode: \"none\",\n\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.isProviderRequestReady(\"local\"), true);\n\t});\n\n\tit(\"returns false for apiKey provider without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tassert.equal(registry.isProviderRequestReady(\"anthropic\"), false);\n\t});\n\n\tit(\"returns true for apiKey provider with stored auth\", () => {\n\t\tconst registry = createRegistry(() => true);\n\t\tassert.equal(registry.isProviderRequestReady(\"anthropic\"), true);\n\t});\n});\n\n// ─── isReady callback ─────────────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — isReady callback\", () => {\n\tit(\"calls isReady and returns its result for externalCli provider\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"cli-down\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tisReady: () => false,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.isProviderRequestReady(\"cli-down\"), false);\n\t});\n\n\tit(\"calls isReady for apiKey provider (overrides hasAuth)\", () => {\n\t\tconst registry = createRegistry(() => true);\n\t\tregistry.registerProvider(\"strict-provider\", {\n\t\t\tapiKey: \"MY_KEY\",\n\t\t\tbaseUrl: \"https://api.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tisReady: () => false,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.isProviderRequestReady(\"strict-provider\"), false);\n\t});\n\n\tit(\"isReady returning true makes provider available\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"healthy-cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tisReady: () => true,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.isProviderRequestReady(\"healthy-cli\"), true);\n\t});\n\n\tit(\"falls through to default behavior when isReady not provided\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"no-callback\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\t// externalCli without isReady → true (default)\n\t\tassert.equal(registry.isProviderRequestReady(\"no-callback\"), true);\n\t});\n});\n\n// ─── getAvailable ─────────────────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — getAvailable\", () => {\n\tit(\"includes externalCli models without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"cli-model\")],\n\t\t});\n\t\tassert.ok(findModel(registry, \"cli\", \"cli-model\"));\n\t});\n\n\tit(\"includes none models without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"local\", {\n\t\t\tauthMode: \"none\",\n\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"local-model\")],\n\t\t});\n\t\tassert.ok(findModel(registry, \"local\", \"local-model\"));\n\t});\n\n\tit(\"excludes externalCli models when isReady returns false\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"cli-down\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tisReady: () => false,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(findModel(registry, \"cli-down\", \"m\"), undefined);\n\t});\n\n\tit(\"excludes apiKey models without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tconst available = registry.getAvailable();\n\t\tassert.equal(available.length, 0);\n\t});\n});\n\n// ─── getApiKey ────────────────────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — getApiKey\", () => {\n\tit(\"returns undefined for externalCli provider\", async () => {\n\t\tconst registry = createRegistry();\n\t\tregistry.registerProvider(\"cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tconst model = registry.getAll().find((m) => m.provider === \"cli\")!;\n\t\tassert.equal(await registry.getApiKey(model), undefined);\n\t});\n\n\tit(\"returns undefined for none provider\", async () => {\n\t\tconst registry = createRegistry();\n\t\tregistry.registerProvider(\"local\", {\n\t\t\tauthMode: \"none\",\n\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tconst model = registry.getAll().find((m) => m.provider === \"local\")!;\n\t\tassert.equal(await registry.getApiKey(model), undefined);\n\t});\n\n\tit(\"delegates to authStorage for apiKey provider\", async () => {\n\t\tconst registry = createRegistry();\n\t\tconst key = await registry.getApiKeyForProvider(\"anthropic\");\n\t\tassert.equal(key, undefined);\n\t});\n});\n\n// ─── streamSimple apiKey stripping ────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — streamSimple apiKey boundary\", () => {\n\tit(\"strips apiKey from options for externalCli provider\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tconst apiType = `ext-cli-strip-${Date.now()}`;\n\n\t\tregistry.registerProvider(\"cli-strip\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: apiType as Api,\n\t\t\tstreamSimple: spy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"m\", apiType)],\n\t\t});\n\n\t\tconst provider = getApiProvider(apiType as Api);\n\t\tassert.ok(provider, \"provider must be registered in api registry\");\n\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"cli-strip\", \"m\", apiType),\n\t\t\tmakeContext(),\n\t\t\t{ apiKey: \"should-be-stripped\", maxTokens: 1024 } as SimpleStreamOptions,\n\t\t);\n\n\t\tconst captured = spy.getCapturedOptions();\n\t\tassert.ok(captured, \"streamSimple must have been called\");\n\t\tassert.equal(\"apiKey\" in captured, false, \"apiKey must not exist in options for externalCli provider\");\n\t\tassert.equal(captured.maxTokens, 1024, \"other options must pass through\");\n\t});\n\n\tit(\"strips apiKey from options for none provider\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tconst apiType = `none-strip-${Date.now()}`;\n\n\t\tregistry.registerProvider(\"none-strip\", {\n\t\t\tauthMode: \"none\",\n\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\tapi: apiType as Api,\n\t\t\tstreamSimple: spy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"m\", apiType)],\n\t\t});\n\n\t\tconst provider = getApiProvider(apiType as Api);\n\t\tassert.ok(provider, \"provider must be registered in api registry\");\n\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"none-strip\", \"m\", apiType),\n\t\t\tmakeContext(),\n\t\t\t{ apiKey: \"should-be-stripped\", maxTokens: 2048 } as SimpleStreamOptions,\n\t\t);\n\n\t\tconst captured = spy.getCapturedOptions();\n\t\tassert.ok(captured, \"streamSimple must have been called\");\n\t\tassert.equal(\"apiKey\" in captured, false, \"apiKey must not exist in options for none provider\");\n\t\tassert.equal(captured.maxTokens, 2048, \"other options must pass through\");\n\t});\n\n\tit(\"preserves apiKey in options for apiKey provider\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tconst apiType = `apikey-preserve-${Date.now()}`;\n\n\t\tregistry.registerProvider(\"apikey-preserve\", {\n\t\t\tapiKey: \"MY_KEY\",\n\t\t\tbaseUrl: \"https://api.local\",\n\t\t\tapi: apiType as Api,\n\t\t\tstreamSimple: spy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"m\", apiType)],\n\t\t});\n\n\t\tconst provider = getApiProvider(apiType as Api);\n\t\tassert.ok(provider, \"provider must be registered in api registry\");\n\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"apikey-preserve\", \"m\", apiType),\n\t\t\tmakeContext(),\n\t\t\t{ apiKey: \"sk-real-key\", maxTokens: 4096 } as SimpleStreamOptions,\n\t\t);\n\n\t\tconst captured = spy.getCapturedOptions();\n\t\tassert.ok(captured, \"streamSimple must have been called\");\n\t\tassert.equal(captured.apiKey, \"sk-real-key\", \"apiKey must be preserved for apiKey provider\");\n\t\tassert.equal(captured.maxTokens, 4096, \"other options must pass through\");\n\t});\n\n\tit(\"handles undefined options for externalCli provider\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tconst apiType = `ext-cli-undef-${Date.now()}`;\n\n\t\tregistry.registerProvider(\"cli-undef\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: apiType as Api,\n\t\t\tstreamSimple: spy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"m\", apiType)],\n\t\t});\n\n\t\tconst provider = getApiProvider(apiType as Api);\n\t\tassert.ok(provider, \"provider must be registered in api registry\");\n\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"cli-undef\", \"m\", apiType),\n\t\t\tmakeContext(),\n\t\t\tundefined,\n\t\t);\n\n\t\tconst captured = spy.getCapturedOptions();\n\t\tassert.ok(captured !== undefined, \"streamSimple must have been called\");\n\t\tassert.equal(\"apiKey\" in captured, false, \"apiKey must not exist even when options is undefined\");\n\t});\n\n\tit(\"strips apiKey but preserves signal and other fields for externalCli\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tconst apiType = `ext-cli-fields-${Date.now()}`;\n\t\tconst abortController = new AbortController();\n\n\t\tregistry.registerProvider(\"cli-fields\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: apiType as Api,\n\t\t\tstreamSimple: spy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"m\", apiType)],\n\t\t});\n\n\t\tconst provider = getApiProvider(apiType as Api);\n\t\tassert.ok(provider, \"provider must be registered in api registry\");\n\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"cli-fields\", \"m\", apiType),\n\t\t\tmakeContext(),\n\t\t\t{ apiKey: \"strip-me\", maxTokens: 8192, signal: abortController.signal, reasoning: \"high\" } as SimpleStreamOptions,\n\t\t);\n\n\t\tconst captured = spy.getCapturedOptions();\n\t\tassert.ok(captured, \"streamSimple must have been called\");\n\t\tassert.equal(\"apiKey\" in captured, false, \"apiKey must be stripped\");\n\t\tassert.equal(captured.maxTokens, 8192, \"maxTokens must pass through\");\n\t\tassert.equal(captured.signal, abortController.signal, \"signal must pass through\");\n\t\tassert.equal((captured as Record<string, unknown>).reasoning, \"high\", \"reasoning must pass through\");\n\t});\n});\n\n// ─── Provider-scoped stream routing (#2533) ───────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — provider-scoped stream routing\", () => {\n\tit(\"does not clobber built-in stream handler when custom provider uses same api\", () => {\n\t\tconst registry = createRegistry(() => true);\n\t\tconst customSpy = createStreamSpy();\n\n\t\t// Register a custom provider with the same API type as a built-in (anthropic-messages).\n\t\t// This simulates the claude-code-cli extension registering with api: \"anthropic-messages\".\n\t\tregistry.registerProvider(\"custom-cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"local://custom\",\n\t\t\tapi: \"anthropic-messages\",\n\t\t\tstreamSimple: customSpy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"custom-model\", \"anthropic-messages\")],\n\t\t});\n\n\t\t// The built-in anthropic-messages provider should still be accessible\n\t\t// when calling streamSimple with a model from the built-in provider.\n\t\tconst provider = getApiProvider(\"anthropic-messages\" as Api);\n\t\tassert.ok(provider, \"anthropic-messages provider must still be registered\");\n\n\t\t// Call with a built-in anthropic model — should NOT hit the custom spy.\n\t\t// The built-in handler will throw (no API key), which proves the routing\n\t\t// correctly delegates to the built-in instead of the custom handler.\n\t\tassert.throws(\n\t\t\t() => provider.streamSimple(\n\t\t\t\tmakeModel(\"anthropic\", \"claude-sonnet-4-6\", \"anthropic-messages\"),\n\t\t\t\tmakeContext(),\n\t\t\t\t{ maxTokens: 4096 } as SimpleStreamOptions,\n\t\t\t),\n\t\t\t(err: Error) => err.message.includes(\"API key\"),\n\t\t\t\"built-in Anthropic handler must be invoked (throws because no API key in tests)\",\n\t\t);\n\n\t\tassert.equal(\n\t\t\tcustomSpy.getCapturedOptions(),\n\t\t\tundefined,\n\t\t\t\"custom provider's streamSimple must NOT be called for anthropic provider models\",\n\t\t);\n\t});\n\n\tit(\"routes to custom provider when model.provider matches\", () => {\n\t\tconst registry = createRegistry(() => true);\n\t\tconst customSpy = createStreamSpy();\n\n\t\tregistry.registerProvider(\"custom-cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"local://custom\",\n\t\t\tapi: \"anthropic-messages\",\n\t\t\tstreamSimple: customSpy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"custom-model\", \"anthropic-messages\")],\n\t\t});\n\n\t\tconst provider = getApiProvider(\"anthropic-messages\" as Api);\n\t\tassert.ok(provider);\n\n\t\t// Call with the custom provider's model — should hit the custom spy\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"custom-cli\", \"custom-model\", \"anthropic-messages\"),\n\t\t\tmakeContext(),\n\t\t\t{ maxTokens: 2048 } as SimpleStreamOptions,\n\t\t);\n\n\t\tconst captured = customSpy.getCapturedOptions();\n\t\tassert.ok(captured, \"custom provider's streamSimple must be called for its own models\");\n\t\tassert.equal(captured.maxTokens, 2048);\n\t});\n});\n"]}
|
|
1
|
+
{"version":3,"file":"model-registry-auth-mode.test.js","sourceRoot":"","sources":["../../src/core/model-registry-auth-mode.test.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from \"node:assert/strict\";\nimport { describe, it } from \"node:test\";\nimport type { Api, Model, SimpleStreamOptions, Context, AssistantMessageEventStream } from \"@gsd/pi-ai\";\nimport { getApiProvider } from \"@gsd/pi-ai\";\nimport { AuthStorage, type AuthStorageData } from \"./auth-storage.js\";\nimport { ModelRegistry } from \"./model-registry.js\";\n\nfunction createRegistry(hasAuthFn?: (provider: string) => boolean): ModelRegistry {\n\tconst authStorage = {\n\t\tsetFallbackResolver: () => {},\n\t\tonCredentialChange: () => {},\n\t\tgetOAuthProviders: () => [],\n\t\tget: () => undefined,\n\t\thasAuth: hasAuthFn ?? (() => false),\n\t\tgetApiKey: async () => undefined,\n\t} as unknown as AuthStorage;\n\n\treturn new ModelRegistry(authStorage, undefined);\n}\n\nfunction createInMemoryRegistry(data: AuthStorageData = {}): ModelRegistry {\n\treturn new ModelRegistry(AuthStorage.inMemory(data), undefined);\n}\n\nfunction createProviderModel(id: string, api?: string): NonNullable<Parameters<ModelRegistry[\"registerProvider\"]>[1][\"models\"]>[number] {\n\treturn {\n\t\tid,\n\t\tname: id,\n\t\tapi: (api ?? \"openai-completions\") as Api,\n\t\treasoning: false,\n\t\tinput: [\"text\"],\n\t\tcost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },\n\t\tcontextWindow: 128000,\n\t\tmaxTokens: 16384,\n\t};\n}\n\nfunction findModel(registry: ModelRegistry, provider: string, id: string): Model<Api> | undefined {\n\treturn registry.getAvailable().find((m) => m.provider === provider && m.id === id);\n}\n\nfunction makeModel(provider: string, id: string, api: string): Model<Api> {\n\treturn {\n\t\tid,\n\t\tname: id,\n\t\tapi: api as Api,\n\t\tprovider,\n\t\tbaseUrl: `${provider}:`,\n\t\treasoning: false,\n\t\tinput: [\"text\"],\n\t\tcost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },\n\t\tcontextWindow: 128000,\n\t\tmaxTokens: 16384,\n\t};\n}\n\nfunction makeContext(): Context {\n\treturn {\n\t\tsystemPrompt: \"test\",\n\t\tmessages: [{ role: \"user\", content: \"hello\", timestamp: Date.now() }],\n\t};\n}\n\n/** No-op streamSimple for tests that need one to pass validation but don't inspect it. */\nconst noopStreamSimple = (_model: Model<Api>, _context: Context, _options?: SimpleStreamOptions) => {\n\treturn {\n\t\t[Symbol.asyncIterator]() { return { next: async () => ({ value: undefined, done: true as const }) }; },\n\t\tresult: () => Promise.resolve({ role: \"assistant\" as const, content: [], api: \"test\" as Api, provider: \"test\", model: \"test\", usage: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, totalTokens: 0, cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 } }, stopReason: \"stop\" as const, timestamp: Date.now() }),\n\t\tpush: () => {},\n\t\tend: () => {},\n\t} as unknown as AssistantMessageEventStream;\n};\n\n/** Create a spy streamSimple that captures the options it receives and returns a stub stream. */\nfunction createStreamSpy(): {\n\tstreamSimple: (model: Model<Api>, context: Context, options?: SimpleStreamOptions) => AssistantMessageEventStream;\n\tgetCapturedOptions: () => SimpleStreamOptions | undefined;\n} {\n\tlet capturedOptions: SimpleStreamOptions | undefined;\n\tconst streamSimple = (_model: Model<Api>, _context: Context, options?: SimpleStreamOptions) => {\n\t\tcapturedOptions = options;\n\t\t// Return a minimal stub that satisfies AssistantMessageEventStream\n\t\treturn {\n\t\t\t[Symbol.asyncIterator]() { return { next: async () => ({ value: undefined, done: true as const }) }; },\n\t\t\tresult: () => Promise.resolve({ role: \"assistant\" as const, content: [], api: \"test\" as Api, provider: \"test\", model: \"test\", usage: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, totalTokens: 0, cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 } }, stopReason: \"stop\" as const, timestamp: Date.now() }),\n\t\t\tpush: () => {},\n\t\t\tend: () => {},\n\t\t} as unknown as AssistantMessageEventStream;\n\t};\n\treturn { streamSimple, getCapturedOptions: () => capturedOptions };\n}\n\n// ─── Registration ─────────────────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — registration\", () => {\n\tit(\"registers externalCli provider with streamSimple and without apiKey/oauth\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tassert.doesNotThrow(() => {\n\t\t\tregistry.registerProvider(\"cli-provider\", {\n\t\t\t\tauthMode: \"externalCli\",\n\t\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tstreamSimple: spy.streamSimple,\n\t\t\t\tmodels: [createProviderModel(\"cli-model\")],\n\t\t\t});\n\t\t});\n\t});\n\n\tit(\"registers none provider with streamSimple and without apiKey/oauth\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tassert.doesNotThrow(() => {\n\t\t\tregistry.registerProvider(\"none-provider\", {\n\t\t\t\tauthMode: \"none\",\n\t\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tstreamSimple: spy.streamSimple,\n\t\t\t\tmodels: [createProviderModel(\"local-model\")],\n\t\t\t});\n\t\t});\n\t});\n\n\tit(\"rejects apiKey provider without apiKey or oauth — message mentions authMode\", () => {\n\t\tconst registry = createRegistry();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"apikey-provider\", {\n\t\t\t\tauthMode: \"apiKey\",\n\t\t\t\tbaseUrl: \"https://api.local\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"authMode\"), \"error message must mention authMode\");\n\t\t\tassert.ok(err.message.includes(\"externalCli\"), \"error message must suggest externalCli\");\n\t\t\treturn true;\n\t\t});\n\t});\n\n\tit(\"rejects provider with no authMode and no apiKey/oauth (defaults to apiKey)\", () => {\n\t\tconst registry = createRegistry();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"bare-provider\", {\n\t\t\t\tbaseUrl: \"https://api.local\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"authMode\"), \"error message must mention authMode\");\n\t\t\treturn true;\n\t\t});\n\t});\n\n\tit(\"rejects externalCli provider without streamSimple\", () => {\n\t\tconst registry = createRegistry();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"cli-no-stream\", {\n\t\t\t\tauthMode: \"externalCli\",\n\t\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"streamSimple\"), \"error message must mention streamSimple\");\n\t\t\tassert.ok(err.message.includes(\"externalCli\"), \"error message must mention authMode\");\n\t\t\treturn true;\n\t\t});\n\t});\n\n\tit(\"rejects none provider without streamSimple\", () => {\n\t\tconst registry = createRegistry();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"none-no-stream\", {\n\t\t\t\tauthMode: \"none\",\n\t\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"streamSimple\"), \"error message must mention streamSimple\");\n\t\t\tassert.ok(err.message.includes(\"none\"), \"error message must mention authMode\");\n\t\t\treturn true;\n\t\t});\n\t});\n\n\tit(\"rejects externalCli provider that also sets apiKey\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"cli-with-key\", {\n\t\t\t\tauthMode: \"externalCli\",\n\t\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tapiKey: \"SHOULD_NOT_EXIST\",\n\t\t\t\tstreamSimple: spy.streamSimple,\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"apiKey\"), \"error message must mention apiKey\");\n\t\t\tassert.ok(err.message.includes(\"externalCli\"), \"error message must mention authMode\");\n\t\t\treturn true;\n\t\t});\n\t});\n\n\tit(\"rejects none provider that also sets apiKey\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tassert.throws(() => {\n\t\t\tregistry.registerProvider(\"none-with-key\", {\n\t\t\t\tauthMode: \"none\",\n\t\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\t\tapi: \"openai-completions\",\n\t\t\t\tapiKey: \"SHOULD_NOT_EXIST\",\n\t\t\t\tstreamSimple: spy.streamSimple,\n\t\t\t\tmodels: [createProviderModel(\"model\")],\n\t\t\t});\n\t\t}, (err: Error) => {\n\t\t\tassert.ok(err.message.includes(\"apiKey\"), \"error message must mention apiKey\");\n\t\t\tassert.ok(err.message.includes(\"none\"), \"error message must mention authMode\");\n\t\t\treturn true;\n\t\t});\n\t});\n});\n\n// ─── getProviderAuthMode ──────────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — getProviderAuthMode\", () => {\n\tit(\"returns apiKey for unregistered (built-in) providers\", () => {\n\t\tconst registry = createRegistry();\n\t\tassert.equal(registry.getProviderAuthMode(\"anthropic\"), \"apiKey\");\n\t});\n\n\tit(\"returns explicit authMode when set\", () => {\n\t\tconst registry = createRegistry();\n\t\tregistry.registerProvider(\"cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.getProviderAuthMode(\"cli\"), \"externalCli\");\n\t});\n\n\tit(\"returns none when authMode is none\", () => {\n\t\tconst registry = createRegistry();\n\t\tregistry.registerProvider(\"local\", {\n\t\t\tauthMode: \"none\",\n\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.getProviderAuthMode(\"local\"), \"none\");\n\t});\n});\n\n// ─── isProviderRequestReady ───────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — isProviderRequestReady\", () => {\n\tit(\"returns true for externalCli without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.isProviderRequestReady(\"cli\"), true);\n\t});\n\n\tit(\"returns true for none without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"local\", {\n\t\t\tauthMode: \"none\",\n\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.isProviderRequestReady(\"local\"), true);\n\t});\n\n\tit(\"returns false for apiKey provider without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tassert.equal(registry.isProviderRequestReady(\"anthropic\"), false);\n\t});\n\n\tit(\"returns true for apiKey provider with stored auth\", () => {\n\t\tconst registry = createRegistry(() => true);\n\t\tassert.equal(registry.isProviderRequestReady(\"anthropic\"), true);\n\t});\n});\n\n// ─── isReady callback ─────────────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — isReady callback\", () => {\n\tit(\"calls isReady and returns its result for externalCli provider\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"cli-down\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tisReady: () => false,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.isProviderRequestReady(\"cli-down\"), false);\n\t});\n\n\tit(\"calls isReady for apiKey provider (overrides hasAuth)\", () => {\n\t\tconst registry = createRegistry(() => true);\n\t\tregistry.registerProvider(\"strict-provider\", {\n\t\t\tapiKey: \"MY_KEY\",\n\t\t\tbaseUrl: \"https://api.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tisReady: () => false,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.isProviderRequestReady(\"strict-provider\"), false);\n\t});\n\n\tit(\"isReady returning true makes provider available\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"healthy-cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tisReady: () => true,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(registry.isProviderRequestReady(\"healthy-cli\"), true);\n\t});\n\n\tit(\"falls through to default behavior when isReady not provided\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"no-callback\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\t// externalCli without isReady → true (default)\n\t\tassert.equal(registry.isProviderRequestReady(\"no-callback\"), true);\n\t});\n});\n\n// ─── getAvailable ─────────────────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — getAvailable\", () => {\n\tit(\"includes externalCli models without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"cli-model\")],\n\t\t});\n\t\tassert.ok(findModel(registry, \"cli\", \"cli-model\"));\n\t});\n\n\tit(\"includes none models without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"local\", {\n\t\t\tauthMode: \"none\",\n\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"local-model\")],\n\t\t});\n\t\tassert.ok(findModel(registry, \"local\", \"local-model\"));\n\t});\n\n\tit(\"excludes externalCli models when isReady returns false\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tregistry.registerProvider(\"cli-down\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tisReady: () => false,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tassert.equal(findModel(registry, \"cli-down\", \"m\"), undefined);\n\t});\n\n\tit(\"excludes apiKey models without stored auth\", () => {\n\t\tconst registry = createRegistry(() => false);\n\t\tconst available = registry.getAvailable();\n\t\tassert.equal(available.length, 0);\n\t});\n\n\tit(\"prunes Codex models removed from ChatGPT-backed openai-codex OAuth\", () => {\n\t\tconst registry = createInMemoryRegistry({\n\t\t\t\"openai-codex\": {\n\t\t\t\ttype: \"oauth\",\n\t\t\t\taccess: \"oauth-access\",\n\t\t\t\trefresh: \"oauth-refresh\",\n\t\t\t\texpires: Date.now() + 60_000,\n\t\t\t\taccountId: \"acct_123\",\n\t\t\t},\n\t\t});\n\n\t\tassert.equal(registry.find(\"openai-codex\", \"gpt-5.1-codex-max\"), undefined);\n\t\tassert.equal(registry.find(\"openai-codex\", \"gpt-5.1\"), undefined);\n\t\tassert.equal(findModel(registry, \"openai-codex\", \"gpt-5.2-codex\"), undefined);\n\t\tassert.ok(registry.find(\"openai-codex\", \"gpt-5.4\"));\n\t\tassert.ok(findModel(registry, \"openai-codex\", \"gpt-5.4\"));\n\t\tassert.ok(registry.find(\"openai-codex\", \"gpt-5.4-mini\"));\n\t\tassert.ok(findModel(registry, \"openai-codex\", \"gpt-5.4-mini\"));\n\t});\n\n\tit(\"keeps API-backed OpenAI Codex-capable models available\", () => {\n\t\tconst registry = createInMemoryRegistry({\n\t\t\topenai: {\n\t\t\t\ttype: \"api_key\",\n\t\t\t\tkey: \"sk-test\",\n\t\t\t},\n\t\t});\n\n\t\tassert.ok(registry.find(\"openai\", \"gpt-5.2-codex\"));\n\t\tassert.ok(findModel(registry, \"openai\", \"gpt-5.2-codex\"));\n\t});\n});\n\n// ─── getApiKey ────────────────────────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — getApiKey\", () => {\n\tit(\"returns undefined for externalCli provider\", async () => {\n\t\tconst registry = createRegistry();\n\t\tregistry.registerProvider(\"cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tconst model = registry.getAll().find((m) => m.provider === \"cli\")!;\n\t\tassert.equal(await registry.getApiKey(model), undefined);\n\t});\n\n\tit(\"returns undefined for none provider\", async () => {\n\t\tconst registry = createRegistry();\n\t\tregistry.registerProvider(\"local\", {\n\t\t\tauthMode: \"none\",\n\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\tapi: \"openai-completions\",\n\t\t\tstreamSimple: noopStreamSimple,\n\t\t\tmodels: [createProviderModel(\"m\")],\n\t\t});\n\t\tconst model = registry.getAll().find((m) => m.provider === \"local\")!;\n\t\tassert.equal(await registry.getApiKey(model), undefined);\n\t});\n\n\tit(\"delegates to authStorage for apiKey provider\", async () => {\n\t\tconst registry = createRegistry();\n\t\tconst key = await registry.getApiKeyForProvider(\"anthropic\");\n\t\tassert.equal(key, undefined);\n\t});\n});\n\n// ─── streamSimple apiKey stripping ────────────────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — streamSimple apiKey boundary\", () => {\n\tit(\"strips apiKey from options for externalCli provider\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tconst apiType = `ext-cli-strip-${Date.now()}`;\n\n\t\tregistry.registerProvider(\"cli-strip\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: apiType as Api,\n\t\t\tstreamSimple: spy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"m\", apiType)],\n\t\t});\n\n\t\tconst provider = getApiProvider(apiType as Api);\n\t\tassert.ok(provider, \"provider must be registered in api registry\");\n\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"cli-strip\", \"m\", apiType),\n\t\t\tmakeContext(),\n\t\t\t{ apiKey: \"should-be-stripped\", maxTokens: 1024 } as SimpleStreamOptions,\n\t\t);\n\n\t\tconst captured = spy.getCapturedOptions();\n\t\tassert.ok(captured, \"streamSimple must have been called\");\n\t\tassert.equal(\"apiKey\" in captured, false, \"apiKey must not exist in options for externalCli provider\");\n\t\tassert.equal(captured.maxTokens, 1024, \"other options must pass through\");\n\t});\n\n\tit(\"strips apiKey from options for none provider\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tconst apiType = `none-strip-${Date.now()}`;\n\n\t\tregistry.registerProvider(\"none-strip\", {\n\t\t\tauthMode: \"none\",\n\t\t\tbaseUrl: \"http://localhost:11434\",\n\t\t\tapi: apiType as Api,\n\t\t\tstreamSimple: spy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"m\", apiType)],\n\t\t});\n\n\t\tconst provider = getApiProvider(apiType as Api);\n\t\tassert.ok(provider, \"provider must be registered in api registry\");\n\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"none-strip\", \"m\", apiType),\n\t\t\tmakeContext(),\n\t\t\t{ apiKey: \"should-be-stripped\", maxTokens: 2048 } as SimpleStreamOptions,\n\t\t);\n\n\t\tconst captured = spy.getCapturedOptions();\n\t\tassert.ok(captured, \"streamSimple must have been called\");\n\t\tassert.equal(\"apiKey\" in captured, false, \"apiKey must not exist in options for none provider\");\n\t\tassert.equal(captured.maxTokens, 2048, \"other options must pass through\");\n\t});\n\n\tit(\"preserves apiKey in options for apiKey provider\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tconst apiType = `apikey-preserve-${Date.now()}`;\n\n\t\tregistry.registerProvider(\"apikey-preserve\", {\n\t\t\tapiKey: \"MY_KEY\",\n\t\t\tbaseUrl: \"https://api.local\",\n\t\t\tapi: apiType as Api,\n\t\t\tstreamSimple: spy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"m\", apiType)],\n\t\t});\n\n\t\tconst provider = getApiProvider(apiType as Api);\n\t\tassert.ok(provider, \"provider must be registered in api registry\");\n\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"apikey-preserve\", \"m\", apiType),\n\t\t\tmakeContext(),\n\t\t\t{ apiKey: \"sk-real-key\", maxTokens: 4096 } as SimpleStreamOptions,\n\t\t);\n\n\t\tconst captured = spy.getCapturedOptions();\n\t\tassert.ok(captured, \"streamSimple must have been called\");\n\t\tassert.equal(captured.apiKey, \"sk-real-key\", \"apiKey must be preserved for apiKey provider\");\n\t\tassert.equal(captured.maxTokens, 4096, \"other options must pass through\");\n\t});\n\n\tit(\"handles undefined options for externalCli provider\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tconst apiType = `ext-cli-undef-${Date.now()}`;\n\n\t\tregistry.registerProvider(\"cli-undef\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: apiType as Api,\n\t\t\tstreamSimple: spy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"m\", apiType)],\n\t\t});\n\n\t\tconst provider = getApiProvider(apiType as Api);\n\t\tassert.ok(provider, \"provider must be registered in api registry\");\n\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"cli-undef\", \"m\", apiType),\n\t\t\tmakeContext(),\n\t\t\tundefined,\n\t\t);\n\n\t\tconst captured = spy.getCapturedOptions();\n\t\tassert.ok(captured !== undefined, \"streamSimple must have been called\");\n\t\tassert.equal(\"apiKey\" in captured, false, \"apiKey must not exist even when options is undefined\");\n\t});\n\n\tit(\"strips apiKey but preserves signal and other fields for externalCli\", () => {\n\t\tconst registry = createRegistry();\n\t\tconst spy = createStreamSpy();\n\t\tconst apiType = `ext-cli-fields-${Date.now()}`;\n\t\tconst abortController = new AbortController();\n\n\t\tregistry.registerProvider(\"cli-fields\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"https://cli.local\",\n\t\t\tapi: apiType as Api,\n\t\t\tstreamSimple: spy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"m\", apiType)],\n\t\t});\n\n\t\tconst provider = getApiProvider(apiType as Api);\n\t\tassert.ok(provider, \"provider must be registered in api registry\");\n\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"cli-fields\", \"m\", apiType),\n\t\t\tmakeContext(),\n\t\t\t{ apiKey: \"strip-me\", maxTokens: 8192, signal: abortController.signal, reasoning: \"high\" } as SimpleStreamOptions,\n\t\t);\n\n\t\tconst captured = spy.getCapturedOptions();\n\t\tassert.ok(captured, \"streamSimple must have been called\");\n\t\tassert.equal(\"apiKey\" in captured, false, \"apiKey must be stripped\");\n\t\tassert.equal(captured.maxTokens, 8192, \"maxTokens must pass through\");\n\t\tassert.equal(captured.signal, abortController.signal, \"signal must pass through\");\n\t\tassert.equal((captured as Record<string, unknown>).reasoning, \"high\", \"reasoning must pass through\");\n\t});\n});\n\n// ─── Provider-scoped stream routing (#2533) ───────────────────────────────────\n\ndescribe(\"ModelRegistry authMode — provider-scoped stream routing\", () => {\n\tit(\"does not clobber built-in stream handler when custom provider uses same api\", () => {\n\t\tconst registry = createRegistry(() => true);\n\t\tconst customSpy = createStreamSpy();\n\n\t\t// Register a custom provider with the same API type as a built-in (anthropic-messages).\n\t\t// This simulates the claude-code-cli extension registering with api: \"anthropic-messages\".\n\t\tregistry.registerProvider(\"custom-cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"local://custom\",\n\t\t\tapi: \"anthropic-messages\",\n\t\t\tstreamSimple: customSpy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"custom-model\", \"anthropic-messages\")],\n\t\t});\n\n\t\t// The built-in anthropic-messages provider should still be accessible\n\t\t// when calling streamSimple with a model from the built-in provider.\n\t\tconst provider = getApiProvider(\"anthropic-messages\" as Api);\n\t\tassert.ok(provider, \"anthropic-messages provider must still be registered\");\n\n\t\t// Call with a built-in anthropic model — should NOT hit the custom spy.\n\t\t// The built-in handler will throw (no API key), which proves the routing\n\t\t// correctly delegates to the built-in instead of the custom handler.\n\t\tassert.throws(\n\t\t\t() => provider.streamSimple(\n\t\t\t\tmakeModel(\"anthropic\", \"claude-sonnet-4-6\", \"anthropic-messages\"),\n\t\t\t\tmakeContext(),\n\t\t\t\t{ maxTokens: 4096 } as SimpleStreamOptions,\n\t\t\t),\n\t\t\t(err: Error) => err.message.includes(\"API key\"),\n\t\t\t\"built-in Anthropic handler must be invoked (throws because no API key in tests)\",\n\t\t);\n\n\t\tassert.equal(\n\t\t\tcustomSpy.getCapturedOptions(),\n\t\t\tundefined,\n\t\t\t\"custom provider's streamSimple must NOT be called for anthropic provider models\",\n\t\t);\n\t});\n\n\tit(\"routes to custom provider when model.provider matches\", () => {\n\t\tconst registry = createRegistry(() => true);\n\t\tconst customSpy = createStreamSpy();\n\n\t\tregistry.registerProvider(\"custom-cli\", {\n\t\t\tauthMode: \"externalCli\",\n\t\t\tbaseUrl: \"local://custom\",\n\t\t\tapi: \"anthropic-messages\",\n\t\t\tstreamSimple: customSpy.streamSimple,\n\t\t\tmodels: [createProviderModel(\"custom-model\", \"anthropic-messages\")],\n\t\t});\n\n\t\tconst provider = getApiProvider(\"anthropic-messages\" as Api);\n\t\tassert.ok(provider);\n\n\t\t// Call with the custom provider's model — should hit the custom spy\n\t\tprovider.streamSimple(\n\t\t\tmakeModel(\"custom-cli\", \"custom-model\", \"anthropic-messages\"),\n\t\t\tmakeContext(),\n\t\t\t{ maxTokens: 2048 } as SimpleStreamOptions,\n\t\t);\n\n\t\tconst captured = customSpy.getCapturedOptions();\n\t\tassert.ok(captured, \"custom provider's streamSimple must be called for its own models\");\n\t\tassert.equal(captured.maxTokens, 2048);\n\t});\n});\n"]}
|