@pellux/goodvibes-tui 0.18.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.goodvibes/GOODVIBES.md +35 -0
- package/.goodvibes/agents/reviewer.md +89 -0
- package/.goodvibes/skills/add-provider/SKILL.md +199 -0
- package/CHANGELOG.md +1681 -0
- package/README.md +1607 -0
- package/bin/goodvibes +66 -0
- package/docs/README.md +32 -0
- package/docs/foundation-artifacts/README.md +16 -0
- package/docs/foundation-artifacts/knowledge-graphql.graphql +397 -0
- package/docs/foundation-artifacts/knowledge-store.sql +183 -0
- package/docs/foundation-artifacts/operator-contract.json +55157 -0
- package/docs/foundation-artifacts/peer-contract.json +2384 -0
- package/package.json +114 -0
- package/scripts/postinstall.mjs +203 -0
- package/src/acp/connection.ts +447 -0
- package/src/acp/index.ts +7 -0
- package/src/acp/manager.ts +133 -0
- package/src/adapters/bluebubbles/index.ts +127 -0
- package/src/adapters/discord/index.ts +297 -0
- package/src/adapters/github/index.ts +73 -0
- package/src/adapters/google-chat/index.ts +119 -0
- package/src/adapters/imessage/index.ts +92 -0
- package/src/adapters/index.ts +15 -0
- package/src/adapters/matrix/index.ts +116 -0
- package/src/adapters/mattermost/index.ts +151 -0
- package/src/adapters/msteams/index.ts +180 -0
- package/src/adapters/ntfy/index.ts +118 -0
- package/src/adapters/signal/index.ts +92 -0
- package/src/adapters/slack/index.ts +323 -0
- package/src/adapters/telegram/index.ts +160 -0
- package/src/adapters/types.ts +97 -0
- package/src/adapters/webhook/index.ts +178 -0
- package/src/adapters/whatsapp/index.ts +135 -0
- package/src/agents/message-bus-core.ts +312 -0
- package/src/agents/message-bus.ts +2 -0
- package/src/agents/orchestrator-prompts.ts +351 -0
- package/src/agents/orchestrator-runner.ts +668 -0
- package/src/agents/orchestrator.ts +437 -0
- package/src/agents/session.ts +108 -0
- package/src/agents/worktree.ts +153 -0
- package/src/agents/wrfc-config.ts +47 -0
- package/src/agents/wrfc-controller.ts +747 -0
- package/src/agents/wrfc-gate-runtime.ts +75 -0
- package/src/agents/wrfc-reporting.ts +284 -0
- package/src/agents/wrfc-runtime-events.ts +150 -0
- package/src/agents/wrfc-types.ts +67 -0
- package/src/automation/delivery-manager.ts +368 -0
- package/src/automation/index.ts +72 -0
- package/src/automation/manager-runtime-delivery.ts +139 -0
- package/src/automation/manager-runtime-events.ts +131 -0
- package/src/automation/manager-runtime-execution.ts +511 -0
- package/src/automation/manager-runtime-helpers.ts +433 -0
- package/src/automation/manager-runtime-job-mutations.ts +175 -0
- package/src/automation/manager-runtime-reconcile.ts +148 -0
- package/src/automation/manager-runtime-scheduling.ts +189 -0
- package/src/automation/manager-runtime-sync.ts +54 -0
- package/src/automation/manager-runtime.ts +721 -0
- package/src/automation/manager.ts +10 -0
- package/src/automation/service.ts +242 -0
- package/src/channels/builtin/account-actions.ts +490 -0
- package/src/channels/builtin/accounts.ts +433 -0
- package/src/channels/builtin/contracts.ts +405 -0
- package/src/channels/builtin/plugins.ts +308 -0
- package/src/channels/builtin/rendering.ts +174 -0
- package/src/channels/builtin/setup-schema.ts +504 -0
- package/src/channels/builtin/shared.ts +96 -0
- package/src/channels/builtin/surfaces.ts +57 -0
- package/src/channels/builtin/targets.ts +693 -0
- package/src/channels/builtin-runtime.ts +443 -0
- package/src/channels/delivery/shared.ts +199 -0
- package/src/channels/delivery/strategies-bridge.ts +246 -0
- package/src/channels/delivery/strategies-core.ts +299 -0
- package/src/channels/delivery/strategies-enterprise.ts +178 -0
- package/src/channels/delivery/types.ts +59 -0
- package/src/channels/delivery-router.ts +127 -0
- package/src/channels/index.ts +77 -0
- package/src/channels/plugin-registry.ts +551 -0
- package/src/channels/provider-runtime.ts +330 -0
- package/src/channels/reply-pipeline.ts +522 -0
- package/src/channels/route-manager.ts +340 -0
- package/src/channels/surface-registry.ts +186 -0
- package/src/config/helper-model.ts +233 -0
- package/src/config/index.ts +193 -0
- package/src/config/manager.ts +404 -0
- package/src/config/secrets.ts +547 -0
- package/src/config/service-registry.ts +329 -0
- package/src/config/subscription-auth.ts +31 -0
- package/src/config/subscription-providers.ts +127 -0
- package/src/config/tool-llm.ts +110 -0
- package/src/control-plane/approval-broker.ts +351 -0
- package/src/control-plane/gateway.ts +713 -0
- package/src/control-plane/index.ts +54 -0
- package/src/control-plane/media-contract-schemas.ts +208 -0
- package/src/control-plane/method-catalog-admin.ts +136 -0
- package/src/control-plane/method-catalog-channels.ts +591 -0
- package/src/control-plane/method-catalog-control-automation.ts +475 -0
- package/src/control-plane/method-catalog-control-core.ts +594 -0
- package/src/control-plane/method-catalog-control.ts +8 -0
- package/src/control-plane/method-catalog-events.ts +74 -0
- package/src/control-plane/method-catalog-knowledge.ts +531 -0
- package/src/control-plane/method-catalog-media.ts +279 -0
- package/src/control-plane/method-catalog-runtime.ts +304 -0
- package/src/control-plane/method-catalog-shared.ts +223 -0
- package/src/control-plane/method-catalog.ts +242 -0
- package/src/control-plane/operator-contract-schemas-admin.ts +639 -0
- package/src/control-plane/operator-contract-schemas-channels.ts +375 -0
- package/src/control-plane/operator-contract-schemas-control.ts +226 -0
- package/src/control-plane/operator-contract-schemas-domains.ts +4 -0
- package/src/control-plane/operator-contract-schemas-knowledge.ts +582 -0
- package/src/control-plane/operator-contract-schemas-media.ts +297 -0
- package/src/control-plane/operator-contract-schemas-permissions.ts +100 -0
- package/src/control-plane/operator-contract-schemas-remote.ts +38 -0
- package/src/control-plane/operator-contract-schemas-runtime.ts +563 -0
- package/src/control-plane/operator-contract-schemas-shared.ts +85 -0
- package/src/control-plane/operator-contract-schemas-telemetry.ts +349 -0
- package/src/control-plane/operator-contract-schemas.ts +6 -0
- package/src/control-plane/operator-contract.ts +163 -0
- package/src/control-plane/session-broker.ts +780 -0
- package/src/core/compaction-sections.ts +492 -0
- package/src/core/compaction-types.ts +147 -0
- package/src/core/composer-state.ts +59 -0
- package/src/core/context-compaction.ts +542 -0
- package/src/core/conversation-compaction.ts +68 -0
- package/src/core/conversation-diff.ts +55 -0
- package/src/core/conversation-rendering.ts +343 -0
- package/src/core/conversation-utils.ts +72 -0
- package/src/core/conversation.ts +775 -0
- package/src/core/event-replay.ts +287 -0
- package/src/core/orchestrator-context-runtime.ts +407 -0
- package/src/core/orchestrator-follow-up-runtime.ts +134 -0
- package/src/core/orchestrator-runtime.ts +132 -0
- package/src/core/orchestrator-tool-runtime.ts +468 -0
- package/src/core/orchestrator-turn-helpers.ts +355 -0
- package/src/core/orchestrator-turn-loop.ts +443 -0
- package/src/core/orchestrator.ts +733 -0
- package/src/core/plan-command-handler.ts +169 -0
- package/src/core/system-message-router.ts +210 -0
- package/src/core/transcript-events/classify.ts +95 -0
- package/src/core/transcript-events/index.ts +15 -0
- package/src/daemon/cli.ts +88 -0
- package/src/daemon/control-plane.ts +522 -0
- package/src/daemon/facade-composition.ts +397 -0
- package/src/daemon/facade.ts +638 -0
- package/src/daemon/helpers.ts +74 -0
- package/src/daemon/http/router-route-contexts.ts +370 -0
- package/src/daemon/http/router.ts +531 -0
- package/src/daemon/http-listener.ts +301 -0
- package/src/daemon/index.ts +3 -0
- package/src/daemon/server.ts +1 -0
- package/src/daemon/service-manager.ts +413 -0
- package/src/daemon/surface-actions.ts +183 -0
- package/src/daemon/surface-delivery.ts +530 -0
- package/src/daemon/surface-policy.ts +60 -0
- package/src/daemon/transport-events.ts +110 -0
- package/src/daemon/types.ts +191 -0
- package/src/export/markdown.ts +213 -0
- package/src/export/session-export.ts +633 -0
- package/src/git/index.ts +1 -0
- package/src/git/service.ts +414 -0
- package/src/hooks/chain-engine.ts +414 -0
- package/src/hooks/dispatcher.ts +414 -0
- package/src/hooks/hook-api.ts +170 -0
- package/src/hooks/index.ts +48 -0
- package/src/hooks/runners/agent.ts +93 -0
- package/src/hooks/runners/prompt.ts +69 -0
- package/src/hooks/workbench.ts +360 -0
- package/src/input/autocomplete.ts +96 -0
- package/src/input/bookmark-modal.ts +115 -0
- package/src/input/command-registry.ts +300 -0
- package/src/input/commands/branch-runtime.ts +72 -0
- package/src/input/commands/config.ts +515 -0
- package/src/input/commands/control-room-runtime.ts +255 -0
- package/src/input/commands/conversation-runtime.ts +207 -0
- package/src/input/commands/diff-runtime.ts +161 -0
- package/src/input/commands/discovery-runtime.ts +45 -0
- package/src/input/commands/eval.ts +204 -0
- package/src/input/commands/experience-runtime.ts +278 -0
- package/src/input/commands/git-runtime.ts +81 -0
- package/src/input/commands/guidance-runtime.ts +101 -0
- package/src/input/commands/health-runtime.ts +434 -0
- package/src/input/commands/hooks-runtime.ts +148 -0
- package/src/input/commands/incident-runtime.ts +95 -0
- package/src/input/commands/integration-runtime.ts +394 -0
- package/src/input/commands/intelligence-runtime.ts +223 -0
- package/src/input/commands/knowledge.ts +368 -0
- package/src/input/commands/local-auth-runtime.ts +105 -0
- package/src/input/commands/local-provider-runtime.ts +170 -0
- package/src/input/commands/local-runtime.ts +458 -0
- package/src/input/commands/local-setup-review.ts +192 -0
- package/src/input/commands/local-setup-transfer.ts +134 -0
- package/src/input/commands/local-setup.ts +292 -0
- package/src/input/commands/managed-runtime.ts +208 -0
- package/src/input/commands/marketplace-runtime.ts +290 -0
- package/src/input/commands/mcp-runtime.ts +202 -0
- package/src/input/commands/memory-product-runtime.ts +111 -0
- package/src/input/commands/memory.ts +151 -0
- package/src/input/commands/notify-runtime.ts +83 -0
- package/src/input/commands/operator-panel-runtime.ts +141 -0
- package/src/input/commands/operator-runtime.ts +392 -0
- package/src/input/commands/permissions-runtime.ts +104 -0
- package/src/input/commands/planning-runtime.ts +97 -0
- package/src/input/commands/platform-access-runtime.ts +422 -0
- package/src/input/commands/platform-runtime.ts +6 -0
- package/src/input/commands/platform-sandbox-qemu.ts +137 -0
- package/src/input/commands/platform-sandbox-runtime.ts +406 -0
- package/src/input/commands/platform-sandbox-session.ts +128 -0
- package/src/input/commands/platform-services-runtime.ts +246 -0
- package/src/input/commands/policy-dispatch.ts +339 -0
- package/src/input/commands/policy.ts +17 -0
- package/src/input/commands/product-runtime.ts +351 -0
- package/src/input/commands/profile-sync-runtime.ts +99 -0
- package/src/input/commands/provider-accounts-runtime.ts +113 -0
- package/src/input/commands/provider.ts +363 -0
- package/src/input/commands/quit-shared.ts +162 -0
- package/src/input/commands/recall-bundle.ts +132 -0
- package/src/input/commands/recall-capture.ts +152 -0
- package/src/input/commands/recall-query.ts +229 -0
- package/src/input/commands/recall-review.ts +98 -0
- package/src/input/commands/recall-shared.ts +22 -0
- package/src/input/commands/remote-runtime-pool.ts +106 -0
- package/src/input/commands/remote-runtime-setup.ts +199 -0
- package/src/input/commands/remote-runtime.ts +531 -0
- package/src/input/commands/replay-runtime.ts +18 -0
- package/src/input/commands/runtime-services.ts +279 -0
- package/src/input/commands/schedule-runtime.ts +332 -0
- package/src/input/commands/services-runtime.ts +207 -0
- package/src/input/commands/session-content.ts +408 -0
- package/src/input/commands/session-workflow.ts +464 -0
- package/src/input/commands/session.ts +376 -0
- package/src/input/commands/settings-sync-runtime.ts +173 -0
- package/src/input/commands/share-runtime.ts +114 -0
- package/src/input/commands/shell-core.ts +320 -0
- package/src/input/commands/skills-runtime.ts +221 -0
- package/src/input/commands/subscription-runtime.ts +434 -0
- package/src/input/commands/tasks-runtime.ts +230 -0
- package/src/input/commands/teamwork-runtime.ts +374 -0
- package/src/input/commands/teleport-runtime.ts +57 -0
- package/src/input/commands/worktree-runtime.ts +137 -0
- package/src/input/commands.ts +127 -0
- package/src/input/file-picker.ts +192 -0
- package/src/input/handler-command-route.ts +106 -0
- package/src/input/handler-content-actions.ts +465 -0
- package/src/input/handler-feed-routes.ts +541 -0
- package/src/input/handler-feed.ts +361 -0
- package/src/input/handler-modal-routes.ts +335 -0
- package/src/input/handler-modal-stack.ts +237 -0
- package/src/input/handler-modal-token-routes.ts +272 -0
- package/src/input/handler-picker-routes.ts +416 -0
- package/src/input/handler-prompt-buffer.ts +320 -0
- package/src/input/handler-shortcuts.ts +195 -0
- package/src/input/handler-ui-state.ts +294 -0
- package/src/input/handler.ts +798 -0
- package/src/input/input-history.ts +267 -0
- package/src/input/keybindings.ts +256 -0
- package/src/input/model-picker.ts +730 -0
- package/src/input/panel-integration-actions.ts +77 -0
- package/src/input/profile-picker-modal.ts +222 -0
- package/src/input/search.ts +100 -0
- package/src/input/selection-modal.ts +163 -0
- package/src/input/selection.ts +135 -0
- package/src/input/session-picker-modal.ts +136 -0
- package/src/input/settings-modal.ts +718 -0
- package/src/input/submission-intent.ts +18 -0
- package/src/input/submission-router.ts +64 -0
- package/src/integrations/index.ts +42 -0
- package/src/integrations/notifier.ts +206 -0
- package/src/integrations/webhooks.ts +177 -0
- package/src/knowledge/consolidation.ts +346 -0
- package/src/knowledge/graphql.ts +324 -0
- package/src/knowledge/index.ts +60 -0
- package/src/knowledge/ingest-compile.ts +386 -0
- package/src/knowledge/ingest-context.ts +18 -0
- package/src/knowledge/ingest-inputs.ts +387 -0
- package/src/knowledge/ingest.ts +20 -0
- package/src/knowledge/internal.ts +257 -0
- package/src/knowledge/knowledge-api.ts +432 -0
- package/src/knowledge/lint.ts +121 -0
- package/src/knowledge/memory-sync.ts +62 -0
- package/src/knowledge/packet.ts +370 -0
- package/src/knowledge/scheduling.ts +283 -0
- package/src/knowledge/service.ts +715 -0
- package/src/main.ts +798 -0
- package/src/mcp/client.ts +383 -0
- package/src/mcp/index.ts +12 -0
- package/src/mcp/mcp-api.ts +90 -0
- package/src/mcp/registry.ts +508 -0
- package/src/media/builtin-image-understanding.ts +303 -0
- package/src/media/builtin-providers.ts +26 -0
- package/src/media/index.ts +18 -0
- package/src/multimodal/index.ts +13 -0
- package/src/multimodal/service.ts +492 -0
- package/src/panels/agent-inspector-panel.ts +515 -0
- package/src/panels/agent-inspector-shared.ts +94 -0
- package/src/panels/agent-logs-panel.ts +539 -0
- package/src/panels/agent-logs-shared.ts +129 -0
- package/src/panels/approval-panel.ts +169 -0
- package/src/panels/automation-control-panel.ts +253 -0
- package/src/panels/base-panel.ts +72 -0
- package/src/panels/builtin/agent.ts +88 -0
- package/src/panels/builtin/development.ts +111 -0
- package/src/panels/builtin/knowledge.ts +26 -0
- package/src/panels/builtin/operations.ts +385 -0
- package/src/panels/builtin/session.ts +61 -0
- package/src/panels/builtin/shared.ts +240 -0
- package/src/panels/builtin-panels.ts +23 -0
- package/src/panels/cockpit-panel.ts +183 -0
- package/src/panels/communication-panel.ts +191 -0
- package/src/panels/context-visualizer-panel.ts +199 -0
- package/src/panels/control-plane-panel.ts +266 -0
- package/src/panels/cost-tracker-panel.ts +444 -0
- package/src/panels/debug-panel.ts +432 -0
- package/src/panels/diff-panel.ts +518 -0
- package/src/panels/docs-panel.ts +283 -0
- package/src/panels/eval-panel.ts +399 -0
- package/src/panels/file-explorer-panel.ts +556 -0
- package/src/panels/file-preview-panel.ts +412 -0
- package/src/panels/forensics-panel.ts +364 -0
- package/src/panels/git-panel.ts +630 -0
- package/src/panels/hooks-panel.ts +274 -0
- package/src/panels/incident-review-panel.ts +247 -0
- package/src/panels/index.ts +48 -0
- package/src/panels/intelligence-panel.ts +176 -0
- package/src/panels/knowledge-panel.ts +328 -0
- package/src/panels/local-auth-panel.ts +146 -0
- package/src/panels/marketplace-panel.ts +223 -0
- package/src/panels/mcp-panel.ts +260 -0
- package/src/panels/memory-panel.ts +293 -0
- package/src/panels/ops-control-panel.ts +184 -0
- package/src/panels/ops-strategy-panel.ts +235 -0
- package/src/panels/orchestration-panel.ts +254 -0
- package/src/panels/panel-list-panel.ts +508 -0
- package/src/panels/panel-manager.ts +538 -0
- package/src/panels/panel-picker.ts +106 -0
- package/src/panels/plan-dashboard-panel.ts +272 -0
- package/src/panels/plugins-panel.ts +201 -0
- package/src/panels/policy-panel.ts +308 -0
- package/src/panels/polish.ts +668 -0
- package/src/panels/provider-account-snapshot.ts +259 -0
- package/src/panels/provider-accounts-panel.ts +221 -0
- package/src/panels/provider-health-domains.ts +211 -0
- package/src/panels/provider-health-panel.ts +725 -0
- package/src/panels/provider-health-tracker.ts +115 -0
- package/src/panels/provider-stats-panel.ts +366 -0
- package/src/panels/remote-panel.ts +449 -0
- package/src/panels/routes-panel.ts +228 -0
- package/src/panels/sandbox-panel.ts +289 -0
- package/src/panels/schedule-panel.ts +344 -0
- package/src/panels/search-focus.ts +32 -0
- package/src/panels/security-panel.ts +329 -0
- package/src/panels/services-panel.ts +271 -0
- package/src/panels/session-browser-panel.ts +399 -0
- package/src/panels/session-maintenance.ts +125 -0
- package/src/panels/settings-sync-panel.ts +164 -0
- package/src/panels/skills-panel.ts +475 -0
- package/src/panels/subscription-panel.ts +273 -0
- package/src/panels/symbol-outline-panel.ts +486 -0
- package/src/panels/system-messages-panel.ts +224 -0
- package/src/panels/tasks-panel.ts +448 -0
- package/src/panels/thinking-panel.ts +304 -0
- package/src/panels/token-budget-panel.ts +469 -0
- package/src/panels/tool-inspector-panel.ts +434 -0
- package/src/panels/types.ts +44 -0
- package/src/panels/watchers-panel.ts +241 -0
- package/src/panels/welcome-panel.ts +64 -0
- package/src/panels/worktree-panel.ts +180 -0
- package/src/panels/wrfc-panel.ts +480 -0
- package/src/permissions/briefs/build.ts +88 -0
- package/src/permissions/manager.ts +356 -0
- package/src/permissions/prompt.ts +184 -0
- package/src/plugins/api.ts +383 -0
- package/src/plugins/loader.ts +304 -0
- package/src/plugins/manager.ts +481 -0
- package/src/profiles/shape.ts +58 -0
- package/src/providers/amazon-bedrock-mantle.ts +50 -0
- package/src/providers/amazon-bedrock.ts +61 -0
- package/src/providers/anthropic-compat.ts +373 -0
- package/src/providers/anthropic-sdk-provider.ts +230 -0
- package/src/providers/anthropic-vertex.ts +59 -0
- package/src/providers/anthropic.ts +469 -0
- package/src/providers/auto-register.ts +417 -0
- package/src/providers/builtin-catalog.ts +326 -0
- package/src/providers/builtin-registry.ts +575 -0
- package/src/providers/cache-planner.ts +258 -0
- package/src/providers/capabilities.ts +601 -0
- package/src/providers/custom-loader.ts +425 -0
- package/src/providers/discovered-compat.ts +18 -0
- package/src/providers/discovered-factory.ts +61 -0
- package/src/providers/discovered-traits.ts +138 -0
- package/src/providers/gemini.ts +462 -0
- package/src/providers/github-copilot.ts +254 -0
- package/src/providers/index.ts +47 -0
- package/src/providers/interface.ts +185 -0
- package/src/providers/llama-cpp.ts +402 -0
- package/src/providers/lm-studio-helpers.ts +367 -0
- package/src/providers/lm-studio.ts +484 -0
- package/src/providers/model-catalog-cache.ts +221 -0
- package/src/providers/model-catalog-notifications.ts +97 -0
- package/src/providers/model-catalog-synthetic.ts +202 -0
- package/src/providers/model-catalog.ts +211 -0
- package/src/providers/model-limits.ts +280 -0
- package/src/providers/ollama.ts +469 -0
- package/src/providers/openai-codex.ts +472 -0
- package/src/providers/openai-compat.ts +615 -0
- package/src/providers/openai.ts +231 -0
- package/src/providers/optimizer.ts +381 -0
- package/src/providers/provider-api.ts +553 -0
- package/src/providers/registry-helpers.ts +34 -0
- package/src/providers/registry-models.ts +77 -0
- package/src/providers/registry-types.ts +67 -0
- package/src/providers/registry.ts +729 -0
- package/src/providers/runtime-metadata.ts +149 -0
- package/src/providers/runtime-snapshot.ts +130 -0
- package/src/providers/synthetic.ts +561 -0
- package/src/providers/tier-prompts.ts +84 -0
- package/src/providers/tool-formats.ts +414 -0
- package/src/renderer/agent-detail-modal.ts +285 -0
- package/src/renderer/autocomplete-overlay.ts +154 -0
- package/src/renderer/block-actions.ts +76 -0
- package/src/renderer/bookmark-modal.ts +101 -0
- package/src/renderer/bottom-bar.ts +58 -0
- package/src/renderer/buffer.ts +34 -0
- package/src/renderer/code-block.ts +373 -0
- package/src/renderer/compositor.ts +261 -0
- package/src/renderer/context-inspector.ts +219 -0
- package/src/renderer/conversation-layout.ts +67 -0
- package/src/renderer/conversation-overlays.ts +123 -0
- package/src/renderer/conversation-surface.ts +260 -0
- package/src/renderer/diff-view.ts +132 -0
- package/src/renderer/diff.ts +122 -0
- package/src/renderer/file-picker-overlay.ts +101 -0
- package/src/renderer/file-tree.ts +153 -0
- package/src/renderer/git-status.ts +89 -0
- package/src/renderer/help-overlay.ts +247 -0
- package/src/renderer/history-search-overlay.ts +73 -0
- package/src/renderer/layout-engine.ts +97 -0
- package/src/renderer/layout.ts +32 -0
- package/src/renderer/live-tail-modal.ts +156 -0
- package/src/renderer/markdown.ts +777 -0
- package/src/renderer/modal-factory.ts +467 -0
- package/src/renderer/modal-utils.ts +24 -0
- package/src/renderer/model-picker-overlay.ts +396 -0
- package/src/renderer/overlay-box.ts +165 -0
- package/src/renderer/overlay-viewport.ts +104 -0
- package/src/renderer/panel-composite.ts +80 -0
- package/src/renderer/panel-picker-overlay.ts +202 -0
- package/src/renderer/panel-tab-bar.ts +69 -0
- package/src/renderer/panel-workspace-bar.ts +38 -0
- package/src/renderer/process-indicator.ts +96 -0
- package/src/renderer/process-modal.ts +295 -0
- package/src/renderer/profile-picker-modal.ts +129 -0
- package/src/renderer/progress.ts +98 -0
- package/src/renderer/search-overlay.ts +54 -0
- package/src/renderer/selection-modal-overlay.ts +214 -0
- package/src/renderer/semantic-diff.ts +369 -0
- package/src/renderer/session-picker-modal.ts +127 -0
- package/src/renderer/settings-modal.ts +701 -0
- package/src/renderer/shell-surface.ts +88 -0
- package/src/renderer/surface-layout.ts +101 -0
- package/src/renderer/syntax-highlighter.ts +542 -0
- package/src/renderer/system-message.ts +83 -0
- package/src/renderer/tab-strip.ts +108 -0
- package/src/renderer/text-layout.ts +31 -0
- package/src/renderer/thinking.ts +17 -0
- package/src/renderer/tool-call.ts +233 -0
- package/src/renderer/ui-factory.ts +524 -0
- package/src/renderer/ui-primitives.ts +96 -0
- package/src/runtime/auth/inspection.ts +125 -0
- package/src/runtime/bootstrap-background.ts +147 -0
- package/src/runtime/bootstrap-command-context.ts +265 -0
- package/src/runtime/bootstrap-command-parts.ts +357 -0
- package/src/runtime/bootstrap-core.ts +375 -0
- package/src/runtime/bootstrap-helpers.ts +88 -0
- package/src/runtime/bootstrap-hook-bridge.ts +271 -0
- package/src/runtime/bootstrap-runtime-events.ts +254 -0
- package/src/runtime/bootstrap-services.ts +197 -0
- package/src/runtime/bootstrap-shell.ts +262 -0
- package/src/runtime/bootstrap.ts +488 -0
- package/src/runtime/compaction/index.ts +90 -0
- package/src/runtime/compaction/lifecycle.ts +167 -0
- package/src/runtime/compaction/manager.ts +474 -0
- package/src/runtime/compaction/quality-score.ts +279 -0
- package/src/runtime/compaction/resume-repair.ts +183 -0
- package/src/runtime/compaction/strategies/autocompact.ts +65 -0
- package/src/runtime/compaction/strategies/boundary-commit.ts +106 -0
- package/src/runtime/compaction/strategies/collapse.ts +90 -0
- package/src/runtime/compaction/strategies/index.ts +23 -0
- package/src/runtime/compaction/strategies/microcompact.ts +74 -0
- package/src/runtime/compaction/strategies/reactive.ts +89 -0
- package/src/runtime/compaction/types.ts +221 -0
- package/src/runtime/context.ts +158 -0
- package/src/runtime/diagnostics/actions.ts +776 -0
- package/src/runtime/diagnostics/index.ts +99 -0
- package/src/runtime/diagnostics/panels/agents.ts +252 -0
- package/src/runtime/diagnostics/panels/events.ts +188 -0
- package/src/runtime/diagnostics/panels/health.ts +242 -0
- package/src/runtime/diagnostics/panels/index.ts +24 -0
- package/src/runtime/diagnostics/panels/ops.ts +156 -0
- package/src/runtime/diagnostics/panels/policy.ts +176 -0
- package/src/runtime/diagnostics/panels/tasks.ts +251 -0
- package/src/runtime/diagnostics/panels/tool-calls.ts +267 -0
- package/src/runtime/diagnostics/provider.ts +262 -0
- package/src/runtime/ecosystem/catalog.ts +606 -0
- package/src/runtime/ecosystem/recommendations.ts +117 -0
- package/src/runtime/emitters/agents.ts +96 -0
- package/src/runtime/emitters/automation.ts +112 -0
- package/src/runtime/emitters/communication.ts +53 -0
- package/src/runtime/emitters/compaction.ts +161 -0
- package/src/runtime/emitters/control-plane.ts +65 -0
- package/src/runtime/emitters/deliveries.ts +65 -0
- package/src/runtime/emitters/forensics.ts +17 -0
- package/src/runtime/emitters/index.ts +59 -0
- package/src/runtime/emitters/knowledge.ts +129 -0
- package/src/runtime/emitters/mcp.ts +95 -0
- package/src/runtime/emitters/ops.ts +163 -0
- package/src/runtime/emitters/orchestration.ts +87 -0
- package/src/runtime/emitters/permissions.ts +98 -0
- package/src/runtime/emitters/planner.ts +23 -0
- package/src/runtime/emitters/plugins.ts +78 -0
- package/src/runtime/emitters/providers.ts +30 -0
- package/src/runtime/emitters/routes.ts +57 -0
- package/src/runtime/emitters/security.ts +53 -0
- package/src/runtime/emitters/session.ts +93 -0
- package/src/runtime/emitters/surfaces.ts +57 -0
- package/src/runtime/emitters/tasks.ts +69 -0
- package/src/runtime/emitters/tools.ts +140 -0
- package/src/runtime/emitters/transport.ts +78 -0
- package/src/runtime/emitters/turn.ts +155 -0
- package/src/runtime/emitters/ui.ts +57 -0
- package/src/runtime/emitters/watchers.ts +57 -0
- package/src/runtime/emitters/workflows.ts +79 -0
- package/src/runtime/eval/index.ts +48 -0
- package/src/runtime/eval/runner.ts +163 -0
- package/src/runtime/eval/suites.ts +264 -0
- package/src/runtime/events/domain-map.ts +148 -0
- package/src/runtime/events/index.ts +194 -0
- package/src/runtime/events/turn.ts +60 -0
- package/src/runtime/events/workflows.ts +17 -0
- package/src/runtime/forensics/collector.ts +693 -0
- package/src/runtime/forensics/index.ts +23 -0
- package/src/runtime/foundation-clients.ts +78 -0
- package/src/runtime/foundation-services.ts +96 -0
- package/src/runtime/guidance.ts +183 -0
- package/src/runtime/health/effect-handlers.ts +189 -0
- package/src/runtime/health/index.ts +70 -0
- package/src/runtime/health/wiring.ts +115 -0
- package/src/runtime/index.ts +174 -0
- package/src/runtime/integration/helpers.ts +640 -0
- package/src/runtime/lifecycle.ts +107 -0
- package/src/runtime/mcp/index.ts +68 -0
- package/src/runtime/mcp/manager.ts +513 -0
- package/src/runtime/network/inbound.ts +131 -0
- package/src/runtime/network/index.ts +30 -0
- package/src/runtime/network/outbound.ts +292 -0
- package/src/runtime/network/shared.ts +82 -0
- package/src/runtime/operator-client.ts +235 -0
- package/src/runtime/ops/control-plane.ts +363 -0
- package/src/runtime/ops/index.ts +122 -0
- package/src/runtime/ops/playbooks/index.ts +10 -0
- package/src/runtime/ops/playbooks/session-unrecoverable.ts +196 -0
- package/src/runtime/ops/playbooks/stuck-turn.ts +197 -0
- package/src/runtime/ops/runtime-context.ts +100 -0
- package/src/runtime/ops-api.ts +27 -0
- package/src/runtime/orchestration/spawn-policy.ts +83 -0
- package/src/runtime/peer-client.ts +404 -0
- package/src/runtime/perf/index.ts +57 -0
- package/src/runtime/perf/slo-collector.ts +375 -0
- package/src/runtime/permissions/index.ts +190 -0
- package/src/runtime/permissions/policy-runtime.ts +175 -0
- package/src/runtime/permissions/preflight.ts +101 -0
- package/src/runtime/permissions/rule-suggestions.ts +36 -0
- package/src/runtime/plugins/hot-reload.ts +221 -0
- package/src/runtime/plugins/index.ts +84 -0
- package/src/runtime/plugins/lifecycle.ts +95 -0
- package/src/runtime/plugins/manager.ts +474 -0
- package/src/runtime/plugins/manifest.ts +167 -0
- package/src/runtime/plugins/quarantine.ts +202 -0
- package/src/runtime/plugins/trust.ts +291 -0
- package/src/runtime/plugins/types.ts +205 -0
- package/src/runtime/provider-accounts/registry.ts +326 -0
- package/src/runtime/remote/distributed-runtime-contract-schemas.ts +386 -0
- package/src/runtime/remote/index.ts +488 -0
- package/src/runtime/remote/runner-registry.ts +438 -0
- package/src/runtime/remote/supervisor.ts +70 -0
- package/src/runtime/runtime-hook-api.ts +5 -0
- package/src/runtime/runtime-knowledge-api.ts +14 -0
- package/src/runtime/runtime-mcp-api.ts +5 -0
- package/src/runtime/runtime-ops-api.ts +86 -0
- package/src/runtime/runtime-provider-api.ts +18 -0
- package/src/runtime/sandbox/backend.ts +291 -0
- package/src/runtime/sandbox/manager.ts +364 -0
- package/src/runtime/sandbox/provisioning.ts +422 -0
- package/src/runtime/sandbox/session-registry.ts +289 -0
- package/src/runtime/services.ts +541 -0
- package/src/runtime/session-maintenance.ts +188 -0
- package/src/runtime/session-persistence.ts +288 -0
- package/src/runtime/session-return-context.ts +195 -0
- package/src/runtime/settings/control-plane-store.ts +258 -0
- package/src/runtime/settings/control-plane.ts +599 -0
- package/src/runtime/shell-command-extensions.ts +54 -0
- package/src/runtime/shell-command-ops.ts +207 -0
- package/src/runtime/shell-command-platform.ts +47 -0
- package/src/runtime/shell-command-services.ts +143 -0
- package/src/runtime/shell-command-workspace.ts +31 -0
- package/src/runtime/store/domains/conversation.ts +181 -0
- package/src/runtime/store/domains/domain-read-matrix.ts +17 -0
- package/src/runtime/store/domains/index.ts +222 -0
- package/src/runtime/store/domains/panels.ts +117 -0
- package/src/runtime/store/domains/permissions.ts +143 -0
- package/src/runtime/store/domains/ui-perf.ts +103 -0
- package/src/runtime/store/helpers/reducers/conversation.ts +228 -0
- package/src/runtime/store/helpers/reducers/lifecycle.ts +440 -0
- package/src/runtime/store/helpers/reducers/shared.ts +60 -0
- package/src/runtime/store/helpers/reducers/sync.ts +555 -0
- package/src/runtime/store/helpers/reducers.ts +30 -0
- package/src/runtime/store/index.ts +304 -0
- package/src/runtime/store/selectors/index.ts +354 -0
- package/src/runtime/store/state.ts +137 -0
- package/src/runtime/tasks/adapters/acp-adapter.ts +211 -0
- package/src/runtime/tasks/adapters/agent-adapter.ts +208 -0
- package/src/runtime/tasks/adapters/index.ts +16 -0
- package/src/runtime/tasks/adapters/process-adapter.ts +214 -0
- package/src/runtime/tasks/adapters/scheduler-adapter.ts +193 -0
- package/src/runtime/tasks/index.ts +68 -0
- package/src/runtime/tasks/manager.ts +415 -0
- package/src/runtime/telemetry/api-helpers.ts +517 -0
- package/src/runtime/telemetry/api.ts +768 -0
- package/src/runtime/telemetry/index.ts +178 -0
- package/src/runtime/telemetry/instrumentation/domain-bridge-agent-session.ts +440 -0
- package/src/runtime/telemetry/instrumentation/domain-bridge-plugin-mcp.ts +200 -0
- package/src/runtime/telemetry/instrumentation/domain-bridge-shared.ts +18 -0
- package/src/runtime/telemetry/instrumentation/domain-bridge-transport-task.ts +204 -0
- package/src/runtime/telemetry/instrumentation/domain-bridge.ts +125 -0
- package/src/runtime/telemetry/instrumentation/index.ts +67 -0
- package/src/runtime/tools/context.ts +114 -0
- package/src/runtime/tools/index.ts +46 -0
- package/src/runtime/tools/phased-executor.ts +448 -0
- package/src/runtime/tools/phases/budget.ts +130 -0
- package/src/runtime/tools/phases/execute.ts +69 -0
- package/src/runtime/tools/phases/index.ts +13 -0
- package/src/runtime/tools/phases/map-output.ts +98 -0
- package/src/runtime/tools/phases/permission.ts +133 -0
- package/src/runtime/tools/phases/posthook.ts +57 -0
- package/src/runtime/tools/phases/prehook.ts +68 -0
- package/src/runtime/tools/phases/validate.ts +53 -0
- package/src/runtime/transports/direct.ts +73 -0
- package/src/runtime/transports/http-helpers.ts +218 -0
- package/src/runtime/transports/http-types.ts +364 -0
- package/src/runtime/transports/http.ts +629 -0
- package/src/runtime/transports/realtime.ts +50 -0
- package/src/runtime/transports/remote-events.ts +16 -0
- package/src/runtime/transports/shared.ts +39 -0
- package/src/runtime/transports/ui-runtime-events.ts +35 -0
- package/src/runtime/ui/index.ts +39 -0
- package/src/runtime/ui/model-picker/data-provider.ts +182 -0
- package/src/runtime/ui/model-picker/health-enrichment.ts +228 -0
- package/src/runtime/ui/model-picker/index.ts +59 -0
- package/src/runtime/ui/model-picker/types.ts +149 -0
- package/src/runtime/ui/provider-health/data-provider.ts +244 -0
- package/src/runtime/ui/provider-health/fallback-visualizer.ts +69 -0
- package/src/runtime/ui/provider-health/index.ts +46 -0
- package/src/runtime/ui/provider-health/types.ts +146 -0
- package/src/runtime/ui-events.ts +46 -0
- package/src/runtime/ui-read-model-helpers.ts +32 -0
- package/src/runtime/ui-read-models-core.ts +95 -0
- package/src/runtime/ui-read-models-observability-maintenance.ts +81 -0
- package/src/runtime/ui-read-models-observability-options.ts +5 -0
- package/src/runtime/ui-read-models-observability-remote.ts +73 -0
- package/src/runtime/ui-read-models-observability-security.ts +172 -0
- package/src/runtime/ui-read-models-observability-system.ts +217 -0
- package/src/runtime/ui-read-models-observability.ts +59 -0
- package/src/runtime/ui-read-models-operations.ts +203 -0
- package/src/runtime/ui-read-models.ts +61 -0
- package/src/runtime/ui-service-queries.ts +114 -0
- package/src/runtime/ui-services.ts +163 -0
- package/src/runtime/worktree/registry.ts +252 -0
- package/src/scripts/process-messages.ts +42 -0
- package/src/sessions/manager.ts +388 -0
- package/src/shell/blocking-input.ts +89 -0
- package/src/shell/ui-openers.ts +163 -0
- package/src/state/file-watcher.ts +294 -0
- package/src/state/index.ts +56 -0
- package/src/state/knowledge-injection.ts +214 -0
- package/src/state/memory-embedding-http.ts +642 -0
- package/src/state/memory-embeddings.ts +312 -0
- package/src/state/memory-ingest.ts +132 -0
- package/src/state/memory-registry.ts +111 -0
- package/src/state/memory-store-helpers.ts +160 -0
- package/src/state/memory-store.ts +728 -0
- package/src/state/memory-vector-store.ts +418 -0
- package/src/templates/manager.ts +187 -0
- package/src/tools/agent/index.ts +610 -0
- package/src/tools/agent/manager.ts +476 -0
- package/src/tools/analyze/git-modes.ts +380 -0
- package/src/tools/analyze/index.ts +128 -0
- package/src/tools/channel/agent-tools.ts +16 -0
- package/src/tools/channel/index.ts +268 -0
- package/src/tools/control/index.ts +90 -0
- package/src/tools/edit/core.ts +619 -0
- package/src/tools/edit/index.ts +4 -0
- package/src/tools/edit/phased.ts +33 -0
- package/src/tools/fetch/index.ts +3 -0
- package/src/tools/fetch/phased.ts +34 -0
- package/src/tools/fetch/runtime.ts +499 -0
- package/src/tools/index.ts +186 -0
- package/src/tools/mcp/index.ts +190 -0
- package/src/tools/remote-trigger/index.ts +130 -0
- package/src/tools/repl/index.ts +318 -0
- package/src/tools/shared/auto-heal.ts +282 -0
- package/src/tools/state/index.ts +688 -0
- package/src/tools/web-search/index.ts +38 -0
- package/src/tools/write/index.ts +604 -0
- package/src/tools/write/phased.ts +41 -0
- package/src/types/generated/foundation-client-types.ts +22 -0
- package/src/types/sql-js.d.ts +15 -0
- package/src/utils/splash-lines.ts +46 -0
- package/src/version.ts +17 -0
- package/src/watchers/index.ts +11 -0
- package/src/watchers/registry.ts +517 -0
- package/src/web-search/index.ts +26 -0
- package/src/web-search/provider-registry.ts +64 -0
- package/src/web-search/providers/brave.ts +100 -0
- package/src/web-search/providers/duckduckgo.ts +270 -0
- package/src/web-search/providers/exa.ts +77 -0
- package/src/web-search/providers/firecrawl.ts +90 -0
- package/src/web-search/providers/perplexity.ts +86 -0
- package/src/web-search/providers/searxng.ts +88 -0
- package/src/web-search/providers/shared.ts +249 -0
- package/src/web-search/providers/tavily.ts +90 -0
- package/src/web-search/service.ts +142 -0
- package/src/widget/index.ts +2 -0
- package/src/widget/types.ts +9 -0
- package/src/widget/widget.ts +8 -0
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Plugin capability manifest validation and resolution.
|
|
3
|
+
*
|
|
4
|
+
* Implements deny-by-default capability enforcement:
|
|
5
|
+
* 1. Parse raw manifest capabilities (unknown → typed)
|
|
6
|
+
* 2. Validate that each requested capability is a known capability
|
|
7
|
+
* 3. Run the runtime's capability policy to grant or deny each capability
|
|
8
|
+
* 4. Return a resolved PluginCapabilityManifest
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import { logger } from '@pellux/goodvibes-sdk/platform/utils/logger';
|
|
12
|
+
import {
|
|
13
|
+
ALL_CAPABILITIES,
|
|
14
|
+
HIGH_RISK_CAPABILITIES,
|
|
15
|
+
type PluginCapability,
|
|
16
|
+
type PluginCapabilityManifest,
|
|
17
|
+
type PluginManifestV2,
|
|
18
|
+
} from './types.ts';
|
|
19
|
+
import { type PluginTrustTier, filterCapabilitiesByTrust } from './trust.ts';
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Default capability policy: grant all valid capabilities.
|
|
23
|
+
*
|
|
24
|
+
* Callers can supply a stricter policy via PluginLifecycleManagerOptions.
|
|
25
|
+
*/
|
|
26
|
+
function defaultPolicy(_pluginName: string, _capability: PluginCapability): boolean {
|
|
27
|
+
return true;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
/**
|
|
31
|
+
* Returns whether a string value is a known PluginCapability.
|
|
32
|
+
*/
|
|
33
|
+
function isKnownCapability(value: string): value is PluginCapability {
|
|
34
|
+
return (ALL_CAPABILITIES as ReadonlyArray<string>).includes(value);
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* resolveCapabilityManifest — Parse and resolve the capability manifest for a
|
|
39
|
+
* plugin, applying the runtime's capability policy and trust-tier constraints.
|
|
40
|
+
*
|
|
41
|
+
* Evaluation order:
|
|
42
|
+
* 1. Unknown capability strings are filtered out (warn + ignore).
|
|
43
|
+
* 2. Trust-tier constraints are applied — high-risk capabilities blocked
|
|
44
|
+
* unless the plugin has the `trusted` tier.
|
|
45
|
+
* 3. The runtime capability policy callback is applied to the remaining set.
|
|
46
|
+
*
|
|
47
|
+
* @param pluginName - Plugin name (for logging).
|
|
48
|
+
* @param manifest - The raw plugin manifest (may have `capabilities` array).
|
|
49
|
+
* @param policy - Capability grant/deny callback. Defaults to permissive.
|
|
50
|
+
* @param trustTier - Effective trust tier for this plugin. Defaults to 'untrusted'.
|
|
51
|
+
* @returns A fully-resolved PluginCapabilityManifest.
|
|
52
|
+
*/
|
|
53
|
+
export function resolveCapabilityManifest(
|
|
54
|
+
pluginName: string,
|
|
55
|
+
manifest: PluginManifestV2,
|
|
56
|
+
policy: (name: string, cap: PluginCapability) => boolean = defaultPolicy,
|
|
57
|
+
trustTier: PluginTrustTier = 'untrusted',
|
|
58
|
+
): PluginCapabilityManifest {
|
|
59
|
+
const rawRequested = manifest.capabilities ?? [];
|
|
60
|
+
|
|
61
|
+
// Validate and filter unknown capability strings.
|
|
62
|
+
const requested: PluginCapability[] = [];
|
|
63
|
+
for (const raw of rawRequested) {
|
|
64
|
+
if (isKnownCapability(raw)) {
|
|
65
|
+
requested.push(raw);
|
|
66
|
+
} else {
|
|
67
|
+
logger.warn(
|
|
68
|
+
`[plugin-lifecycle:${pluginName}] Unknown capability '${raw}' — ignored`,
|
|
69
|
+
);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// Apply trust-tier constraints before the policy callback.
|
|
74
|
+
const trustResult = filterCapabilitiesByTrust(requested, trustTier);
|
|
75
|
+
if (trustResult.blocked.length > 0) {
|
|
76
|
+
logger.warn(
|
|
77
|
+
`[plugin-lifecycle:${pluginName}] Trust tier '${trustTier}' blocks` +
|
|
78
|
+
` high-risk capabilities: [${trustResult.blocked.join(', ')}]` +
|
|
79
|
+
' — trust escalation required',
|
|
80
|
+
);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// Only the trust-permitted capabilities proceed to policy evaluation.
|
|
84
|
+
const afterTrust = trustResult.permitted;
|
|
85
|
+
|
|
86
|
+
const granted: PluginCapability[] = [];
|
|
87
|
+
const denied: PluginCapability[] = [...trustResult.blocked];
|
|
88
|
+
const denialReasons: Partial<Record<PluginCapability, string>> = { ...trustResult.reasons };
|
|
89
|
+
|
|
90
|
+
for (const cap of afterTrust) {
|
|
91
|
+
if (policy(pluginName, cap)) {
|
|
92
|
+
granted.push(cap);
|
|
93
|
+
} else {
|
|
94
|
+
denied.push(cap);
|
|
95
|
+
denialReasons[cap] = `Capability '${cap}' denied by runtime policy`;
|
|
96
|
+
logger.warn(
|
|
97
|
+
`[plugin-lifecycle:${pluginName}] Capability '${cap}' denied by policy`,
|
|
98
|
+
);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
logger.debug(
|
|
103
|
+
`[plugin-lifecycle:${pluginName}] Capabilities resolved (trust=${trustTier})` +
|
|
104
|
+
` — granted: [${granted.join(', ')}]` +
|
|
105
|
+
(denied.length > 0 ? `, denied: [${denied.join(', ')}]` : ''),
|
|
106
|
+
);
|
|
107
|
+
|
|
108
|
+
return {
|
|
109
|
+
requested: Object.freeze(requested),
|
|
110
|
+
granted,
|
|
111
|
+
denied,
|
|
112
|
+
denialReasons,
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
/**
|
|
117
|
+
* isHighRiskCapability — Returns whether a capability is classified as high-risk.
|
|
118
|
+
* High-risk capabilities require the `trusted` tier to be granted.
|
|
119
|
+
*/
|
|
120
|
+
export function isHighRiskCapability(capability: PluginCapability): boolean {
|
|
121
|
+
return (HIGH_RISK_CAPABILITIES as ReadonlyArray<string>).includes(capability);
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
/**
|
|
125
|
+
* hasCapability — Returns whether a plugin has been granted a specific
|
|
126
|
+
* capability after manifest resolution.
|
|
127
|
+
*/
|
|
128
|
+
export function hasCapability(
|
|
129
|
+
manifest: PluginCapabilityManifest,
|
|
130
|
+
capability: PluginCapability,
|
|
131
|
+
): boolean {
|
|
132
|
+
return manifest.granted.includes(capability);
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
/**
|
|
136
|
+
* validateManifestV2 — Light validation of the PluginManifestV2 shape.
|
|
137
|
+
* Returns null on success or an error string on failure.
|
|
138
|
+
*/
|
|
139
|
+
export function validateManifestV2(manifest: unknown): string | null {
|
|
140
|
+
if (!manifest || typeof manifest !== 'object') {
|
|
141
|
+
return 'manifest must be an object';
|
|
142
|
+
}
|
|
143
|
+
const m = manifest as Record<string, unknown>;
|
|
144
|
+
if (typeof m['name'] !== 'string' || !m['name']) {
|
|
145
|
+
return "manifest.name must be a non-empty string";
|
|
146
|
+
}
|
|
147
|
+
if (typeof m['version'] !== 'string' || !m['version']) {
|
|
148
|
+
return "manifest.version must be a non-empty string";
|
|
149
|
+
}
|
|
150
|
+
if (typeof m['description'] !== 'string') {
|
|
151
|
+
return "manifest.description must be a string";
|
|
152
|
+
}
|
|
153
|
+
if (m['capabilities'] !== undefined) {
|
|
154
|
+
if (!Array.isArray(m['capabilities'])) {
|
|
155
|
+
return "manifest.capabilities must be an array";
|
|
156
|
+
}
|
|
157
|
+
for (const cap of m['capabilities'] as unknown[]) {
|
|
158
|
+
if (typeof cap !== 'string') {
|
|
159
|
+
return `manifest.capabilities entries must be strings, got: ${typeof cap}`;
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
if (m['minRuntimeVersion'] !== undefined && typeof m['minRuntimeVersion'] !== 'string') {
|
|
164
|
+
return "manifest.minRuntimeVersion must be a string";
|
|
165
|
+
}
|
|
166
|
+
return null;
|
|
167
|
+
}
|
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Plugin quarantine engine.
|
|
3
|
+
*
|
|
4
|
+
* Quarantine removes a plugin's unsafe contribution effects without fully
|
|
5
|
+
* unloading it. This allows the operator to isolate a suspicious plugin,
|
|
6
|
+
* inspect it, then either restore or permanently disable it.
|
|
7
|
+
*
|
|
8
|
+
* Quarantine effects:
|
|
9
|
+
* - All high-risk capabilities are revoked in the resolved manifest.
|
|
10
|
+
* - The plugin is moved to a `quarantined` lifecycle bucket in the store.
|
|
11
|
+
* - A quarantine record is created with a timestamp and reason.
|
|
12
|
+
*
|
|
13
|
+
* Restore path:
|
|
14
|
+
* - `lift()` — Restores previously revoked capabilities (if trust was upgraded).
|
|
15
|
+
* - The caller is responsible for reloading the plugin after lifting.
|
|
16
|
+
*/
|
|
17
|
+
|
|
18
|
+
import { logger } from '@pellux/goodvibes-sdk/platform/utils/logger';
|
|
19
|
+
import type { PluginCapability, PluginCapabilityManifest } from './types.ts';
|
|
20
|
+
import { isHighRiskCapability } from './manifest.ts';
|
|
21
|
+
|
|
22
|
+
// ── Quarantine Record ─────────────────────────────────────────────────────────
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* A record describing a plugin currently in quarantine.
|
|
26
|
+
*/
|
|
27
|
+
export interface QuarantineRecord {
|
|
28
|
+
/** Plugin name. */
|
|
29
|
+
readonly pluginName: string;
|
|
30
|
+
/** Unix epoch ms when quarantine was applied. */
|
|
31
|
+
readonly quarantinedAt: number;
|
|
32
|
+
/** Human-readable reason for quarantine. */
|
|
33
|
+
readonly reason: string;
|
|
34
|
+
/** The capabilities that were revoked when quarantine was applied. */
|
|
35
|
+
readonly revokedCapabilities: ReadonlyArray<PluginCapability>;
|
|
36
|
+
/** Whether the quarantine has been lifted. */
|
|
37
|
+
lifted: boolean;
|
|
38
|
+
/** Unix epoch ms when quarantine was lifted, if applicable. */
|
|
39
|
+
liftedAt?: number;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// ── Quarantine Engine ─────────────────────────────────────────────────────────
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* PluginQuarantineEngine — Tracks quarantined plugins and applies/revokes
|
|
46
|
+
* capability restrictions.
|
|
47
|
+
*
|
|
48
|
+
* This is intentionally separate from the PluginLifecycleManager so that
|
|
49
|
+
* quarantine can be applied without triggering a full state machine transition.
|
|
50
|
+
* The lifecycle manager delegates to this engine when quarantine is requested.
|
|
51
|
+
*/
|
|
52
|
+
export class PluginQuarantineEngine {
|
|
53
|
+
private readonly records = new Map<string, QuarantineRecord>();
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* quarantine — Apply quarantine to a plugin.
|
|
57
|
+
*
|
|
58
|
+
* Revokes all high-risk capabilities from the plugin's resolved manifest
|
|
59
|
+
* and creates a quarantine record. The plugin remains in memory but its
|
|
60
|
+
* unsafe contributions are neutralised.
|
|
61
|
+
*
|
|
62
|
+
* @param pluginName - Plugin identifier.
|
|
63
|
+
* @param capabilityManifest - The plugin's live capability manifest (mutated in place).
|
|
64
|
+
* @param reason - Human-readable reason for quarantine.
|
|
65
|
+
* @returns The quarantine record, or null if already quarantined.
|
|
66
|
+
*/
|
|
67
|
+
quarantine(
|
|
68
|
+
pluginName: string,
|
|
69
|
+
capabilityManifest: PluginCapabilityManifest,
|
|
70
|
+
reason: string,
|
|
71
|
+
): QuarantineRecord | null {
|
|
72
|
+
if (this.isQuarantined(pluginName)) {
|
|
73
|
+
logger.warn(`[plugin-quarantine] ${pluginName}: already quarantined — skipping`);
|
|
74
|
+
return null;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
// Identify which currently-granted capabilities are high-risk.
|
|
78
|
+
const revokedCapabilities: PluginCapability[] = capabilityManifest.granted.filter(
|
|
79
|
+
(cap) => isHighRiskCapability(cap),
|
|
80
|
+
);
|
|
81
|
+
|
|
82
|
+
// Strip high-risk capabilities from the live manifest.
|
|
83
|
+
capabilityManifest.granted = capabilityManifest.granted.filter(
|
|
84
|
+
(cap) => !isHighRiskCapability(cap),
|
|
85
|
+
);
|
|
86
|
+
|
|
87
|
+
// Record denied reason for each revoked cap. Collect first, then assign once.
|
|
88
|
+
const newDenied: PluginCapability[] = [];
|
|
89
|
+
for (const cap of revokedCapabilities) {
|
|
90
|
+
newDenied.push(cap);
|
|
91
|
+
capabilityManifest.denialReasons[cap] = `Capability '${cap}' revoked: plugin quarantined — ${reason}`;
|
|
92
|
+
}
|
|
93
|
+
capabilityManifest.denied = [...capabilityManifest.denied, ...newDenied];
|
|
94
|
+
|
|
95
|
+
const record: QuarantineRecord = {
|
|
96
|
+
pluginName,
|
|
97
|
+
quarantinedAt: Date.now(),
|
|
98
|
+
reason,
|
|
99
|
+
revokedCapabilities: Object.freeze(revokedCapabilities),
|
|
100
|
+
lifted: false,
|
|
101
|
+
};
|
|
102
|
+
|
|
103
|
+
this.records.set(pluginName, record);
|
|
104
|
+
|
|
105
|
+
logger.warn(
|
|
106
|
+
`[plugin-quarantine] ${pluginName}: quarantined — ${reason}` +
|
|
107
|
+
(revokedCapabilities.length > 0
|
|
108
|
+
? ` (revoked: [${revokedCapabilities.join(', ')}])`
|
|
109
|
+
: ' (no high-risk capabilities were granted)'),
|
|
110
|
+
);
|
|
111
|
+
|
|
112
|
+
return record;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* lift — Lift quarantine for a plugin.
|
|
117
|
+
*
|
|
118
|
+
* Previously revoked capabilities are NOT automatically restored here;
|
|
119
|
+
* the caller should trigger a re-resolve of the capability manifest
|
|
120
|
+
* (e.g. by reloading the plugin) after lifting so that trust-tier
|
|
121
|
+
* constraints are re-evaluated with the new tier.
|
|
122
|
+
*
|
|
123
|
+
* @returns true if quarantine was successfully lifted; false if not found.
|
|
124
|
+
*/
|
|
125
|
+
lift(pluginName: string): boolean {
|
|
126
|
+
const record = this.records.get(pluginName);
|
|
127
|
+
if (!record) {
|
|
128
|
+
logger.debug(`[plugin-quarantine] ${pluginName}: no quarantine record found — nothing to lift`);
|
|
129
|
+
return false;
|
|
130
|
+
}
|
|
131
|
+
if (record.lifted) {
|
|
132
|
+
logger.debug(`[plugin-quarantine] ${pluginName}: quarantine already lifted`);
|
|
133
|
+
return false;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
record.lifted = true;
|
|
137
|
+
record.liftedAt = Date.now();
|
|
138
|
+
|
|
139
|
+
logger.info(`[plugin-quarantine] ${pluginName}: quarantine lifted`);
|
|
140
|
+
return true;
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
/** Returns whether a plugin is currently quarantined (and not lifted). */
|
|
144
|
+
isQuarantined(pluginName: string): boolean {
|
|
145
|
+
const record = this.records.get(pluginName);
|
|
146
|
+
return record !== undefined && !record.lifted;
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
/** Returns the quarantine record for a plugin, or undefined. */
|
|
150
|
+
getRecord(pluginName: string): Readonly<QuarantineRecord> | undefined {
|
|
151
|
+
return this.records.get(pluginName);
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
/** Returns all quarantine records (including lifted ones). */
|
|
155
|
+
getAllRecords(): ReadonlyArray<Readonly<QuarantineRecord>> {
|
|
156
|
+
return Array.from(this.records.values());
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
/** Returns only active (not-lifted) quarantine records. */
|
|
160
|
+
getActiveQuarantines(): ReadonlyArray<Readonly<QuarantineRecord>> {
|
|
161
|
+
return Array.from(this.records.values()).filter((r) => !r.lifted);
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
/**
|
|
165
|
+
* applyToNewManifest — Apply quarantine constraints to a freshly-resolved
|
|
166
|
+
* capability manifest. Used when a plugin is reloaded while under quarantine.
|
|
167
|
+
*
|
|
168
|
+
* Unlike `quarantine()`, this does not create a new record — it reuses the
|
|
169
|
+
* existing one. Call this during manifest re-resolution if `isQuarantined()`
|
|
170
|
+
* is true.
|
|
171
|
+
*/
|
|
172
|
+
applyToNewManifest(
|
|
173
|
+
pluginName: string,
|
|
174
|
+
capabilityManifest: PluginCapabilityManifest,
|
|
175
|
+
): void {
|
|
176
|
+
if (!this.isQuarantined(pluginName)) return;
|
|
177
|
+
|
|
178
|
+
const toRevoke: PluginCapability[] = capabilityManifest.granted.filter(
|
|
179
|
+
(cap) => isHighRiskCapability(cap),
|
|
180
|
+
);
|
|
181
|
+
|
|
182
|
+
if (toRevoke.length === 0) return;
|
|
183
|
+
|
|
184
|
+
capabilityManifest.granted = capabilityManifest.granted.filter(
|
|
185
|
+
(cap) => !isHighRiskCapability(cap),
|
|
186
|
+
);
|
|
187
|
+
|
|
188
|
+
// Collect all denied caps first, then assign once to avoid quadratic churn.
|
|
189
|
+
const reason = this.records.get(pluginName)?.reason ?? 'quarantined';
|
|
190
|
+
const newDenied: PluginCapability[] = [];
|
|
191
|
+
for (const cap of toRevoke) {
|
|
192
|
+
newDenied.push(cap);
|
|
193
|
+
capabilityManifest.denialReasons[cap] = `Capability '${cap}' blocked: plugin is quarantined — ${reason}`;
|
|
194
|
+
}
|
|
195
|
+
capabilityManifest.denied = [...capabilityManifest.denied, ...newDenied];
|
|
196
|
+
|
|
197
|
+
logger.debug(
|
|
198
|
+
`[plugin-quarantine] ${pluginName}: quarantine re-applied to reloaded manifest` +
|
|
199
|
+
` (blocked: [${toRevoke.join(', ')}])`,
|
|
200
|
+
);
|
|
201
|
+
}
|
|
202
|
+
}
|
|
@@ -0,0 +1,291 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Plugin extension trust framework.
|
|
3
|
+
*
|
|
4
|
+
* Defines trust tiers (untrusted, limited, trusted), signed manifest
|
|
5
|
+
* validation for the trusted tier, and the PluginTrustStore that manages
|
|
6
|
+
* trust records with persistence support.
|
|
7
|
+
*
|
|
8
|
+
* Trust tiers gate access to high-risk capabilities:
|
|
9
|
+
* - untrusted — only safe, read-only capabilities allowed
|
|
10
|
+
* - limited — moderate capabilities; high-risk capabilities blocked
|
|
11
|
+
* - trusted — full capability set; requires signed manifest validation
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
import { createHmac, timingSafeEqual } from 'node:crypto';
|
|
15
|
+
import { logger } from '@pellux/goodvibes-sdk/platform/utils/logger';
|
|
16
|
+
import type { PluginCapability } from './types.ts';
|
|
17
|
+
import { isHighRiskCapability } from './manifest.ts';
|
|
18
|
+
|
|
19
|
+
// ── Trust Tier ────────────────────────────────────────────────────────────────
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* The three trust tiers available to a plugin.
|
|
23
|
+
*
|
|
24
|
+
* - `untrusted` — Default for newly discovered plugins. Only safe capabilities
|
|
25
|
+
* are accessible. The plugin may not have been reviewed.
|
|
26
|
+
* - `limited` — Operator-reviewed plugin. Moderate capabilities granted.
|
|
27
|
+
* High-risk capabilities (shell.exec, filesystem.write, network.outbound)
|
|
28
|
+
* remain blocked without explicit trust escalation.
|
|
29
|
+
* - `trusted` — Fully trusted plugin. Requires a valid signed manifest.
|
|
30
|
+
* All declared capabilities may be granted (subject to runtime policy).
|
|
31
|
+
*/
|
|
32
|
+
export type PluginTrustTier = 'untrusted' | 'limited' | 'trusted';
|
|
33
|
+
|
|
34
|
+
// ── Trust Record ──────────────────────────────────────────────────────────────
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* A persisted trust record for a single plugin.
|
|
38
|
+
*/
|
|
39
|
+
export interface PluginTrustRecord {
|
|
40
|
+
/** Plugin identifier (manifest name). */
|
|
41
|
+
readonly pluginName: string;
|
|
42
|
+
/** Current trust tier. */
|
|
43
|
+
tier: PluginTrustTier;
|
|
44
|
+
/** Unix epoch ms when the trust record was last updated. */
|
|
45
|
+
updatedAt: number;
|
|
46
|
+
/** Who or what granted this trust level. */
|
|
47
|
+
grantedBy: 'operator' | 'signed-manifest';
|
|
48
|
+
/**
|
|
49
|
+
* Fingerprint of the verified signature for trusted-tier plugins.
|
|
50
|
+
* Undefined for untrusted/limited plugins.
|
|
51
|
+
*/
|
|
52
|
+
signatureFingerprint?: string;
|
|
53
|
+
/** Optional human-readable note attached by the operator. */
|
|
54
|
+
note?: string;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
// ── Signature Validation ──────────────────────────────────────────────────────
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Result of validating a plugin's signed manifest.
|
|
61
|
+
*/
|
|
62
|
+
export interface SignatureValidationResult {
|
|
63
|
+
/** Whether the signature is valid. */
|
|
64
|
+
valid: boolean;
|
|
65
|
+
/** A stable fingerprint derived from the signature (e.g. hex digest prefix). */
|
|
66
|
+
fingerprint?: string;
|
|
67
|
+
/** Human-readable failure reason. Only set when `valid` is false. */
|
|
68
|
+
reason?: string;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* validatePluginSignature — Validates the manifest signature for a plugin
|
|
73
|
+
* seeking the `trusted` tier.
|
|
74
|
+
*
|
|
75
|
+
* The signature field in PluginManifestV2 is expected to be a base64-encoded
|
|
76
|
+
* HMAC-SHA256 of the canonical manifest JSON (name + version + capabilities
|
|
77
|
+
* sorted and serialised). For production use, callers should supply a real
|
|
78
|
+
* key; this implementation uses a structural check so external tooling can
|
|
79
|
+
* provide real crypto without requiring Node.js crypto APIs at import time.
|
|
80
|
+
*
|
|
81
|
+
* @param manifest - The raw manifest object containing the `signature` field.
|
|
82
|
+
* @param publicKey - Optional verification key. When omitted, structural
|
|
83
|
+
* validity only is checked (suitable for CI/test).
|
|
84
|
+
*/
|
|
85
|
+
export function validatePluginSignature(
|
|
86
|
+
manifest: { name: string; version: string; capabilities?: string[]; signature?: string },
|
|
87
|
+
publicKey?: string,
|
|
88
|
+
): SignatureValidationResult {
|
|
89
|
+
const { name, version, capabilities = [], signature } = manifest;
|
|
90
|
+
|
|
91
|
+
if (!signature || typeof signature !== 'string' || signature.trim().length === 0) {
|
|
92
|
+
return { valid: false, reason: 'No signature field present in manifest' };
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// Structural check: signature must be a non-empty hex or base64 string.
|
|
96
|
+
const isStructurallyValid = /^[A-Za-z0-9+/=]{32,}$/.test(signature.trim());
|
|
97
|
+
if (!isStructurallyValid) {
|
|
98
|
+
return { valid: false, reason: 'Signature field does not match expected format (base64/hex, min 32 chars)' };
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
// Canonical payload that should have been signed.
|
|
102
|
+
const sortedCapabilities = [...capabilities].sort();
|
|
103
|
+
const payload = JSON.stringify({ name, version, capabilities: sortedCapabilities });
|
|
104
|
+
|
|
105
|
+
// When a public key is provided, perform full HMAC verification.
|
|
106
|
+
if (publicKey) {
|
|
107
|
+
const expected = createHmac('sha256', publicKey)
|
|
108
|
+
.update(payload)
|
|
109
|
+
.digest('base64');
|
|
110
|
+
const sigBuf = Buffer.from(signature.trim(), 'base64');
|
|
111
|
+
const expBuf = Buffer.from(expected, 'base64');
|
|
112
|
+
if (sigBuf.length !== expBuf.length || !timingSafeEqual(sigBuf, expBuf)) {
|
|
113
|
+
return { valid: false, reason: 'HMAC mismatch' };
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
// Derive a short fingerprint for record keeping.
|
|
118
|
+
const fingerprint = signature.trim().slice(0, 16);
|
|
119
|
+
|
|
120
|
+
logger.debug(
|
|
121
|
+
`[plugin-trust] Manifest signature validated — plugin=${name} fingerprint=${fingerprint}` +
|
|
122
|
+
(publicKey ? ' (full HMAC)' : ' (structural only)'),
|
|
123
|
+
);
|
|
124
|
+
|
|
125
|
+
return { valid: true, fingerprint };
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
// ── Capability filtering by trust tier ───────────────────────────────────────
|
|
129
|
+
|
|
130
|
+
/**
|
|
131
|
+
* Capabilities that are safe for any trust tier (including untrusted).
|
|
132
|
+
*/
|
|
133
|
+
export const SAFE_CAPABILITIES: ReadonlyArray<PluginCapability> = [
|
|
134
|
+
'register.tool',
|
|
135
|
+
'register.provider',
|
|
136
|
+
'register.panel',
|
|
137
|
+
'register.hook',
|
|
138
|
+
'filesystem.read',
|
|
139
|
+
] as const;
|
|
140
|
+
|
|
141
|
+
/**
|
|
142
|
+
* filterCapabilitiesByTrust — Returns the subset of `requested` capabilities
|
|
143
|
+
* that are permitted for the given trust tier.
|
|
144
|
+
*
|
|
145
|
+
* - `untrusted`: only SAFE_CAPABILITIES
|
|
146
|
+
* - `limited`: all capabilities except HIGH_RISK_CAPABILITIES
|
|
147
|
+
* - `trusted`: all capabilities (HIGH_RISK_CAPABILITIES included)
|
|
148
|
+
*/
|
|
149
|
+
export function filterCapabilitiesByTrust(
|
|
150
|
+
requested: ReadonlyArray<PluginCapability>,
|
|
151
|
+
tier: PluginTrustTier,
|
|
152
|
+
): { permitted: PluginCapability[]; blocked: PluginCapability[]; reasons: Partial<Record<PluginCapability, string>> } {
|
|
153
|
+
const permitted: PluginCapability[] = [];
|
|
154
|
+
const blocked: PluginCapability[] = [];
|
|
155
|
+
const reasons: Partial<Record<PluginCapability, string>> = {};
|
|
156
|
+
|
|
157
|
+
for (const cap of requested) {
|
|
158
|
+
if (tier === 'trusted') {
|
|
159
|
+
permitted.push(cap);
|
|
160
|
+
} else if (tier === 'limited') {
|
|
161
|
+
if (isHighRiskCapability(cap)) {
|
|
162
|
+
blocked.push(cap);
|
|
163
|
+
reasons[cap] = `Capability '${cap}' requires trust tier 'trusted' (current: limited)`;
|
|
164
|
+
} else {
|
|
165
|
+
permitted.push(cap);
|
|
166
|
+
}
|
|
167
|
+
} else {
|
|
168
|
+
// untrusted
|
|
169
|
+
if ((SAFE_CAPABILITIES as ReadonlyArray<string>).includes(cap)) {
|
|
170
|
+
permitted.push(cap);
|
|
171
|
+
} else {
|
|
172
|
+
blocked.push(cap);
|
|
173
|
+
reasons[cap] = `Capability '${cap}' requires trust tier 'limited' or higher (current: untrusted)`;
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
return { permitted, blocked, reasons };
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
// ── Trust Store ───────────────────────────────────────────────────────────────
|
|
182
|
+
|
|
183
|
+
/**
|
|
184
|
+
* PluginTrustStore — In-memory trust registry for all plugins.
|
|
185
|
+
*
|
|
186
|
+
* Callers are responsible for persistence (serialise/deserialise via
|
|
187
|
+
* `exportRecords` / `importRecords`). The PluginManager bridges this to
|
|
188
|
+
* the plugins.json state file.
|
|
189
|
+
*/
|
|
190
|
+
export class PluginTrustStore {
|
|
191
|
+
private readonly records = new Map<string, PluginTrustRecord>();
|
|
192
|
+
|
|
193
|
+
/**
|
|
194
|
+
* Returns the trust record for a plugin, or `undefined` if not yet assessed.
|
|
195
|
+
* Callers should treat `undefined` as implicitly `untrusted`.
|
|
196
|
+
*/
|
|
197
|
+
getRecord(pluginName: string): Readonly<PluginTrustRecord> | undefined {
|
|
198
|
+
return this.records.get(pluginName);
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
/**
|
|
202
|
+
* Returns the trust tier for a plugin.
|
|
203
|
+
* Plugins without an explicit record are treated as `untrusted`.
|
|
204
|
+
*/
|
|
205
|
+
getTier(pluginName: string): PluginTrustTier {
|
|
206
|
+
return this.records.get(pluginName)?.tier ?? 'untrusted';
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
/**
|
|
210
|
+
* setTier — Explicitly assign a trust tier to a plugin.
|
|
211
|
+
*
|
|
212
|
+
* Intended for operator use via `/plugin trust`.
|
|
213
|
+
* For the `trusted` tier, prefer `trustSigned()` which also validates the signature.
|
|
214
|
+
*/
|
|
215
|
+
setTier(
|
|
216
|
+
pluginName: string,
|
|
217
|
+
tier: PluginTrustTier,
|
|
218
|
+
options: { note?: string } = {},
|
|
219
|
+
): PluginTrustRecord {
|
|
220
|
+
const record: PluginTrustRecord = {
|
|
221
|
+
pluginName,
|
|
222
|
+
tier,
|
|
223
|
+
updatedAt: Date.now(),
|
|
224
|
+
grantedBy: 'operator',
|
|
225
|
+
note: options.note,
|
|
226
|
+
};
|
|
227
|
+
this.records.set(pluginName, record);
|
|
228
|
+
logger.info(`[plugin-trust] ${pluginName}: tier set to '${tier}'${options.note ? ` — ${options.note}` : ''}`);
|
|
229
|
+
return record;
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
/**
|
|
233
|
+
* trustSigned — Elevate a plugin to the `trusted` tier after verifying its
|
|
234
|
+
* signed manifest. Returns `{ ok: false, reason }` if validation fails.
|
|
235
|
+
*/
|
|
236
|
+
trustSigned(
|
|
237
|
+
pluginName: string,
|
|
238
|
+
manifest: { name: string; version: string; capabilities?: string[]; signature?: string },
|
|
239
|
+
publicKey?: string,
|
|
240
|
+
): { ok: true; record: PluginTrustRecord } | { ok: false; reason: string } {
|
|
241
|
+
const validation = validatePluginSignature(manifest, publicKey);
|
|
242
|
+
if (!validation.valid) {
|
|
243
|
+
logger.warn(`[plugin-trust] ${pluginName}: signature validation failed — ${validation.reason}`);
|
|
244
|
+
return { ok: false, reason: validation.reason! };
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
const record: PluginTrustRecord = {
|
|
248
|
+
pluginName,
|
|
249
|
+
tier: 'trusted',
|
|
250
|
+
updatedAt: Date.now(),
|
|
251
|
+
grantedBy: 'signed-manifest',
|
|
252
|
+
signatureFingerprint: validation.fingerprint,
|
|
253
|
+
};
|
|
254
|
+
this.records.set(pluginName, record);
|
|
255
|
+
logger.info(`[plugin-trust] ${pluginName}: elevated to 'trusted' via signed manifest (fingerprint=${validation.fingerprint})`);
|
|
256
|
+
return { ok: true, record };
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
/**
|
|
260
|
+
* verify — Verify the current signature on a plugin manifest without
|
|
261
|
+
* changing its tier. Useful for `/plugin verify` inspection.
|
|
262
|
+
*/
|
|
263
|
+
verify(
|
|
264
|
+
manifest: { name: string; version: string; capabilities?: string[]; signature?: string },
|
|
265
|
+
publicKey?: string,
|
|
266
|
+
): SignatureValidationResult {
|
|
267
|
+
return validatePluginSignature(manifest, publicKey);
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
/** Returns all trust records as an array. */
|
|
271
|
+
getAllRecords(): ReadonlyArray<Readonly<PluginTrustRecord>> {
|
|
272
|
+
return Array.from(this.records.values());
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
/** Export all records for persistence. */
|
|
276
|
+
exportRecords(): Record<string, PluginTrustRecord> {
|
|
277
|
+
const out: Record<string, PluginTrustRecord> = {};
|
|
278
|
+
for (const [name, record] of this.records) {
|
|
279
|
+
out[name] = { ...record };
|
|
280
|
+
}
|
|
281
|
+
return out;
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
/** Import records from persisted state. Merges into existing records. */
|
|
285
|
+
importRecords(records: Record<string, PluginTrustRecord>): void {
|
|
286
|
+
for (const [name, record] of Object.entries(records)) {
|
|
287
|
+
this.records.set(name, record);
|
|
288
|
+
}
|
|
289
|
+
logger.debug(`[plugin-trust] Imported ${Object.keys(records).length} trust record(s)`);
|
|
290
|
+
}
|
|
291
|
+
}
|