cowork-os 0.3.21
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/LICENSE +21 -0
- package/README.md +1638 -0
- package/bin/cowork.js +42 -0
- package/build/entitlements.mac.plist +16 -0
- package/build/icon.icns +0 -0
- package/build/icon.png +0 -0
- package/dist/electron/electron/activity/ActivityRepository.js +190 -0
- package/dist/electron/electron/agent/browser/browser-service.js +639 -0
- package/dist/electron/electron/agent/context-manager.js +225 -0
- package/dist/electron/electron/agent/custom-skill-loader.js +566 -0
- package/dist/electron/electron/agent/daemon.js +975 -0
- package/dist/electron/electron/agent/executor.js +3561 -0
- package/dist/electron/electron/agent/llm/anthropic-provider.js +155 -0
- package/dist/electron/electron/agent/llm/bedrock-provider.js +202 -0
- package/dist/electron/electron/agent/llm/gemini-provider.js +375 -0
- package/dist/electron/electron/agent/llm/index.js +34 -0
- package/dist/electron/electron/agent/llm/ollama-provider.js +263 -0
- package/dist/electron/electron/agent/llm/openai-oauth.js +101 -0
- package/dist/electron/electron/agent/llm/openai-provider.js +657 -0
- package/dist/electron/electron/agent/llm/openrouter-provider.js +232 -0
- package/dist/electron/electron/agent/llm/pricing.js +160 -0
- package/dist/electron/electron/agent/llm/provider-factory.js +880 -0
- package/dist/electron/electron/agent/llm/types.js +178 -0
- package/dist/electron/electron/agent/queue-manager.js +378 -0
- package/dist/electron/electron/agent/sandbox/docker-sandbox.js +402 -0
- package/dist/electron/electron/agent/sandbox/macos-sandbox.js +407 -0
- package/dist/electron/electron/agent/sandbox/runner.js +410 -0
- package/dist/electron/electron/agent/sandbox/sandbox-factory.js +228 -0
- package/dist/electron/electron/agent/sandbox/security-utils.js +258 -0
- package/dist/electron/electron/agent/search/brave-provider.js +119 -0
- package/dist/electron/electron/agent/search/google-provider.js +100 -0
- package/dist/electron/electron/agent/search/index.js +28 -0
- package/dist/electron/electron/agent/search/provider-factory.js +395 -0
- package/dist/electron/electron/agent/search/serpapi-provider.js +112 -0
- package/dist/electron/electron/agent/search/tavily-provider.js +90 -0
- package/dist/electron/electron/agent/search/types.js +40 -0
- package/dist/electron/electron/agent/security/index.js +12 -0
- package/dist/electron/electron/agent/security/input-sanitizer.js +303 -0
- package/dist/electron/electron/agent/security/output-filter.js +217 -0
- package/dist/electron/electron/agent/skill-eligibility.js +281 -0
- package/dist/electron/electron/agent/skill-registry.js +396 -0
- package/dist/electron/electron/agent/skills/document.js +878 -0
- package/dist/electron/electron/agent/skills/image-generator.js +225 -0
- package/dist/electron/electron/agent/skills/organizer.js +141 -0
- package/dist/electron/electron/agent/skills/presentation.js +367 -0
- package/dist/electron/electron/agent/skills/spreadsheet.js +165 -0
- package/dist/electron/electron/agent/tools/browser-tools.js +523 -0
- package/dist/electron/electron/agent/tools/builtin-settings.js +384 -0
- package/dist/electron/electron/agent/tools/canvas-tools.js +530 -0
- package/dist/electron/electron/agent/tools/cron-tools.js +577 -0
- package/dist/electron/electron/agent/tools/edit-tools.js +194 -0
- package/dist/electron/electron/agent/tools/file-tools.js +719 -0
- package/dist/electron/electron/agent/tools/glob-tools.js +283 -0
- package/dist/electron/electron/agent/tools/grep-tools.js +387 -0
- package/dist/electron/electron/agent/tools/image-tools.js +111 -0
- package/dist/electron/electron/agent/tools/mention-tools.js +282 -0
- package/dist/electron/electron/agent/tools/node-tools.js +476 -0
- package/dist/electron/electron/agent/tools/registry.js +2719 -0
- package/dist/electron/electron/agent/tools/search-tools.js +91 -0
- package/dist/electron/electron/agent/tools/shell-tools.js +574 -0
- package/dist/electron/electron/agent/tools/skill-tools.js +274 -0
- package/dist/electron/electron/agent/tools/system-tools.js +578 -0
- package/dist/electron/electron/agent/tools/web-fetch-tools.js +444 -0
- package/dist/electron/electron/agent/tools/x-tools.js +264 -0
- package/dist/electron/electron/agents/AgentRoleRepository.js +420 -0
- package/dist/electron/electron/agents/HeartbeatService.js +356 -0
- package/dist/electron/electron/agents/MentionRepository.js +197 -0
- package/dist/electron/electron/agents/TaskSubscriptionRepository.js +168 -0
- package/dist/electron/electron/agents/WorkingStateRepository.js +229 -0
- package/dist/electron/electron/canvas/canvas-manager.js +714 -0
- package/dist/electron/electron/canvas/canvas-preload.js +53 -0
- package/dist/electron/electron/canvas/canvas-protocol.js +195 -0
- package/dist/electron/electron/canvas/canvas-store.js +174 -0
- package/dist/electron/electron/canvas/index.js +13 -0
- package/dist/electron/electron/control-plane/client.js +364 -0
- package/dist/electron/electron/control-plane/handlers.js +572 -0
- package/dist/electron/electron/control-plane/index.js +41 -0
- package/dist/electron/electron/control-plane/node-manager.js +264 -0
- package/dist/electron/electron/control-plane/protocol.js +194 -0
- package/dist/electron/electron/control-plane/remote-client.js +437 -0
- package/dist/electron/electron/control-plane/server.js +640 -0
- package/dist/electron/electron/control-plane/settings.js +369 -0
- package/dist/electron/electron/control-plane/ssh-tunnel.js +549 -0
- package/dist/electron/electron/cron/index.js +30 -0
- package/dist/electron/electron/cron/schedule.js +190 -0
- package/dist/electron/electron/cron/service.js +614 -0
- package/dist/electron/electron/cron/store.js +155 -0
- package/dist/electron/electron/cron/types.js +82 -0
- package/dist/electron/electron/cron/webhook.js +258 -0
- package/dist/electron/electron/database/SecureSettingsRepository.js +444 -0
- package/dist/electron/electron/database/TaskLabelRepository.js +120 -0
- package/dist/electron/electron/database/repositories.js +1781 -0
- package/dist/electron/electron/database/schema.js +978 -0
- package/dist/electron/electron/extensions/index.js +33 -0
- package/dist/electron/electron/extensions/loader.js +313 -0
- package/dist/electron/electron/extensions/registry.js +485 -0
- package/dist/electron/electron/extensions/types.js +11 -0
- package/dist/electron/electron/gateway/channel-registry.js +1102 -0
- package/dist/electron/electron/gateway/channels/bluebubbles-client.js +479 -0
- package/dist/electron/electron/gateway/channels/bluebubbles.js +432 -0
- package/dist/electron/electron/gateway/channels/discord.js +975 -0
- package/dist/electron/electron/gateway/channels/email-client.js +593 -0
- package/dist/electron/electron/gateway/channels/email.js +443 -0
- package/dist/electron/electron/gateway/channels/google-chat.js +631 -0
- package/dist/electron/electron/gateway/channels/imessage-client.js +363 -0
- package/dist/electron/electron/gateway/channels/imessage.js +465 -0
- package/dist/electron/electron/gateway/channels/index.js +36 -0
- package/dist/electron/electron/gateway/channels/line-client.js +470 -0
- package/dist/electron/electron/gateway/channels/line.js +479 -0
- package/dist/electron/electron/gateway/channels/matrix-client.js +432 -0
- package/dist/electron/electron/gateway/channels/matrix.js +592 -0
- package/dist/electron/electron/gateway/channels/mattermost-client.js +394 -0
- package/dist/electron/electron/gateway/channels/mattermost.js +496 -0
- package/dist/electron/electron/gateway/channels/signal-client.js +500 -0
- package/dist/electron/electron/gateway/channels/signal.js +582 -0
- package/dist/electron/electron/gateway/channels/slack.js +415 -0
- package/dist/electron/electron/gateway/channels/teams.js +596 -0
- package/dist/electron/electron/gateway/channels/telegram.js +1390 -0
- package/dist/electron/electron/gateway/channels/twitch-client.js +502 -0
- package/dist/electron/electron/gateway/channels/twitch.js +396 -0
- package/dist/electron/electron/gateway/channels/types.js +8 -0
- package/dist/electron/electron/gateway/channels/whatsapp.js +953 -0
- package/dist/electron/electron/gateway/context-policy.js +268 -0
- package/dist/electron/electron/gateway/index.js +1063 -0
- package/dist/electron/electron/gateway/infrastructure.js +496 -0
- package/dist/electron/electron/gateway/router.js +2700 -0
- package/dist/electron/electron/gateway/security.js +375 -0
- package/dist/electron/electron/gateway/session.js +115 -0
- package/dist/electron/electron/gateway/tunnel.js +503 -0
- package/dist/electron/electron/guardrails/guardrail-manager.js +348 -0
- package/dist/electron/electron/hooks/gmail-watcher.js +300 -0
- package/dist/electron/electron/hooks/index.js +46 -0
- package/dist/electron/electron/hooks/mappings.js +381 -0
- package/dist/electron/electron/hooks/server.js +480 -0
- package/dist/electron/electron/hooks/settings.js +447 -0
- package/dist/electron/electron/hooks/types.js +41 -0
- package/dist/electron/electron/ipc/canvas-handlers.js +158 -0
- package/dist/electron/electron/ipc/handlers.js +3138 -0
- package/dist/electron/electron/ipc/mission-control-handlers.js +141 -0
- package/dist/electron/electron/main.js +448 -0
- package/dist/electron/electron/mcp/client/MCPClientManager.js +330 -0
- package/dist/electron/electron/mcp/client/MCPServerConnection.js +437 -0
- package/dist/electron/electron/mcp/client/transports/SSETransport.js +304 -0
- package/dist/electron/electron/mcp/client/transports/StdioTransport.js +307 -0
- package/dist/electron/electron/mcp/client/transports/WebSocketTransport.js +329 -0
- package/dist/electron/electron/mcp/host/MCPHostServer.js +354 -0
- package/dist/electron/electron/mcp/host/ToolAdapter.js +100 -0
- package/dist/electron/electron/mcp/registry/MCPRegistryManager.js +497 -0
- package/dist/electron/electron/mcp/settings.js +446 -0
- package/dist/electron/electron/mcp/types.js +59 -0
- package/dist/electron/electron/memory/MemoryService.js +435 -0
- package/dist/electron/electron/notifications/index.js +17 -0
- package/dist/electron/electron/notifications/service.js +118 -0
- package/dist/electron/electron/notifications/store.js +144 -0
- package/dist/electron/electron/preload.js +842 -0
- package/dist/electron/electron/reports/StandupReportService.js +272 -0
- package/dist/electron/electron/security/concurrency.js +293 -0
- package/dist/electron/electron/security/index.js +15 -0
- package/dist/electron/electron/security/policy-manager.js +435 -0
- package/dist/electron/electron/settings/appearance-manager.js +193 -0
- package/dist/electron/electron/settings/personality-manager.js +724 -0
- package/dist/electron/electron/settings/x-manager.js +58 -0
- package/dist/electron/electron/tailscale/exposure.js +188 -0
- package/dist/electron/electron/tailscale/index.js +28 -0
- package/dist/electron/electron/tailscale/settings.js +205 -0
- package/dist/electron/electron/tailscale/tailscale.js +355 -0
- package/dist/electron/electron/tray/QuickInputWindow.js +568 -0
- package/dist/electron/electron/tray/TrayManager.js +895 -0
- package/dist/electron/electron/tray/index.js +9 -0
- package/dist/electron/electron/updater/index.js +6 -0
- package/dist/electron/electron/updater/update-manager.js +418 -0
- package/dist/electron/electron/utils/env-migration.js +209 -0
- package/dist/electron/electron/utils/process.js +102 -0
- package/dist/electron/electron/utils/rate-limiter.js +104 -0
- package/dist/electron/electron/utils/validation.js +419 -0
- package/dist/electron/electron/utils/x-cli.js +177 -0
- package/dist/electron/electron/voice/VoiceService.js +507 -0
- package/dist/electron/electron/voice/index.js +14 -0
- package/dist/electron/electron/voice/voice-settings-manager.js +359 -0
- package/dist/electron/shared/channelMessages.js +170 -0
- package/dist/electron/shared/types.js +1185 -0
- package/package.json +159 -0
- package/resources/skills/1password.json +10 -0
- package/resources/skills/add-documentation.json +31 -0
- package/resources/skills/analyze-csv.json +17 -0
- package/resources/skills/apple-notes.json +10 -0
- package/resources/skills/apple-reminders.json +10 -0
- package/resources/skills/auto-commenter.json +10 -0
- package/resources/skills/bear-notes.json +10 -0
- package/resources/skills/bird.json +35 -0
- package/resources/skills/blogwatcher.json +10 -0
- package/resources/skills/blucli.json +10 -0
- package/resources/skills/bluebubbles.json +10 -0
- package/resources/skills/camsnap.json +10 -0
- package/resources/skills/clean-imports.json +18 -0
- package/resources/skills/code-review.json +18 -0
- package/resources/skills/coding-agent.json +10 -0
- package/resources/skills/compare-files.json +23 -0
- package/resources/skills/convert-code.json +34 -0
- package/resources/skills/create-changelog.json +24 -0
- package/resources/skills/debug-error.json +17 -0
- package/resources/skills/dependency-check.json +10 -0
- package/resources/skills/discord.json +10 -0
- package/resources/skills/eightctl.json +10 -0
- package/resources/skills/explain-code.json +29 -0
- package/resources/skills/extract-todos.json +18 -0
- package/resources/skills/food-order.json +10 -0
- package/resources/skills/gemini.json +10 -0
- package/resources/skills/generate-readme.json +10 -0
- package/resources/skills/gifgrep.json +10 -0
- package/resources/skills/git-commit.json +10 -0
- package/resources/skills/github.json +10 -0
- package/resources/skills/gog.json +10 -0
- package/resources/skills/goplaces.json +10 -0
- package/resources/skills/himalaya.json +10 -0
- package/resources/skills/imsg.json +10 -0
- package/resources/skills/karpathy-guidelines.json +12 -0
- package/resources/skills/last30days.json +26 -0
- package/resources/skills/local-places.json +10 -0
- package/resources/skills/mcporter.json +10 -0
- package/resources/skills/model-usage.json +10 -0
- package/resources/skills/nano-banana-pro.json +10 -0
- package/resources/skills/nano-pdf.json +10 -0
- package/resources/skills/notion.json +10 -0
- package/resources/skills/obsidian.json +10 -0
- package/resources/skills/openai-image-gen.json +10 -0
- package/resources/skills/openai-whisper-api.json +10 -0
- package/resources/skills/openai-whisper.json +10 -0
- package/resources/skills/openhue.json +10 -0
- package/resources/skills/oracle.json +10 -0
- package/resources/skills/ordercli.json +10 -0
- package/resources/skills/peekaboo.json +10 -0
- package/resources/skills/project-structure.json +10 -0
- package/resources/skills/proofread.json +17 -0
- package/resources/skills/refactor-code.json +31 -0
- package/resources/skills/rename-symbol.json +23 -0
- package/resources/skills/sag.json +10 -0
- package/resources/skills/security-audit.json +18 -0
- package/resources/skills/session-logs.json +10 -0
- package/resources/skills/sherpa-onnx-tts.json +10 -0
- package/resources/skills/skill-creator.json +15 -0
- package/resources/skills/skill-hub.json +29 -0
- package/resources/skills/slack.json +10 -0
- package/resources/skills/songsee.json +10 -0
- package/resources/skills/sonoscli.json +10 -0
- package/resources/skills/spotify-player.json +10 -0
- package/resources/skills/startup-cfo.json +55 -0
- package/resources/skills/summarize-folder.json +18 -0
- package/resources/skills/summarize.json +10 -0
- package/resources/skills/things-mac.json +10 -0
- package/resources/skills/tmux.json +10 -0
- package/resources/skills/translate.json +36 -0
- package/resources/skills/trello.json +10 -0
- package/resources/skills/video-frames.json +10 -0
- package/resources/skills/voice-call.json +10 -0
- package/resources/skills/wacli.json +10 -0
- package/resources/skills/weather.json +10 -0
- package/resources/skills/write-tests.json +31 -0
- package/src/electron/activity/ActivityRepository.ts +238 -0
- package/src/electron/agent/browser/browser-service.ts +721 -0
- package/src/electron/agent/context-manager.ts +257 -0
- package/src/electron/agent/custom-skill-loader.ts +634 -0
- package/src/electron/agent/daemon.ts +1097 -0
- package/src/electron/agent/executor.ts +4017 -0
- package/src/electron/agent/llm/anthropic-provider.ts +175 -0
- package/src/electron/agent/llm/bedrock-provider.ts +236 -0
- package/src/electron/agent/llm/gemini-provider.ts +422 -0
- package/src/electron/agent/llm/index.ts +9 -0
- package/src/electron/agent/llm/ollama-provider.ts +347 -0
- package/src/electron/agent/llm/openai-oauth.ts +127 -0
- package/src/electron/agent/llm/openai-provider.ts +686 -0
- package/src/electron/agent/llm/openrouter-provider.ts +273 -0
- package/src/electron/agent/llm/pricing.ts +180 -0
- package/src/electron/agent/llm/provider-factory.ts +971 -0
- package/src/electron/agent/llm/types.ts +291 -0
- package/src/electron/agent/queue-manager.ts +408 -0
- package/src/electron/agent/sandbox/docker-sandbox.ts +453 -0
- package/src/electron/agent/sandbox/macos-sandbox.ts +426 -0
- package/src/electron/agent/sandbox/runner.ts +453 -0
- package/src/electron/agent/sandbox/sandbox-factory.ts +337 -0
- package/src/electron/agent/sandbox/security-utils.ts +251 -0
- package/src/electron/agent/search/brave-provider.ts +141 -0
- package/src/electron/agent/search/google-provider.ts +131 -0
- package/src/electron/agent/search/index.ts +6 -0
- package/src/electron/agent/search/provider-factory.ts +450 -0
- package/src/electron/agent/search/serpapi-provider.ts +138 -0
- package/src/electron/agent/search/tavily-provider.ts +108 -0
- package/src/electron/agent/search/types.ts +118 -0
- package/src/electron/agent/security/index.ts +20 -0
- package/src/electron/agent/security/input-sanitizer.ts +380 -0
- package/src/electron/agent/security/output-filter.ts +259 -0
- package/src/electron/agent/skill-eligibility.ts +334 -0
- package/src/electron/agent/skill-registry.ts +457 -0
- package/src/electron/agent/skills/document.ts +1070 -0
- package/src/electron/agent/skills/image-generator.ts +272 -0
- package/src/electron/agent/skills/organizer.ts +131 -0
- package/src/electron/agent/skills/presentation.ts +418 -0
- package/src/electron/agent/skills/spreadsheet.ts +166 -0
- package/src/electron/agent/tools/browser-tools.ts +546 -0
- package/src/electron/agent/tools/builtin-settings.ts +422 -0
- package/src/electron/agent/tools/canvas-tools.ts +572 -0
- package/src/electron/agent/tools/cron-tools.ts +723 -0
- package/src/electron/agent/tools/edit-tools.ts +196 -0
- package/src/electron/agent/tools/file-tools.ts +811 -0
- package/src/electron/agent/tools/glob-tools.ts +303 -0
- package/src/electron/agent/tools/grep-tools.ts +432 -0
- package/src/electron/agent/tools/image-tools.ts +126 -0
- package/src/electron/agent/tools/mention-tools.ts +371 -0
- package/src/electron/agent/tools/node-tools.ts +550 -0
- package/src/electron/agent/tools/registry.ts +3052 -0
- package/src/electron/agent/tools/search-tools.ts +111 -0
- package/src/electron/agent/tools/shell-tools.ts +651 -0
- package/src/electron/agent/tools/skill-tools.ts +340 -0
- package/src/electron/agent/tools/system-tools.ts +665 -0
- package/src/electron/agent/tools/web-fetch-tools.ts +528 -0
- package/src/electron/agent/tools/x-tools.ts +267 -0
- package/src/electron/agents/AgentRoleRepository.ts +557 -0
- package/src/electron/agents/HeartbeatService.ts +469 -0
- package/src/electron/agents/MentionRepository.ts +242 -0
- package/src/electron/agents/TaskSubscriptionRepository.ts +231 -0
- package/src/electron/agents/WorkingStateRepository.ts +278 -0
- package/src/electron/canvas/canvas-manager.ts +818 -0
- package/src/electron/canvas/canvas-preload.ts +102 -0
- package/src/electron/canvas/canvas-protocol.ts +174 -0
- package/src/electron/canvas/canvas-store.ts +200 -0
- package/src/electron/canvas/index.ts +8 -0
- package/src/electron/control-plane/client.ts +527 -0
- package/src/electron/control-plane/handlers.ts +723 -0
- package/src/electron/control-plane/index.ts +51 -0
- package/src/electron/control-plane/node-manager.ts +322 -0
- package/src/electron/control-plane/protocol.ts +269 -0
- package/src/electron/control-plane/remote-client.ts +517 -0
- package/src/electron/control-plane/server.ts +853 -0
- package/src/electron/control-plane/settings.ts +401 -0
- package/src/electron/control-plane/ssh-tunnel.ts +624 -0
- package/src/electron/cron/index.ts +9 -0
- package/src/electron/cron/schedule.ts +217 -0
- package/src/electron/cron/service.ts +743 -0
- package/src/electron/cron/store.ts +165 -0
- package/src/electron/cron/types.ts +291 -0
- package/src/electron/cron/webhook.ts +303 -0
- package/src/electron/database/SecureSettingsRepository.ts +514 -0
- package/src/electron/database/TaskLabelRepository.ts +148 -0
- package/src/electron/database/repositories.ts +2397 -0
- package/src/electron/database/schema.ts +1017 -0
- package/src/electron/extensions/index.ts +18 -0
- package/src/electron/extensions/loader.ts +336 -0
- package/src/electron/extensions/registry.ts +546 -0
- package/src/electron/extensions/types.ts +372 -0
- package/src/electron/gateway/channel-registry.ts +1267 -0
- package/src/electron/gateway/channels/bluebubbles-client.ts +641 -0
- package/src/electron/gateway/channels/bluebubbles.ts +509 -0
- package/src/electron/gateway/channels/discord.ts +1150 -0
- package/src/electron/gateway/channels/email-client.ts +708 -0
- package/src/electron/gateway/channels/email.ts +516 -0
- package/src/electron/gateway/channels/google-chat.ts +760 -0
- package/src/electron/gateway/channels/imessage-client.ts +473 -0
- package/src/electron/gateway/channels/imessage.ts +520 -0
- package/src/electron/gateway/channels/index.ts +21 -0
- package/src/electron/gateway/channels/line-client.ts +598 -0
- package/src/electron/gateway/channels/line.ts +559 -0
- package/src/electron/gateway/channels/matrix-client.ts +632 -0
- package/src/electron/gateway/channels/matrix.ts +655 -0
- package/src/electron/gateway/channels/mattermost-client.ts +526 -0
- package/src/electron/gateway/channels/mattermost.ts +550 -0
- package/src/electron/gateway/channels/signal-client.ts +722 -0
- package/src/electron/gateway/channels/signal.ts +666 -0
- package/src/electron/gateway/channels/slack.ts +458 -0
- package/src/electron/gateway/channels/teams.ts +681 -0
- package/src/electron/gateway/channels/telegram.ts +1727 -0
- package/src/electron/gateway/channels/twitch-client.ts +665 -0
- package/src/electron/gateway/channels/twitch.ts +468 -0
- package/src/electron/gateway/channels/types.ts +1002 -0
- package/src/electron/gateway/channels/whatsapp.ts +1101 -0
- package/src/electron/gateway/context-policy.ts +382 -0
- package/src/electron/gateway/index.ts +1274 -0
- package/src/electron/gateway/infrastructure.ts +645 -0
- package/src/electron/gateway/router.ts +3206 -0
- package/src/electron/gateway/security.ts +422 -0
- package/src/electron/gateway/session.ts +144 -0
- package/src/electron/gateway/tunnel.ts +626 -0
- package/src/electron/guardrails/guardrail-manager.ts +380 -0
- package/src/electron/hooks/gmail-watcher.ts +355 -0
- package/src/electron/hooks/index.ts +30 -0
- package/src/electron/hooks/mappings.ts +404 -0
- package/src/electron/hooks/server.ts +574 -0
- package/src/electron/hooks/settings.ts +466 -0
- package/src/electron/hooks/types.ts +245 -0
- package/src/electron/ipc/canvas-handlers.ts +223 -0
- package/src/electron/ipc/handlers.ts +3661 -0
- package/src/electron/ipc/mission-control-handlers.ts +182 -0
- package/src/electron/main.ts +496 -0
- package/src/electron/mcp/client/MCPClientManager.ts +406 -0
- package/src/electron/mcp/client/MCPServerConnection.ts +514 -0
- package/src/electron/mcp/client/transports/SSETransport.ts +360 -0
- package/src/electron/mcp/client/transports/StdioTransport.ts +355 -0
- package/src/electron/mcp/client/transports/WebSocketTransport.ts +384 -0
- package/src/electron/mcp/host/MCPHostServer.ts +388 -0
- package/src/electron/mcp/host/ToolAdapter.ts +140 -0
- package/src/electron/mcp/registry/MCPRegistryManager.ts +565 -0
- package/src/electron/mcp/settings.ts +468 -0
- package/src/electron/mcp/types.ts +371 -0
- package/src/electron/memory/MemoryService.ts +523 -0
- package/src/electron/notifications/index.ts +16 -0
- package/src/electron/notifications/service.ts +161 -0
- package/src/electron/notifications/store.ts +163 -0
- package/src/electron/preload.ts +2845 -0
- package/src/electron/reports/StandupReportService.ts +356 -0
- package/src/electron/security/concurrency.ts +333 -0
- package/src/electron/security/index.ts +17 -0
- package/src/electron/security/policy-manager.ts +539 -0
- package/src/electron/settings/appearance-manager.ts +182 -0
- package/src/electron/settings/personality-manager.ts +800 -0
- package/src/electron/settings/x-manager.ts +62 -0
- package/src/electron/tailscale/exposure.ts +262 -0
- package/src/electron/tailscale/index.ts +34 -0
- package/src/electron/tailscale/settings.ts +218 -0
- package/src/electron/tailscale/tailscale.ts +379 -0
- package/src/electron/tray/QuickInputWindow.ts +609 -0
- package/src/electron/tray/TrayManager.ts +1005 -0
- package/src/electron/tray/index.ts +6 -0
- package/src/electron/updater/index.ts +1 -0
- package/src/electron/updater/update-manager.ts +447 -0
- package/src/electron/utils/env-migration.ts +203 -0
- package/src/electron/utils/process.ts +124 -0
- package/src/electron/utils/rate-limiter.ts +130 -0
- package/src/electron/utils/validation.ts +493 -0
- package/src/electron/utils/x-cli.ts +198 -0
- package/src/electron/voice/VoiceService.ts +583 -0
- package/src/electron/voice/index.ts +9 -0
- package/src/electron/voice/voice-settings-manager.ts +403 -0
- package/src/renderer/App.tsx +775 -0
- package/src/renderer/components/ActivityFeed.tsx +407 -0
- package/src/renderer/components/ActivityFeedItem.tsx +285 -0
- package/src/renderer/components/AgentRoleCard.tsx +343 -0
- package/src/renderer/components/AgentRoleEditor.tsx +805 -0
- package/src/renderer/components/AgentSquadSettings.tsx +295 -0
- package/src/renderer/components/AgentWorkingStatePanel.tsx +411 -0
- package/src/renderer/components/AppearanceSettings.tsx +122 -0
- package/src/renderer/components/ApprovalDialog.tsx +100 -0
- package/src/renderer/components/BlueBubblesSettings.tsx +505 -0
- package/src/renderer/components/BuiltinToolsSettings.tsx +307 -0
- package/src/renderer/components/CanvasPreview.tsx +1189 -0
- package/src/renderer/components/CommandOutput.tsx +202 -0
- package/src/renderer/components/ContextPolicySettings.tsx +523 -0
- package/src/renderer/components/ControlPlaneSettings.tsx +1134 -0
- package/src/renderer/components/DisclaimerModal.tsx +124 -0
- package/src/renderer/components/DiscordSettings.tsx +436 -0
- package/src/renderer/components/EmailSettings.tsx +606 -0
- package/src/renderer/components/ExtensionsSettings.tsx +542 -0
- package/src/renderer/components/FileViewer.tsx +224 -0
- package/src/renderer/components/GoogleChatSettings.tsx +535 -0
- package/src/renderer/components/GuardrailSettings.tsx +487 -0
- package/src/renderer/components/HooksSettings.tsx +581 -0
- package/src/renderer/components/ImessageSettings.tsx +484 -0
- package/src/renderer/components/LineSettings.tsx +483 -0
- package/src/renderer/components/MCPRegistryBrowser.tsx +386 -0
- package/src/renderer/components/MCPSettings.tsx +943 -0
- package/src/renderer/components/MainContent.tsx +2433 -0
- package/src/renderer/components/MatrixSettings.tsx +510 -0
- package/src/renderer/components/MattermostSettings.tsx +473 -0
- package/src/renderer/components/MemorySettings.tsx +247 -0
- package/src/renderer/components/MentionBadge.tsx +87 -0
- package/src/renderer/components/MentionInput.tsx +409 -0
- package/src/renderer/components/MentionList.tsx +476 -0
- package/src/renderer/components/MissionControlPanel.tsx +1995 -0
- package/src/renderer/components/NodesSettings.tsx +316 -0
- package/src/renderer/components/NotificationPanel.tsx +481 -0
- package/src/renderer/components/Onboarding/AwakeningOrb.tsx +44 -0
- package/src/renderer/components/Onboarding/Onboarding.tsx +443 -0
- package/src/renderer/components/Onboarding/TypewriterText.tsx +102 -0
- package/src/renderer/components/Onboarding/index.ts +3 -0
- package/src/renderer/components/OnboardingModal.tsx +698 -0
- package/src/renderer/components/PairingCodeDisplay.tsx +324 -0
- package/src/renderer/components/PersonalitySettings.tsx +597 -0
- package/src/renderer/components/QueueSettings.tsx +119 -0
- package/src/renderer/components/QuickTaskFAB.tsx +71 -0
- package/src/renderer/components/RightPanel.tsx +413 -0
- package/src/renderer/components/ScheduledTasksSettings.tsx +1328 -0
- package/src/renderer/components/SearchSettings.tsx +328 -0
- package/src/renderer/components/Settings.tsx +1504 -0
- package/src/renderer/components/Sidebar.tsx +344 -0
- package/src/renderer/components/SignalSettings.tsx +673 -0
- package/src/renderer/components/SkillHubBrowser.tsx +458 -0
- package/src/renderer/components/SkillParameterModal.tsx +185 -0
- package/src/renderer/components/SkillsSettings.tsx +451 -0
- package/src/renderer/components/SlackSettings.tsx +442 -0
- package/src/renderer/components/StandupReportViewer.tsx +614 -0
- package/src/renderer/components/TaskBoard.tsx +498 -0
- package/src/renderer/components/TaskBoardCard.tsx +357 -0
- package/src/renderer/components/TaskBoardColumn.tsx +211 -0
- package/src/renderer/components/TaskLabelManager.tsx +472 -0
- package/src/renderer/components/TaskQueuePanel.tsx +144 -0
- package/src/renderer/components/TaskQuickActions.tsx +492 -0
- package/src/renderer/components/TaskTimeline.tsx +216 -0
- package/src/renderer/components/TaskView.tsx +162 -0
- package/src/renderer/components/TeamsSettings.tsx +518 -0
- package/src/renderer/components/TelegramSettings.tsx +421 -0
- package/src/renderer/components/Toast.tsx +76 -0
- package/src/renderer/components/TraySettings.tsx +189 -0
- package/src/renderer/components/TwitchSettings.tsx +511 -0
- package/src/renderer/components/UpdateSettings.tsx +295 -0
- package/src/renderer/components/VoiceIndicator.tsx +270 -0
- package/src/renderer/components/VoiceSettings.tsx +867 -0
- package/src/renderer/components/WhatsAppSettings.tsx +721 -0
- package/src/renderer/components/WorkingStateEditor.tsx +309 -0
- package/src/renderer/components/WorkingStateHistory.tsx +481 -0
- package/src/renderer/components/WorkspaceSelector.tsx +150 -0
- package/src/renderer/components/XSettings.tsx +311 -0
- package/src/renderer/global.d.ts +9 -0
- package/src/renderer/hooks/useAgentContext.ts +153 -0
- package/src/renderer/hooks/useOnboardingFlow.ts +548 -0
- package/src/renderer/hooks/useVoiceInput.ts +268 -0
- package/src/renderer/index.html +12 -0
- package/src/renderer/main.tsx +10 -0
- package/src/renderer/public/cowork-os-logo.png +0 -0
- package/src/renderer/quick-input.html +164 -0
- package/src/renderer/styles/index.css +14504 -0
- package/src/renderer/utils/agentMessages.ts +749 -0
- package/src/renderer/utils/voice-directives.ts +169 -0
- package/src/shared/channelMessages.ts +213 -0
- package/src/shared/types.ts +3608 -0
- package/tsconfig.electron.json +26 -0
- package/tsconfig.json +26 -0
- package/tsconfig.node.json +10 -0
- package/vite.config.ts +23 -0
|
@@ -0,0 +1,272 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.StandupReportService = void 0;
|
|
4
|
+
const uuid_1 = require("uuid");
|
|
5
|
+
/**
|
|
6
|
+
* Service for generating and managing daily standup reports
|
|
7
|
+
*/
|
|
8
|
+
class StandupReportService {
|
|
9
|
+
constructor(db, deliverToChannel) {
|
|
10
|
+
this.db = db;
|
|
11
|
+
this.deliverToChannel = deliverToChannel;
|
|
12
|
+
}
|
|
13
|
+
/**
|
|
14
|
+
* Generate a standup report for a workspace
|
|
15
|
+
* Aggregates task status from the past 24 hours
|
|
16
|
+
*/
|
|
17
|
+
async generateReport(workspaceId, date = new Date()) {
|
|
18
|
+
const reportDate = this.formatDate(date);
|
|
19
|
+
const yesterday = date.getTime() - 24 * 60 * 60 * 1000;
|
|
20
|
+
// Check if report already exists for this date
|
|
21
|
+
const existing = this.getByDate(workspaceId, reportDate);
|
|
22
|
+
if (existing) {
|
|
23
|
+
return existing;
|
|
24
|
+
}
|
|
25
|
+
// Get completed tasks (done column, updated in last 24h)
|
|
26
|
+
const completedTasks = this.getTasksByColumn(workspaceId, 'done', yesterday);
|
|
27
|
+
// Get in-progress tasks
|
|
28
|
+
const inProgressTasks = this.getTasksByColumn(workspaceId, 'in_progress');
|
|
29
|
+
// Get blocked tasks (status = 'blocked' or 'failed')
|
|
30
|
+
const blockedTasks = this.getBlockedTasks(workspaceId);
|
|
31
|
+
// Generate summary
|
|
32
|
+
const summary = this.buildSummary(completedTasks, inProgressTasks, blockedTasks);
|
|
33
|
+
const report = {
|
|
34
|
+
id: (0, uuid_1.v4)(),
|
|
35
|
+
workspaceId,
|
|
36
|
+
reportDate,
|
|
37
|
+
completedTaskIds: completedTasks.map(t => t.id),
|
|
38
|
+
inProgressTaskIds: inProgressTasks.map(t => t.id),
|
|
39
|
+
blockedTaskIds: blockedTasks.map(t => t.id),
|
|
40
|
+
summary,
|
|
41
|
+
createdAt: Date.now(),
|
|
42
|
+
};
|
|
43
|
+
// Save to database
|
|
44
|
+
this.save(report);
|
|
45
|
+
return report;
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Deliver a standup report to a configured channel
|
|
49
|
+
*/
|
|
50
|
+
async deliverReport(report, config) {
|
|
51
|
+
if (!this.deliverToChannel) {
|
|
52
|
+
throw new Error('No delivery handler configured');
|
|
53
|
+
}
|
|
54
|
+
await this.deliverToChannel(report, config);
|
|
55
|
+
// Update report with delivery info
|
|
56
|
+
const deliveredToChannel = `${config.channelType}:${config.channelId}`;
|
|
57
|
+
const stmt = this.db.prepare('UPDATE standup_reports SET delivered_to_channel = ? WHERE id = ?');
|
|
58
|
+
stmt.run(deliveredToChannel, report.id);
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Get the latest standup report for a workspace
|
|
62
|
+
*/
|
|
63
|
+
getLatest(workspaceId) {
|
|
64
|
+
const stmt = this.db.prepare(`
|
|
65
|
+
SELECT * FROM standup_reports
|
|
66
|
+
WHERE workspace_id = ?
|
|
67
|
+
ORDER BY created_at DESC
|
|
68
|
+
LIMIT 1
|
|
69
|
+
`);
|
|
70
|
+
const row = stmt.get(workspaceId);
|
|
71
|
+
return row ? this.mapRowToReport(row) : undefined;
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Get a standup report by date
|
|
75
|
+
*/
|
|
76
|
+
getByDate(workspaceId, reportDate) {
|
|
77
|
+
const stmt = this.db.prepare(`
|
|
78
|
+
SELECT * FROM standup_reports
|
|
79
|
+
WHERE workspace_id = ? AND report_date = ?
|
|
80
|
+
`);
|
|
81
|
+
const row = stmt.get(workspaceId, reportDate);
|
|
82
|
+
return row ? this.mapRowToReport(row) : undefined;
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* List standup reports for a workspace
|
|
86
|
+
*/
|
|
87
|
+
list(query) {
|
|
88
|
+
const conditions = ['workspace_id = ?'];
|
|
89
|
+
const params = [query.workspaceId];
|
|
90
|
+
if (query.startDate) {
|
|
91
|
+
conditions.push('report_date >= ?');
|
|
92
|
+
params.push(query.startDate);
|
|
93
|
+
}
|
|
94
|
+
if (query.endDate) {
|
|
95
|
+
conditions.push('report_date <= ?');
|
|
96
|
+
params.push(query.endDate);
|
|
97
|
+
}
|
|
98
|
+
let sql = `SELECT * FROM standup_reports WHERE ${conditions.join(' AND ')} ORDER BY report_date DESC`;
|
|
99
|
+
if (query.limit) {
|
|
100
|
+
sql += ` LIMIT ${query.limit}`;
|
|
101
|
+
}
|
|
102
|
+
const stmt = this.db.prepare(sql);
|
|
103
|
+
const rows = stmt.all(...params);
|
|
104
|
+
return rows.map(row => this.mapRowToReport(row));
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* Find a report by ID
|
|
108
|
+
*/
|
|
109
|
+
findById(id) {
|
|
110
|
+
const stmt = this.db.prepare('SELECT * FROM standup_reports WHERE id = ?');
|
|
111
|
+
const row = stmt.get(id);
|
|
112
|
+
return row ? this.mapRowToReport(row) : undefined;
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Delete old reports (cleanup)
|
|
116
|
+
*/
|
|
117
|
+
deleteOlderThan(workspaceId, daysToKeep) {
|
|
118
|
+
const cutoffDate = new Date();
|
|
119
|
+
cutoffDate.setDate(cutoffDate.getDate() - daysToKeep);
|
|
120
|
+
const cutoffStr = this.formatDate(cutoffDate);
|
|
121
|
+
const stmt = this.db.prepare('DELETE FROM standup_reports WHERE workspace_id = ? AND report_date < ?');
|
|
122
|
+
const result = stmt.run(workspaceId, cutoffStr);
|
|
123
|
+
return result.changes;
|
|
124
|
+
}
|
|
125
|
+
/**
|
|
126
|
+
* Format the standup report as a message for delivery
|
|
127
|
+
*/
|
|
128
|
+
formatReportMessage(report, tasks) {
|
|
129
|
+
const lines = [
|
|
130
|
+
`**Daily Standup Report - ${report.reportDate}**`,
|
|
131
|
+
'',
|
|
132
|
+
];
|
|
133
|
+
// Completed section
|
|
134
|
+
if (report.completedTaskIds.length > 0) {
|
|
135
|
+
lines.push('**Completed Today:**');
|
|
136
|
+
for (const taskId of report.completedTaskIds) {
|
|
137
|
+
const task = tasks.get(taskId);
|
|
138
|
+
if (task) {
|
|
139
|
+
lines.push(`- ${task.title}`);
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
lines.push('');
|
|
143
|
+
}
|
|
144
|
+
// In Progress section
|
|
145
|
+
if (report.inProgressTaskIds.length > 0) {
|
|
146
|
+
lines.push('**In Progress:**');
|
|
147
|
+
for (const taskId of report.inProgressTaskIds) {
|
|
148
|
+
const task = tasks.get(taskId);
|
|
149
|
+
if (task) {
|
|
150
|
+
lines.push(`- ${task.title}`);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
lines.push('');
|
|
154
|
+
}
|
|
155
|
+
// Blocked section
|
|
156
|
+
if (report.blockedTaskIds.length > 0) {
|
|
157
|
+
lines.push('**Blocked:**');
|
|
158
|
+
for (const taskId of report.blockedTaskIds) {
|
|
159
|
+
const task = tasks.get(taskId);
|
|
160
|
+
if (task) {
|
|
161
|
+
lines.push(`- ${task.title}`);
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
lines.push('');
|
|
165
|
+
}
|
|
166
|
+
// Summary
|
|
167
|
+
lines.push('**Summary:**');
|
|
168
|
+
lines.push(report.summary);
|
|
169
|
+
return lines.join('\n');
|
|
170
|
+
}
|
|
171
|
+
/**
|
|
172
|
+
* Save a report to the database
|
|
173
|
+
*/
|
|
174
|
+
save(report) {
|
|
175
|
+
const stmt = this.db.prepare(`
|
|
176
|
+
INSERT INTO standup_reports (
|
|
177
|
+
id, workspace_id, report_date, completed_task_ids,
|
|
178
|
+
in_progress_task_ids, blocked_task_ids, summary,
|
|
179
|
+
delivered_to_channel, created_at
|
|
180
|
+
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
181
|
+
`);
|
|
182
|
+
stmt.run(report.id, report.workspaceId, report.reportDate, JSON.stringify(report.completedTaskIds), JSON.stringify(report.inProgressTaskIds), JSON.stringify(report.blockedTaskIds), report.summary, report.deliveredToChannel || null, report.createdAt);
|
|
183
|
+
}
|
|
184
|
+
/**
|
|
185
|
+
* Get tasks by board column with optional time filter
|
|
186
|
+
*/
|
|
187
|
+
getTasksByColumn(workspaceId, column, updatedAfter) {
|
|
188
|
+
let sql = `
|
|
189
|
+
SELECT * FROM tasks
|
|
190
|
+
WHERE workspace_id = ? AND board_column = ?
|
|
191
|
+
`;
|
|
192
|
+
const params = [workspaceId, column];
|
|
193
|
+
if (updatedAfter) {
|
|
194
|
+
sql += ' AND updated_at >= ?';
|
|
195
|
+
params.push(updatedAfter);
|
|
196
|
+
}
|
|
197
|
+
sql += ' ORDER BY updated_at DESC';
|
|
198
|
+
const stmt = this.db.prepare(sql);
|
|
199
|
+
const rows = stmt.all(...params);
|
|
200
|
+
return rows.map(row => this.mapRowToTask(row));
|
|
201
|
+
}
|
|
202
|
+
/**
|
|
203
|
+
* Get blocked tasks (status = blocked or failed)
|
|
204
|
+
*/
|
|
205
|
+
getBlockedTasks(workspaceId) {
|
|
206
|
+
const stmt = this.db.prepare(`
|
|
207
|
+
SELECT * FROM tasks
|
|
208
|
+
WHERE workspace_id = ? AND status IN ('blocked', 'failed')
|
|
209
|
+
ORDER BY updated_at DESC
|
|
210
|
+
`);
|
|
211
|
+
const rows = stmt.all(workspaceId);
|
|
212
|
+
return rows.map(row => this.mapRowToTask(row));
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* Build a summary from task lists
|
|
216
|
+
*/
|
|
217
|
+
buildSummary(completed, inProgress, blocked) {
|
|
218
|
+
const parts = [];
|
|
219
|
+
if (completed.length > 0) {
|
|
220
|
+
parts.push(`${completed.length} task${completed.length === 1 ? '' : 's'} completed`);
|
|
221
|
+
}
|
|
222
|
+
if (inProgress.length > 0) {
|
|
223
|
+
parts.push(`${inProgress.length} task${inProgress.length === 1 ? '' : 's'} in progress`);
|
|
224
|
+
}
|
|
225
|
+
if (blocked.length > 0) {
|
|
226
|
+
parts.push(`${blocked.length} task${blocked.length === 1 ? '' : 's'} blocked`);
|
|
227
|
+
}
|
|
228
|
+
if (parts.length === 0) {
|
|
229
|
+
return 'No task activity today.';
|
|
230
|
+
}
|
|
231
|
+
return parts.join(', ') + '.';
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Format date as YYYY-MM-DD
|
|
235
|
+
*/
|
|
236
|
+
formatDate(date) {
|
|
237
|
+
return date.toISOString().split('T')[0];
|
|
238
|
+
}
|
|
239
|
+
/**
|
|
240
|
+
* Map database row to StandupReport
|
|
241
|
+
*/
|
|
242
|
+
mapRowToReport(row) {
|
|
243
|
+
return {
|
|
244
|
+
id: row.id,
|
|
245
|
+
workspaceId: row.workspace_id,
|
|
246
|
+
reportDate: row.report_date,
|
|
247
|
+
completedTaskIds: JSON.parse(row.completed_task_ids || '[]'),
|
|
248
|
+
inProgressTaskIds: JSON.parse(row.in_progress_task_ids || '[]'),
|
|
249
|
+
blockedTaskIds: JSON.parse(row.blocked_task_ids || '[]'),
|
|
250
|
+
summary: row.summary,
|
|
251
|
+
deliveredToChannel: row.delivered_to_channel || undefined,
|
|
252
|
+
createdAt: row.created_at,
|
|
253
|
+
};
|
|
254
|
+
}
|
|
255
|
+
/**
|
|
256
|
+
* Map database row to Task (minimal mapping for standup)
|
|
257
|
+
*/
|
|
258
|
+
mapRowToTask(row) {
|
|
259
|
+
return {
|
|
260
|
+
id: row.id,
|
|
261
|
+
title: row.title,
|
|
262
|
+
prompt: row.prompt,
|
|
263
|
+
status: row.status,
|
|
264
|
+
workspaceId: row.workspace_id,
|
|
265
|
+
createdAt: row.created_at,
|
|
266
|
+
updatedAt: row.updated_at,
|
|
267
|
+
boardColumn: row.board_column,
|
|
268
|
+
assignedAgentRoleId: row.assigned_agent_role_id,
|
|
269
|
+
};
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
exports.StandupReportService = StandupReportService;
|
|
@@ -0,0 +1,293 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Concurrency Safety Module
|
|
4
|
+
*
|
|
5
|
+
* Provides mutex locks and idempotency guarantees for critical operations.
|
|
6
|
+
*/
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
exports.taskIdempotency = exports.approvalIdempotency = exports.pairingMutex = exports.IdempotencyManager = exports.NamedMutexManager = exports.AsyncMutex = void 0;
|
|
9
|
+
/**
|
|
10
|
+
* Simple async mutex for protecting critical sections
|
|
11
|
+
* Prevents race conditions in pairing, approval, and state changes
|
|
12
|
+
*/
|
|
13
|
+
class AsyncMutex {
|
|
14
|
+
constructor() {
|
|
15
|
+
this.locked = false;
|
|
16
|
+
this.queue = [];
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Acquire the lock. Returns a release function.
|
|
20
|
+
*/
|
|
21
|
+
async acquire() {
|
|
22
|
+
return new Promise((resolve) => {
|
|
23
|
+
const tryAcquire = () => {
|
|
24
|
+
if (!this.locked) {
|
|
25
|
+
this.locked = true;
|
|
26
|
+
resolve(() => this.release());
|
|
27
|
+
}
|
|
28
|
+
else {
|
|
29
|
+
this.queue.push(tryAcquire);
|
|
30
|
+
}
|
|
31
|
+
};
|
|
32
|
+
tryAcquire();
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Release the lock
|
|
37
|
+
*/
|
|
38
|
+
release() {
|
|
39
|
+
this.locked = false;
|
|
40
|
+
const next = this.queue.shift();
|
|
41
|
+
if (next) {
|
|
42
|
+
next();
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Execute a function with the lock held
|
|
47
|
+
*/
|
|
48
|
+
async withLock(fn) {
|
|
49
|
+
const release = await this.acquire();
|
|
50
|
+
try {
|
|
51
|
+
return await fn();
|
|
52
|
+
}
|
|
53
|
+
finally {
|
|
54
|
+
release();
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Check if the lock is currently held
|
|
59
|
+
*/
|
|
60
|
+
isLocked() {
|
|
61
|
+
return this.locked;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
exports.AsyncMutex = AsyncMutex;
|
|
65
|
+
/**
|
|
66
|
+
* Named mutex manager for managing multiple locks by key
|
|
67
|
+
*/
|
|
68
|
+
class NamedMutexManager {
|
|
69
|
+
constructor() {
|
|
70
|
+
this.mutexes = new Map();
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Get or create a mutex for a given key
|
|
74
|
+
*/
|
|
75
|
+
getMutex(key) {
|
|
76
|
+
let mutex = this.mutexes.get(key);
|
|
77
|
+
if (!mutex) {
|
|
78
|
+
mutex = new AsyncMutex();
|
|
79
|
+
this.mutexes.set(key, mutex);
|
|
80
|
+
}
|
|
81
|
+
return mutex;
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Execute a function with the named lock held
|
|
85
|
+
*/
|
|
86
|
+
async withLock(key, fn) {
|
|
87
|
+
return this.getMutex(key).withLock(fn);
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Clean up unused mutexes
|
|
91
|
+
*/
|
|
92
|
+
cleanup() {
|
|
93
|
+
for (const [key, mutex] of this.mutexes.entries()) {
|
|
94
|
+
if (!mutex.isLocked()) {
|
|
95
|
+
this.mutexes.delete(key);
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
exports.NamedMutexManager = NamedMutexManager;
|
|
101
|
+
/**
|
|
102
|
+
* Idempotency Manager
|
|
103
|
+
*
|
|
104
|
+
* Ensures operations are idempotent by tracking operation keys and their results.
|
|
105
|
+
* Implements C6-style approval safety: prevents duplicate approvals/denials.
|
|
106
|
+
*/
|
|
107
|
+
class IdempotencyManager {
|
|
108
|
+
constructor(defaultTTLMs = 5 * 60 * 1000) {
|
|
109
|
+
this.entries = new Map();
|
|
110
|
+
this.defaultTTLMs = defaultTTLMs;
|
|
111
|
+
// Periodic cleanup of expired entries
|
|
112
|
+
this.cleanupIntervalId = setInterval(() => {
|
|
113
|
+
this.cleanupExpired();
|
|
114
|
+
}, 60 * 1000); // Every minute
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* Generate an idempotency key for an operation
|
|
118
|
+
*/
|
|
119
|
+
static generateKey(operation, ...args) {
|
|
120
|
+
const parts = [operation, ...args.filter(a => a !== undefined)];
|
|
121
|
+
return parts.join(':');
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Check if an operation with this key is already in progress or completed
|
|
125
|
+
*/
|
|
126
|
+
check(key) {
|
|
127
|
+
const entry = this.entries.get(key);
|
|
128
|
+
if (!entry) {
|
|
129
|
+
return { exists: false };
|
|
130
|
+
}
|
|
131
|
+
// Check if expired
|
|
132
|
+
if (Date.now() > entry.expiresAt) {
|
|
133
|
+
this.entries.delete(key);
|
|
134
|
+
return { exists: false };
|
|
135
|
+
}
|
|
136
|
+
return {
|
|
137
|
+
exists: true,
|
|
138
|
+
status: entry.status,
|
|
139
|
+
result: entry.result,
|
|
140
|
+
};
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* Start tracking an operation (mark as pending)
|
|
144
|
+
* Returns false if operation is already in progress
|
|
145
|
+
*/
|
|
146
|
+
start(key, ttlMs) {
|
|
147
|
+
const existing = this.check(key);
|
|
148
|
+
if (existing.exists) {
|
|
149
|
+
// Already in progress or completed
|
|
150
|
+
return false;
|
|
151
|
+
}
|
|
152
|
+
const now = Date.now();
|
|
153
|
+
this.entries.set(key, {
|
|
154
|
+
key,
|
|
155
|
+
result: undefined,
|
|
156
|
+
createdAt: now,
|
|
157
|
+
expiresAt: now + (ttlMs || this.defaultTTLMs),
|
|
158
|
+
status: 'pending',
|
|
159
|
+
});
|
|
160
|
+
return true;
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Mark an operation as completed with its result
|
|
164
|
+
*/
|
|
165
|
+
complete(key, result) {
|
|
166
|
+
const entry = this.entries.get(key);
|
|
167
|
+
if (entry) {
|
|
168
|
+
entry.status = 'completed';
|
|
169
|
+
entry.result = result;
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* Mark an operation as failed
|
|
174
|
+
*/
|
|
175
|
+
fail(key, error) {
|
|
176
|
+
const entry = this.entries.get(key);
|
|
177
|
+
if (entry) {
|
|
178
|
+
entry.status = 'failed';
|
|
179
|
+
entry.result = error;
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
/**
|
|
183
|
+
* Remove an entry (for retry scenarios)
|
|
184
|
+
*/
|
|
185
|
+
remove(key) {
|
|
186
|
+
this.entries.delete(key);
|
|
187
|
+
}
|
|
188
|
+
/**
|
|
189
|
+
* Execute an operation with idempotency guarantee
|
|
190
|
+
* If operation was already completed, returns cached result
|
|
191
|
+
* If operation is in progress, waits and returns result
|
|
192
|
+
*/
|
|
193
|
+
async execute(key, operation, ttlMs) {
|
|
194
|
+
const existing = this.check(key);
|
|
195
|
+
if (existing.exists && existing.status === 'completed') {
|
|
196
|
+
return { result: existing.result, cached: true };
|
|
197
|
+
}
|
|
198
|
+
if (existing.exists && existing.status === 'pending') {
|
|
199
|
+
// Wait for completion
|
|
200
|
+
const result = await this.waitForCompletion(key);
|
|
201
|
+
return { result, cached: true };
|
|
202
|
+
}
|
|
203
|
+
// Start new operation
|
|
204
|
+
if (!this.start(key, ttlMs)) {
|
|
205
|
+
// Race condition: another call started between check and start
|
|
206
|
+
const result = await this.waitForCompletion(key);
|
|
207
|
+
return { result, cached: true };
|
|
208
|
+
}
|
|
209
|
+
try {
|
|
210
|
+
const result = await operation();
|
|
211
|
+
this.complete(key, result);
|
|
212
|
+
return { result, cached: false };
|
|
213
|
+
}
|
|
214
|
+
catch (error) {
|
|
215
|
+
this.fail(key, error);
|
|
216
|
+
throw error;
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
/**
|
|
220
|
+
* Wait for a pending operation to complete
|
|
221
|
+
*/
|
|
222
|
+
async waitForCompletion(key, timeoutMs = 30000) {
|
|
223
|
+
const startTime = Date.now();
|
|
224
|
+
while (Date.now() - startTime < timeoutMs) {
|
|
225
|
+
const entry = this.entries.get(key);
|
|
226
|
+
if (!entry || Date.now() > entry.expiresAt) {
|
|
227
|
+
throw new Error(`Operation ${key} expired or not found`);
|
|
228
|
+
}
|
|
229
|
+
if (entry.status === 'completed') {
|
|
230
|
+
return entry.result;
|
|
231
|
+
}
|
|
232
|
+
if (entry.status === 'failed') {
|
|
233
|
+
throw entry.result || new Error(`Operation ${key} failed`);
|
|
234
|
+
}
|
|
235
|
+
// Wait a bit before checking again
|
|
236
|
+
await new Promise(resolve => setTimeout(resolve, 100));
|
|
237
|
+
}
|
|
238
|
+
throw new Error(`Timeout waiting for operation ${key}`);
|
|
239
|
+
}
|
|
240
|
+
/**
|
|
241
|
+
* Clean up expired entries
|
|
242
|
+
*/
|
|
243
|
+
cleanupExpired() {
|
|
244
|
+
const now = Date.now();
|
|
245
|
+
for (const [key, entry] of this.entries.entries()) {
|
|
246
|
+
if (now > entry.expiresAt) {
|
|
247
|
+
this.entries.delete(key);
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
/**
|
|
252
|
+
* Get statistics about the idempotency cache
|
|
253
|
+
*/
|
|
254
|
+
getStats() {
|
|
255
|
+
let pending = 0;
|
|
256
|
+
let completed = 0;
|
|
257
|
+
let failed = 0;
|
|
258
|
+
for (const entry of this.entries.values()) {
|
|
259
|
+
switch (entry.status) {
|
|
260
|
+
case 'pending':
|
|
261
|
+
pending++;
|
|
262
|
+
break;
|
|
263
|
+
case 'completed':
|
|
264
|
+
completed++;
|
|
265
|
+
break;
|
|
266
|
+
case 'failed':
|
|
267
|
+
failed++;
|
|
268
|
+
break;
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
return {
|
|
272
|
+
total: this.entries.size,
|
|
273
|
+
pending,
|
|
274
|
+
completed,
|
|
275
|
+
failed,
|
|
276
|
+
};
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* Clean up and stop background tasks
|
|
280
|
+
*/
|
|
281
|
+
destroy() {
|
|
282
|
+
if (this.cleanupIntervalId) {
|
|
283
|
+
clearInterval(this.cleanupIntervalId);
|
|
284
|
+
this.cleanupIntervalId = undefined;
|
|
285
|
+
}
|
|
286
|
+
this.entries.clear();
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
exports.IdempotencyManager = IdempotencyManager;
|
|
290
|
+
// Global instances for common use cases
|
|
291
|
+
exports.pairingMutex = new NamedMutexManager();
|
|
292
|
+
exports.approvalIdempotency = new IdempotencyManager(5 * 60 * 1000); // 5 min TTL
|
|
293
|
+
exports.taskIdempotency = new IdempotencyManager(60 * 1000); // 1 min TTL for task creation
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Security Module
|
|
4
|
+
*
|
|
5
|
+
* Exports all security-related components for the application.
|
|
6
|
+
*/
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
exports.IdempotencyManager = exports.AsyncMutex = exports.isToolAllowedQuick = exports.createPolicyManager = exports.SecurityPolicyManager = void 0;
|
|
9
|
+
var policy_manager_1 = require("./policy-manager");
|
|
10
|
+
Object.defineProperty(exports, "SecurityPolicyManager", { enumerable: true, get: function () { return policy_manager_1.SecurityPolicyManager; } });
|
|
11
|
+
Object.defineProperty(exports, "createPolicyManager", { enumerable: true, get: function () { return policy_manager_1.createPolicyManager; } });
|
|
12
|
+
Object.defineProperty(exports, "isToolAllowedQuick", { enumerable: true, get: function () { return policy_manager_1.isToolAllowedQuick; } });
|
|
13
|
+
var concurrency_1 = require("./concurrency");
|
|
14
|
+
Object.defineProperty(exports, "AsyncMutex", { enumerable: true, get: function () { return concurrency_1.AsyncMutex; } });
|
|
15
|
+
Object.defineProperty(exports, "IdempotencyManager", { enumerable: true, get: function () { return concurrency_1.IdempotencyManager; } });
|