corporateai 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (704) hide show
  1. package/.dockerignore +10 -0
  2. package/.env.example +3 -0
  3. package/.github/workflows/publish-cli.yml +49 -0
  4. package/.mailmap +1 -0
  5. package/AGENTS.md +148 -0
  6. package/CONTRIBUTING.md +75 -0
  7. package/Dockerfile +59 -0
  8. package/Dockerfile.onboard-smoke +42 -0
  9. package/LICENSE +21 -0
  10. package/README.md +93 -0
  11. package/cli/esbuild.config.mjs +11 -0
  12. package/cli/package.json +24 -0
  13. package/cli/scripts/build-cli.mjs +5 -0
  14. package/cli/src/index.ts +27 -0
  15. package/docker-compose.quickstart.yml +18 -0
  16. package/docker-compose.untrusted-review.yml +33 -0
  17. package/docker-compose.yml +38 -0
  18. package/package.json +56 -0
  19. package/patches/embedded-postgres@18.1.0-beta.16.patch +0 -0
  20. package/pnpm-workspace.yaml +4 -0
  21. package/releases/.gitkeep +0 -0
  22. package/releases/v0.0.1.md +36 -0
  23. package/releases/v0.2.7.md +15 -0
  24. package/releases/v0.3.0.md +54 -0
  25. package/releases/v0.3.1.md +55 -0
  26. package/releases/v2026.318.0.md +66 -0
  27. package/releases/v2026.325.0.md +78 -0
  28. package/report/2026-03-13-08-46-token-optimization-implementation.md +48 -0
  29. package/scripts/backup-db.sh +17 -0
  30. package/scripts/build-npm.sh +80 -0
  31. package/scripts/check-forbidden-tokens.mjs +115 -0
  32. package/scripts/clean-onboard-git.sh +14 -0
  33. package/scripts/clean-onboard-npm.sh +13 -0
  34. package/scripts/clean-onboard-ref.sh +86 -0
  35. package/scripts/create-github-release.sh +99 -0
  36. package/scripts/dev-runner-paths.mjs +38 -0
  37. package/scripts/dev-runner.mjs +606 -0
  38. package/scripts/docker-onboard-smoke.sh +306 -0
  39. package/scripts/ensure-plugin-build-deps.mjs +47 -0
  40. package/scripts/generate-company-assets.ts +365 -0
  41. package/scripts/generate-npm-package-json.mjs +113 -0
  42. package/scripts/generate-org-chart-images.ts +694 -0
  43. package/scripts/generate-org-chart-satori-comparison.ts +225 -0
  44. package/scripts/generate-ui-package-json.mjs +31 -0
  45. package/scripts/kill-dev.sh +71 -0
  46. package/scripts/migrate-inline-env-secrets.ts +126 -0
  47. package/scripts/prepare-server-ui-dist.sh +22 -0
  48. package/scripts/provision-worktree.sh +333 -0
  49. package/scripts/release-lib.sh +306 -0
  50. package/scripts/release-package-map.mjs +169 -0
  51. package/scripts/release.sh +312 -0
  52. package/scripts/rollback-latest.sh +111 -0
  53. package/scripts/smoke/openclaw-docker-ui.sh +329 -0
  54. package/scripts/smoke/openclaw-gateway-e2e.sh +954 -0
  55. package/scripts/smoke/openclaw-join.sh +295 -0
  56. package/scripts/smoke/openclaw-sse-standalone.sh +146 -0
  57. package/scripts/workspace-compat.mjs +60 -0
  58. package/server/CHANGELOG.md +130 -0
  59. package/server/package.json +96 -0
  60. package/server/scripts/copy-onboarding-assets.mjs +10 -0
  61. package/server/scripts/dev-watch.ts +33 -0
  62. package/server/src/__tests__/activity-routes.test.ts +70 -0
  63. package/server/src/__tests__/adapter-models.test.ts +105 -0
  64. package/server/src/__tests__/adapter-session-codecs.test.ts +194 -0
  65. package/server/src/__tests__/agent-auth-jwt.test.ts +79 -0
  66. package/server/src/__tests__/agent-instructions-routes.test.ts +318 -0
  67. package/server/src/__tests__/agent-instructions-service.test.ts +361 -0
  68. package/server/src/__tests__/agent-permissions-routes.test.ts +275 -0
  69. package/server/src/__tests__/agent-shortname-collision.test.ts +69 -0
  70. package/server/src/__tests__/agent-skill-contract.test.ts +50 -0
  71. package/server/src/__tests__/agent-skills-routes.test.ts +462 -0
  72. package/server/src/__tests__/app-hmr-port.test.ts +19 -0
  73. package/server/src/__tests__/approval-routes-idempotency.test.ts +110 -0
  74. package/server/src/__tests__/approvals-service.test.ts +107 -0
  75. package/server/src/__tests__/assets.test.ts +250 -0
  76. package/server/src/__tests__/attachment-types.test.ts +97 -0
  77. package/server/src/__tests__/board-mutation-guard.test.ts +105 -0
  78. package/server/src/__tests__/budgets-service.test.ts +311 -0
  79. package/server/src/__tests__/claude-local-adapter-environment.test.ts +92 -0
  80. package/server/src/__tests__/claude-local-adapter.test.ts +31 -0
  81. package/server/src/__tests__/claude-local-skill-sync.test.ts +111 -0
  82. package/server/src/__tests__/cli-auth-routes.test.ts +230 -0
  83. package/server/src/__tests__/codex-local-adapter-environment.test.ts +143 -0
  84. package/server/src/__tests__/codex-local-adapter.test.ts +253 -0
  85. package/server/src/__tests__/codex-local-execute.test.ts +391 -0
  86. package/server/src/__tests__/codex-local-skill-injection.test.ts +175 -0
  87. package/server/src/__tests__/codex-local-skill-sync.test.ts +123 -0
  88. package/server/src/__tests__/companies-route-path-guard.test.ts +56 -0
  89. package/server/src/__tests__/company-branding-route.test.ts +196 -0
  90. package/server/src/__tests__/company-portability-routes.test.ts +175 -0
  91. package/server/src/__tests__/company-portability.test.ts +2186 -0
  92. package/server/src/__tests__/company-skills-routes.test.ts +113 -0
  93. package/server/src/__tests__/company-skills.test.ts +229 -0
  94. package/server/src/__tests__/costs-service.test.ts +226 -0
  95. package/server/src/__tests__/cursor-local-adapter-environment.test.ts +196 -0
  96. package/server/src/__tests__/cursor-local-adapter.test.ts +406 -0
  97. package/server/src/__tests__/cursor-local-execute.test.ts +263 -0
  98. package/server/src/__tests__/cursor-local-skill-injection.test.ts +104 -0
  99. package/server/src/__tests__/cursor-local-skill-sync.test.ts +145 -0
  100. package/server/src/__tests__/dev-runner-paths.test.ts +25 -0
  101. package/server/src/__tests__/dev-server-status.test.ts +66 -0
  102. package/server/src/__tests__/dev-watch-ignore.test.ts +42 -0
  103. package/server/src/__tests__/documents.test.ts +29 -0
  104. package/server/src/__tests__/error-handler.test.ts +53 -0
  105. package/server/src/__tests__/execution-workspace-policy.test.ts +170 -0
  106. package/server/src/__tests__/forbidden-tokens.test.ts +77 -0
  107. package/server/src/__tests__/gemini-local-adapter-environment.test.ts +135 -0
  108. package/server/src/__tests__/gemini-local-adapter.test.ts +190 -0
  109. package/server/src/__tests__/gemini-local-execute.test.ts +172 -0
  110. package/server/src/__tests__/gemini-local-skill-sync.test.ts +90 -0
  111. package/server/src/__tests__/health.test.ts +16 -0
  112. package/server/src/__tests__/heartbeat-process-recovery.test.ts +256 -0
  113. package/server/src/__tests__/heartbeat-run-summary.test.ts +33 -0
  114. package/server/src/__tests__/heartbeat-workspace-session.test.ts +334 -0
  115. package/server/src/__tests__/helpers/embedded-postgres.ts +7 -0
  116. package/server/src/__tests__/hire-hook.test.ts +181 -0
  117. package/server/src/__tests__/instance-settings-routes.test.ts +156 -0
  118. package/server/src/__tests__/invite-accept-gateway-defaults.test.ts +119 -0
  119. package/server/src/__tests__/invite-accept-replay.test.ts +92 -0
  120. package/server/src/__tests__/invite-expiry.test.ts +10 -0
  121. package/server/src/__tests__/invite-join-grants.test.ts +57 -0
  122. package/server/src/__tests__/invite-join-manager.test.ts +33 -0
  123. package/server/src/__tests__/invite-onboarding-text.test.ts +116 -0
  124. package/server/src/__tests__/issue-comment-reopen-routes.test.ts +146 -0
  125. package/server/src/__tests__/issue-goal-fallback.test.ts +99 -0
  126. package/server/src/__tests__/issues-checkout-wakeup.test.ts +48 -0
  127. package/server/src/__tests__/issues-goal-context-routes.test.ts +187 -0
  128. package/server/src/__tests__/issues-service.test.ts +317 -0
  129. package/server/src/__tests__/issues-user-context.test.ts +113 -0
  130. package/server/src/__tests__/log-redaction.test.ts +74 -0
  131. package/server/src/__tests__/monthly-spend-service.test.ts +90 -0
  132. package/server/src/__tests__/normalize-agent-mention-token.test.ts +41 -0
  133. package/server/src/__tests__/openclaw-gateway-adapter.test.ts +626 -0
  134. package/server/src/__tests__/openclaw-invite-prompt-route.test.ts +192 -0
  135. package/server/src/__tests__/opencode-local-adapter-environment.test.ts +97 -0
  136. package/server/src/__tests__/opencode-local-adapter.test.ts +226 -0
  137. package/server/src/__tests__/opencode-local-skill-sync.test.ts +91 -0
  138. package/server/src/__tests__/paperclip-env.test.ts +58 -0
  139. package/server/src/__tests__/paperclip-skill-utils.test.ts +63 -0
  140. package/server/src/__tests__/pi-local-adapter-environment.test.ts +102 -0
  141. package/server/src/__tests__/pi-local-skill-sync.test.ts +95 -0
  142. package/server/src/__tests__/plugin-dev-watcher.test.ts +68 -0
  143. package/server/src/__tests__/plugin-worker-manager.test.ts +43 -0
  144. package/server/src/__tests__/private-hostname-guard.test.ts +56 -0
  145. package/server/src/__tests__/project-shortname-resolution.test.ts +45 -0
  146. package/server/src/__tests__/quota-windows-service.test.ts +56 -0
  147. package/server/src/__tests__/quota-windows.test.ts +1109 -0
  148. package/server/src/__tests__/redaction.test.ts +66 -0
  149. package/server/src/__tests__/routines-e2e.test.ts +276 -0
  150. package/server/src/__tests__/routines-routes.test.ts +271 -0
  151. package/server/src/__tests__/routines-service.test.ts +424 -0
  152. package/server/src/__tests__/storage-local-provider.test.ts +78 -0
  153. package/server/src/__tests__/ui-branding.test.ts +82 -0
  154. package/server/src/__tests__/work-products.test.ts +95 -0
  155. package/server/src/__tests__/workspace-runtime.test.ts +1131 -0
  156. package/server/src/__tests__/worktree-config.test.ts +426 -0
  157. package/server/src/adapters/codex-models.ts +105 -0
  158. package/server/src/adapters/cursor-models.ts +171 -0
  159. package/server/src/adapters/http/execute.ts +42 -0
  160. package/server/src/adapters/http/index.ts +21 -0
  161. package/server/src/adapters/http/test.ts +116 -0
  162. package/server/src/adapters/index.ts +18 -0
  163. package/server/src/adapters/process/execute.ts +77 -0
  164. package/server/src/adapters/process/index.ts +24 -0
  165. package/server/src/adapters/process/test.ts +89 -0
  166. package/server/src/adapters/registry.ts +225 -0
  167. package/server/src/adapters/server-utils-compat.ts +57 -0
  168. package/server/src/adapters/types.ts +30 -0
  169. package/server/src/adapters/utils.ts +48 -0
  170. package/server/src/agent-auth-jwt.ts +141 -0
  171. package/server/src/app.ts +321 -0
  172. package/server/src/attachment-types.ts +74 -0
  173. package/server/src/auth/better-auth.ts +148 -0
  174. package/server/src/board-claim.ts +150 -0
  175. package/server/src/config-file.ts +17 -0
  176. package/server/src/config.ts +260 -0
  177. package/server/src/dev-server-status.ts +103 -0
  178. package/server/src/dev-watch-ignore.ts +36 -0
  179. package/server/src/errors.ts +34 -0
  180. package/server/src/home-paths.ts +95 -0
  181. package/server/src/index.ts +799 -0
  182. package/server/src/log-redaction.ts +146 -0
  183. package/server/src/middleware/auth.ts +178 -0
  184. package/server/src/middleware/board-mutation-guard.ts +66 -0
  185. package/server/src/middleware/error-handler.ts +71 -0
  186. package/server/src/middleware/index.ts +3 -0
  187. package/server/src/middleware/logger.ts +90 -0
  188. package/server/src/middleware/private-hostname-guard.ts +92 -0
  189. package/server/src/middleware/validate.ts +9 -0
  190. package/server/src/onboarding-assets/ceo/AGENTS.md +54 -0
  191. package/server/src/onboarding-assets/ceo/HEARTBEAT.md +72 -0
  192. package/server/src/onboarding-assets/ceo/SOUL.md +33 -0
  193. package/server/src/onboarding-assets/ceo/TOOLS.md +3 -0
  194. package/server/src/onboarding-assets/default/AGENTS.md +3 -0
  195. package/server/src/paths.ts +34 -0
  196. package/server/src/realtime/live-events-ws.ts +274 -0
  197. package/server/src/redaction.ts +59 -0
  198. package/server/src/routes/access.ts +2888 -0
  199. package/server/src/routes/activity.ts +89 -0
  200. package/server/src/routes/agents.ts +2313 -0
  201. package/server/src/routes/approvals.ts +346 -0
  202. package/server/src/routes/assets.ts +341 -0
  203. package/server/src/routes/authz.ts +52 -0
  204. package/server/src/routes/companies.ts +343 -0
  205. package/server/src/routes/company-skills.ts +300 -0
  206. package/server/src/routes/costs.ts +335 -0
  207. package/server/src/routes/dashboard.ts +19 -0
  208. package/server/src/routes/execution-workspaces.ts +182 -0
  209. package/server/src/routes/goals.ts +107 -0
  210. package/server/src/routes/health.ts +94 -0
  211. package/server/src/routes/index.ts +17 -0
  212. package/server/src/routes/instance-settings.ts +95 -0
  213. package/server/src/routes/issues-checkout-wakeup.ts +14 -0
  214. package/server/src/routes/issues.ts +1680 -0
  215. package/server/src/routes/llms.ts +86 -0
  216. package/server/src/routes/org-chart-svg.ts +777 -0
  217. package/server/src/routes/plugin-ui-static.ts +497 -0
  218. package/server/src/routes/plugins.ts +2220 -0
  219. package/server/src/routes/projects.ts +295 -0
  220. package/server/src/routes/routines.ts +300 -0
  221. package/server/src/routes/secrets.ts +166 -0
  222. package/server/src/routes/sidebar-badges.ts +52 -0
  223. package/server/src/secrets/external-stub-providers.ts +32 -0
  224. package/server/src/secrets/local-encrypted-provider.ts +135 -0
  225. package/server/src/secrets/provider-registry.ts +31 -0
  226. package/server/src/secrets/types.ts +23 -0
  227. package/server/src/services/access.ts +381 -0
  228. package/server/src/services/activity-log.ts +95 -0
  229. package/server/src/services/activity.ts +164 -0
  230. package/server/src/services/agent-instructions.ts +735 -0
  231. package/server/src/services/agent-permissions.ts +27 -0
  232. package/server/src/services/agents.ts +694 -0
  233. package/server/src/services/approvals.ts +273 -0
  234. package/server/src/services/assets.ts +23 -0
  235. package/server/src/services/board-auth.ts +355 -0
  236. package/server/src/services/budgets.ts +959 -0
  237. package/server/src/services/companies.ts +313 -0
  238. package/server/src/services/company-export-readme.ts +173 -0
  239. package/server/src/services/company-portability.ts +4263 -0
  240. package/server/src/services/company-skills.ts +2356 -0
  241. package/server/src/services/costs.ts +365 -0
  242. package/server/src/services/cron.ts +373 -0
  243. package/server/src/services/dashboard.ts +110 -0
  244. package/server/src/services/default-agent-instructions.ts +27 -0
  245. package/server/src/services/documents.ts +434 -0
  246. package/server/src/services/execution-workspace-policy.ts +210 -0
  247. package/server/src/services/execution-workspaces.ts +100 -0
  248. package/server/src/services/finance.ts +135 -0
  249. package/server/src/services/goals.ts +81 -0
  250. package/server/src/services/heartbeat-run-summary.ts +35 -0
  251. package/server/src/services/heartbeat.ts +3863 -0
  252. package/server/src/services/hire-hook.ts +114 -0
  253. package/server/src/services/index.ts +32 -0
  254. package/server/src/services/instance-settings.ts +138 -0
  255. package/server/src/services/issue-approvals.ts +175 -0
  256. package/server/src/services/issue-assignment-wakeup.ts +48 -0
  257. package/server/src/services/issue-goal-fallback.ts +56 -0
  258. package/server/src/services/issues.ts +1828 -0
  259. package/server/src/services/live-events.ts +55 -0
  260. package/server/src/services/plugin-capability-validator.ts +450 -0
  261. package/server/src/services/plugin-config-validator.ts +55 -0
  262. package/server/src/services/plugin-dev-watcher.ts +339 -0
  263. package/server/src/services/plugin-event-bus.ts +413 -0
  264. package/server/src/services/plugin-host-service-cleanup.ts +59 -0
  265. package/server/src/services/plugin-host-services.ts +1132 -0
  266. package/server/src/services/plugin-job-coordinator.ts +261 -0
  267. package/server/src/services/plugin-job-scheduler.ts +753 -0
  268. package/server/src/services/plugin-job-store.ts +466 -0
  269. package/server/src/services/plugin-lifecycle.ts +822 -0
  270. package/server/src/services/plugin-loader.ts +1955 -0
  271. package/server/src/services/plugin-log-retention.ts +87 -0
  272. package/server/src/services/plugin-manifest-validator.ts +164 -0
  273. package/server/src/services/plugin-registry.ts +683 -0
  274. package/server/src/services/plugin-runtime-sandbox.ts +222 -0
  275. package/server/src/services/plugin-secrets-handler.ts +355 -0
  276. package/server/src/services/plugin-state-store.ts +238 -0
  277. package/server/src/services/plugin-stream-bus.ts +81 -0
  278. package/server/src/services/plugin-tool-dispatcher.ts +449 -0
  279. package/server/src/services/plugin-tool-registry.ts +450 -0
  280. package/server/src/services/plugin-worker-manager.ts +1343 -0
  281. package/server/src/services/projects.ts +860 -0
  282. package/server/src/services/quota-windows.ts +65 -0
  283. package/server/src/services/routines.ts +1269 -0
  284. package/server/src/services/run-log-store.ts +156 -0
  285. package/server/src/services/secrets.ts +370 -0
  286. package/server/src/services/sidebar-badges.ts +56 -0
  287. package/server/src/services/work-products.ts +124 -0
  288. package/server/src/services/workspace-operation-log-store.ts +156 -0
  289. package/server/src/services/workspace-operations.ts +262 -0
  290. package/server/src/services/workspace-runtime.ts +1565 -0
  291. package/server/src/startup-banner.ts +176 -0
  292. package/server/src/storage/index.ts +35 -0
  293. package/server/src/storage/local-disk-provider.ts +89 -0
  294. package/server/src/storage/provider-registry.ts +18 -0
  295. package/server/src/storage/s3-provider.ts +153 -0
  296. package/server/src/storage/service.ts +131 -0
  297. package/server/src/storage/types.ts +63 -0
  298. package/server/src/ui-branding.ts +217 -0
  299. package/server/src/version.ts +10 -0
  300. package/server/src/worktree-config.ts +468 -0
  301. package/server/tsconfig.json +9 -0
  302. package/server/vitest.config.ts +7 -0
  303. package/skills/paperclip/SKILL.md +365 -0
  304. package/skills/paperclip/references/api-reference.md +647 -0
  305. package/skills/paperclip/references/company-skills.md +193 -0
  306. package/skills/paperclip-create-agent/SKILL.md +142 -0
  307. package/skills/paperclip-create-agent/references/api-reference.md +105 -0
  308. package/skills/paperclip-create-plugin/SKILL.md +102 -0
  309. package/skills/para-memory-files/SKILL.md +104 -0
  310. package/skills/para-memory-files/references/schemas.md +35 -0
  311. package/tests/e2e/onboarding.spec.ts +142 -0
  312. package/tests/e2e/playwright.config.ts +35 -0
  313. package/tests/release-smoke/docker-auth-onboarding.spec.ts +141 -0
  314. package/tests/release-smoke/playwright.config.ts +28 -0
  315. package/tsconfig.base.json +18 -0
  316. package/tsconfig.json +18 -0
  317. package/ui/README.md +12 -0
  318. package/ui/components.json +21 -0
  319. package/ui/index.html +47 -0
  320. package/ui/package.json +73 -0
  321. package/ui/public/android-chrome-192x192.png +0 -0
  322. package/ui/public/android-chrome-512x512.png +0 -0
  323. package/ui/public/apple-touch-icon.png +0 -0
  324. package/ui/public/brands/opencode-logo-dark-square.svg +18 -0
  325. package/ui/public/brands/opencode-logo-light-square.svg +18 -0
  326. package/ui/public/favicon-16x16.png +0 -0
  327. package/ui/public/favicon-32x32.png +0 -0
  328. package/ui/public/favicon.ico +0 -0
  329. package/ui/public/favicon.svg +9 -0
  330. package/ui/public/site.webmanifest +30 -0
  331. package/ui/public/sprites/1-D-1.png +0 -0
  332. package/ui/public/sprites/1-D-2.png +0 -0
  333. package/ui/public/sprites/1-D-3.png +0 -0
  334. package/ui/public/sprites/1-L-1.png +0 -0
  335. package/ui/public/sprites/1-R-1.png +0 -0
  336. package/ui/public/sprites/10-D-1.png +0 -0
  337. package/ui/public/sprites/10-D-2.png +0 -0
  338. package/ui/public/sprites/10-D-3.png +0 -0
  339. package/ui/public/sprites/10-L-1.png +0 -0
  340. package/ui/public/sprites/10-R-1.png +0 -0
  341. package/ui/public/sprites/11-D-1.png +0 -0
  342. package/ui/public/sprites/11-D-2.png +0 -0
  343. package/ui/public/sprites/11-D-3.png +0 -0
  344. package/ui/public/sprites/11-L-1.png +0 -0
  345. package/ui/public/sprites/11-R-1.png +0 -0
  346. package/ui/public/sprites/12-D-1.png +0 -0
  347. package/ui/public/sprites/12-D-2.png +0 -0
  348. package/ui/public/sprites/12-D-3.png +0 -0
  349. package/ui/public/sprites/12-L-1.png +0 -0
  350. package/ui/public/sprites/12-R-1.png +0 -0
  351. package/ui/public/sprites/13-D-1.png +0 -0
  352. package/ui/public/sprites/13-D-2.png +0 -0
  353. package/ui/public/sprites/13-D-3.png +0 -0
  354. package/ui/public/sprites/13-L-1.png +0 -0
  355. package/ui/public/sprites/13-R-1.png +0 -0
  356. package/ui/public/sprites/14-D-1.png +0 -0
  357. package/ui/public/sprites/14-D-2.png +0 -0
  358. package/ui/public/sprites/14-D-3.png +0 -0
  359. package/ui/public/sprites/14-L-1.png +0 -0
  360. package/ui/public/sprites/14-R-1.png +0 -0
  361. package/ui/public/sprites/2-D-1.png +0 -0
  362. package/ui/public/sprites/2-D-2.png +0 -0
  363. package/ui/public/sprites/2-D-3.png +0 -0
  364. package/ui/public/sprites/2-L-1.png +0 -0
  365. package/ui/public/sprites/2-R-1.png +0 -0
  366. package/ui/public/sprites/3-D-1.png +0 -0
  367. package/ui/public/sprites/3-D-2.png +0 -0
  368. package/ui/public/sprites/3-D-3.png +0 -0
  369. package/ui/public/sprites/3-L-1.png +0 -0
  370. package/ui/public/sprites/3-R-1.png +0 -0
  371. package/ui/public/sprites/4-D-1.png +0 -0
  372. package/ui/public/sprites/4-D-2.png +0 -0
  373. package/ui/public/sprites/4-D-3.png +0 -0
  374. package/ui/public/sprites/4-L-1.png +0 -0
  375. package/ui/public/sprites/4-R-1.png +0 -0
  376. package/ui/public/sprites/5-D-1.png +0 -0
  377. package/ui/public/sprites/5-D-2.png +0 -0
  378. package/ui/public/sprites/5-D-3.png +0 -0
  379. package/ui/public/sprites/5-L-1.png +0 -0
  380. package/ui/public/sprites/5-R-1.png +0 -0
  381. package/ui/public/sprites/6-D-1.png +0 -0
  382. package/ui/public/sprites/6-D-2.png +0 -0
  383. package/ui/public/sprites/6-D-3.png +0 -0
  384. package/ui/public/sprites/6-L-1.png +0 -0
  385. package/ui/public/sprites/6-R-1.png +0 -0
  386. package/ui/public/sprites/7-D-1.png +0 -0
  387. package/ui/public/sprites/7-D-2.png +0 -0
  388. package/ui/public/sprites/7-D-3.png +0 -0
  389. package/ui/public/sprites/7-L-1.png +0 -0
  390. package/ui/public/sprites/7-R-1.png +0 -0
  391. package/ui/public/sprites/8-D-1.png +0 -0
  392. package/ui/public/sprites/8-D-2.png +0 -0
  393. package/ui/public/sprites/8-D-3.png +0 -0
  394. package/ui/public/sprites/8-L-1.png +0 -0
  395. package/ui/public/sprites/8-R-1.png +0 -0
  396. package/ui/public/sprites/9-D-1.png +0 -0
  397. package/ui/public/sprites/9-D-2.png +0 -0
  398. package/ui/public/sprites/9-D-3.png +0 -0
  399. package/ui/public/sprites/9-L-1.png +0 -0
  400. package/ui/public/sprites/9-R-1.png +0 -0
  401. package/ui/public/sprites/ceo-lobster.png +0 -0
  402. package/ui/public/sw.js +42 -0
  403. package/ui/public/worktree-favicon-16x16.png +0 -0
  404. package/ui/public/worktree-favicon-32x32.png +0 -0
  405. package/ui/public/worktree-favicon.ico +0 -0
  406. package/ui/public/worktree-favicon.svg +9 -0
  407. package/ui/src/App.tsx +354 -0
  408. package/ui/src/adapters/claude-local/config-fields.tsx +138 -0
  409. package/ui/src/adapters/claude-local/index.ts +13 -0
  410. package/ui/src/adapters/codex-local/config-fields.tsx +104 -0
  411. package/ui/src/adapters/codex-local/index.ts +13 -0
  412. package/ui/src/adapters/cursor/config-fields.tsx +49 -0
  413. package/ui/src/adapters/cursor/index.ts +13 -0
  414. package/ui/src/adapters/gemini-local/config-fields.tsx +51 -0
  415. package/ui/src/adapters/gemini-local/index.ts +13 -0
  416. package/ui/src/adapters/http/build-config.ts +9 -0
  417. package/ui/src/adapters/http/config-fields.tsx +38 -0
  418. package/ui/src/adapters/http/index.ts +12 -0
  419. package/ui/src/adapters/http/parse-stdout.ts +5 -0
  420. package/ui/src/adapters/index.ts +9 -0
  421. package/ui/src/adapters/local-workspace-runtime-fields.tsx +5 -0
  422. package/ui/src/adapters/openclaw-gateway/config-fields.tsx +237 -0
  423. package/ui/src/adapters/openclaw-gateway/index.ts +13 -0
  424. package/ui/src/adapters/opencode-local/config-fields.tsx +72 -0
  425. package/ui/src/adapters/opencode-local/index.ts +13 -0
  426. package/ui/src/adapters/pi-local/config-fields.tsx +49 -0
  427. package/ui/src/adapters/pi-local/index.ts +13 -0
  428. package/ui/src/adapters/process/build-config.ts +18 -0
  429. package/ui/src/adapters/process/config-fields.tsx +77 -0
  430. package/ui/src/adapters/process/index.ts +12 -0
  431. package/ui/src/adapters/process/parse-stdout.ts +5 -0
  432. package/ui/src/adapters/registry.ts +34 -0
  433. package/ui/src/adapters/runtime-json-fields.tsx +122 -0
  434. package/ui/src/adapters/transcript.test.ts +30 -0
  435. package/ui/src/adapters/transcript.ts +62 -0
  436. package/ui/src/adapters/types.ts +34 -0
  437. package/ui/src/api/access.ts +160 -0
  438. package/ui/src/api/activity.ts +37 -0
  439. package/ui/src/api/agents.ts +194 -0
  440. package/ui/src/api/approvals.ts +25 -0
  441. package/ui/src/api/assets.ts +30 -0
  442. package/ui/src/api/auth.ts +74 -0
  443. package/ui/src/api/budgets.ts +21 -0
  444. package/ui/src/api/client.ts +50 -0
  445. package/ui/src/api/companies.ts +59 -0
  446. package/ui/src/api/companySkills.ts +55 -0
  447. package/ui/src/api/costs.ts +60 -0
  448. package/ui/src/api/dashboard.ts +7 -0
  449. package/ui/src/api/execution-workspaces.ts +27 -0
  450. package/ui/src/api/goals.ts +12 -0
  451. package/ui/src/api/health.ts +41 -0
  452. package/ui/src/api/heartbeats.ts +62 -0
  453. package/ui/src/api/index.ts +18 -0
  454. package/ui/src/api/instanceSettings.ts +19 -0
  455. package/ui/src/api/issues.ts +115 -0
  456. package/ui/src/api/plugins.ts +424 -0
  457. package/ui/src/api/projects.ts +34 -0
  458. package/ui/src/api/routines.ts +59 -0
  459. package/ui/src/api/secrets.ts +26 -0
  460. package/ui/src/api/sidebarBadges.ts +7 -0
  461. package/ui/src/components/AccountingModelCard.tsx +69 -0
  462. package/ui/src/components/ActiveAgentsPanel.tsx +157 -0
  463. package/ui/src/components/ActivityCharts.tsx +264 -0
  464. package/ui/src/components/ActivityRow.tsx +147 -0
  465. package/ui/src/components/AgentActionButtons.tsx +51 -0
  466. package/ui/src/components/AgentConfigForm.tsx +1468 -0
  467. package/ui/src/components/AgentIconPicker.tsx +81 -0
  468. package/ui/src/components/AgentProperties.tsx +107 -0
  469. package/ui/src/components/ApprovalCard.tsx +107 -0
  470. package/ui/src/components/ApprovalPayload.tsx +134 -0
  471. package/ui/src/components/AsciiArtAnimation.tsx +355 -0
  472. package/ui/src/components/BillerSpendCard.tsx +146 -0
  473. package/ui/src/components/BreadcrumbBar.tsx +113 -0
  474. package/ui/src/components/BudgetIncidentCard.tsx +101 -0
  475. package/ui/src/components/BudgetPolicyCard.tsx +220 -0
  476. package/ui/src/components/BudgetSidebarMarker.tsx +13 -0
  477. package/ui/src/components/ClaudeSubscriptionPanel.tsx +141 -0
  478. package/ui/src/components/CodexSubscriptionPanel.tsx +158 -0
  479. package/ui/src/components/CommandPalette.tsx +239 -0
  480. package/ui/src/components/CommentThread.tsx +503 -0
  481. package/ui/src/components/CompanyPatternIcon.tsx +212 -0
  482. package/ui/src/components/CompanyRail.tsx +329 -0
  483. package/ui/src/components/CompanySwitcher.tsx +81 -0
  484. package/ui/src/components/CopyText.tsx +56 -0
  485. package/ui/src/components/DevRestartBanner.tsx +89 -0
  486. package/ui/src/components/EmptyState.tsx +27 -0
  487. package/ui/src/components/EntityRow.tsx +69 -0
  488. package/ui/src/components/FilterBar.tsx +39 -0
  489. package/ui/src/components/FinanceBillerCard.tsx +45 -0
  490. package/ui/src/components/FinanceKindCard.tsx +44 -0
  491. package/ui/src/components/FinanceTimelineCard.tsx +72 -0
  492. package/ui/src/components/GoalProperties.tsx +165 -0
  493. package/ui/src/components/GoalTree.tsx +118 -0
  494. package/ui/src/components/Identity.tsx +39 -0
  495. package/ui/src/components/InlineEditor.tsx +248 -0
  496. package/ui/src/components/InlineEntitySelector.tsx +206 -0
  497. package/ui/src/components/InstanceSidebar.tsx +53 -0
  498. package/ui/src/components/IssueDocumentsSection.tsx +892 -0
  499. package/ui/src/components/IssueProperties.tsx +621 -0
  500. package/ui/src/components/IssueRow.tsx +149 -0
  501. package/ui/src/components/IssueWorkspaceCard.tsx +404 -0
  502. package/ui/src/components/IssuesList.tsx +889 -0
  503. package/ui/src/components/JsonSchemaForm.tsx +1048 -0
  504. package/ui/src/components/KanbanBoard.tsx +275 -0
  505. package/ui/src/components/Layout.tsx +441 -0
  506. package/ui/src/components/LiveRunWidget.tsx +160 -0
  507. package/ui/src/components/MarkdownBody.test.tsx +50 -0
  508. package/ui/src/components/MarkdownBody.tsx +152 -0
  509. package/ui/src/components/MarkdownEditor.tsx +622 -0
  510. package/ui/src/components/MetricCard.tsx +53 -0
  511. package/ui/src/components/MobileBottomNav.tsx +123 -0
  512. package/ui/src/components/NewAgentDialog.tsx +223 -0
  513. package/ui/src/components/NewGoalDialog.tsx +283 -0
  514. package/ui/src/components/NewIssueDialog.tsx +1473 -0
  515. package/ui/src/components/NewProjectDialog.tsx +451 -0
  516. package/ui/src/components/OnboardingWizard.tsx +1392 -0
  517. package/ui/src/components/OpenCodeLogoIcon.tsx +22 -0
  518. package/ui/src/components/PackageFileTree.tsx +318 -0
  519. package/ui/src/components/PageSkeleton.tsx +180 -0
  520. package/ui/src/components/PageTabBar.tsx +45 -0
  521. package/ui/src/components/PathInstructionsModal.tsx +143 -0
  522. package/ui/src/components/PriorityIcon.tsx +77 -0
  523. package/ui/src/components/ProjectProperties.tsx +1127 -0
  524. package/ui/src/components/PropertiesPanel.tsx +29 -0
  525. package/ui/src/components/ProviderQuotaCard.tsx +417 -0
  526. package/ui/src/components/QuotaBar.tsx +65 -0
  527. package/ui/src/components/ReportsToPicker.tsx +127 -0
  528. package/ui/src/components/ScheduleEditor.tsx +344 -0
  529. package/ui/src/components/ScrollToBottom.tsx +79 -0
  530. package/ui/src/components/Sidebar.tsx +130 -0
  531. package/ui/src/components/SidebarAgents.tsx +146 -0
  532. package/ui/src/components/SidebarNavItem.tsx +92 -0
  533. package/ui/src/components/SidebarProjects.tsx +234 -0
  534. package/ui/src/components/SidebarSection.tsx +17 -0
  535. package/ui/src/components/StatusBadge.tsx +15 -0
  536. package/ui/src/components/StatusIcon.tsx +71 -0
  537. package/ui/src/components/SwipeToArchive.tsx +152 -0
  538. package/ui/src/components/ToastViewport.tsx +99 -0
  539. package/ui/src/components/WorktreeBanner.tsx +25 -0
  540. package/ui/src/components/agent-config-defaults.ts +31 -0
  541. package/ui/src/components/agent-config-primitives.tsx +476 -0
  542. package/ui/src/components/transcript/RunTranscriptView.test.tsx +84 -0
  543. package/ui/src/components/transcript/RunTranscriptView.tsx +1015 -0
  544. package/ui/src/components/transcript/useLiveRunTranscripts.ts +297 -0
  545. package/ui/src/components/ui/avatar.tsx +107 -0
  546. package/ui/src/components/ui/badge.tsx +48 -0
  547. package/ui/src/components/ui/breadcrumb.tsx +109 -0
  548. package/ui/src/components/ui/button.tsx +64 -0
  549. package/ui/src/components/ui/card.tsx +92 -0
  550. package/ui/src/components/ui/checkbox.tsx +32 -0
  551. package/ui/src/components/ui/collapsible.tsx +33 -0
  552. package/ui/src/components/ui/command.tsx +194 -0
  553. package/ui/src/components/ui/dialog.tsx +156 -0
  554. package/ui/src/components/ui/dropdown-menu.tsx +257 -0
  555. package/ui/src/components/ui/input.tsx +21 -0
  556. package/ui/src/components/ui/label.tsx +22 -0
  557. package/ui/src/components/ui/popover.tsx +88 -0
  558. package/ui/src/components/ui/scroll-area.tsx +56 -0
  559. package/ui/src/components/ui/select.tsx +188 -0
  560. package/ui/src/components/ui/separator.tsx +28 -0
  561. package/ui/src/components/ui/sheet.tsx +143 -0
  562. package/ui/src/components/ui/skeleton.tsx +13 -0
  563. package/ui/src/components/ui/tabs.tsx +89 -0
  564. package/ui/src/components/ui/textarea.tsx +18 -0
  565. package/ui/src/components/ui/tooltip.tsx +57 -0
  566. package/ui/src/components/visual-office/AgentAvatar.tsx +99 -0
  567. package/ui/src/components/visual-office/OfficeViewExact.tsx +417 -0
  568. package/ui/src/components/visual-office/i18n.ts +52 -0
  569. package/ui/src/components/visual-office/office-view/CliUsagePanel.tsx +240 -0
  570. package/ui/src/components/visual-office/office-view/VirtualPadOverlay.tsx +104 -0
  571. package/ui/src/components/visual-office/office-view/buildScene-break-room.ts +248 -0
  572. package/ui/src/components/visual-office/office-view/buildScene-ceo-hallway.ts +345 -0
  573. package/ui/src/components/visual-office/office-view/buildScene-department-agent.ts +242 -0
  574. package/ui/src/components/visual-office/office-view/buildScene-departments.ts +360 -0
  575. package/ui/src/components/visual-office/office-view/buildScene-final-layers.ts +113 -0
  576. package/ui/src/components/visual-office/office-view/buildScene-types.ts +91 -0
  577. package/ui/src/components/visual-office/office-view/buildScene.ts +232 -0
  578. package/ui/src/components/visual-office/office-view/drawing-core.ts +374 -0
  579. package/ui/src/components/visual-office/office-view/drawing-furniture-a.ts +338 -0
  580. package/ui/src/components/visual-office/office-view/drawing-furniture-b.ts +241 -0
  581. package/ui/src/components/visual-office/office-view/model.ts +301 -0
  582. package/ui/src/components/visual-office/office-view/officeTicker.ts +455 -0
  583. package/ui/src/components/visual-office/office-view/officeTickerRoomAndDelivery.ts +133 -0
  584. package/ui/src/components/visual-office/office-view/themes-locale.ts +460 -0
  585. package/ui/src/components/visual-office/office-view/useCliUsage.ts +37 -0
  586. package/ui/src/components/visual-office/office-view/useOfficeDeliveryEffects.ts +465 -0
  587. package/ui/src/components/visual-office/office-view/useOfficePixiRuntime.ts +282 -0
  588. package/ui/src/components/visual-office/types.ts +123 -0
  589. package/ui/src/context/BreadcrumbContext.tsx +44 -0
  590. package/ui/src/context/CompanyContext.tsx +151 -0
  591. package/ui/src/context/DialogContext.tsx +135 -0
  592. package/ui/src/context/LiveUpdatesProvider.test.ts +119 -0
  593. package/ui/src/context/LiveUpdatesProvider.tsx +760 -0
  594. package/ui/src/context/PanelContext.tsx +73 -0
  595. package/ui/src/context/SidebarContext.tsx +43 -0
  596. package/ui/src/context/ThemeContext.tsx +83 -0
  597. package/ui/src/context/ToastContext.tsx +172 -0
  598. package/ui/src/fixtures/runTranscriptFixtures.ts +226 -0
  599. package/ui/src/hooks/useAgentOrder.ts +105 -0
  600. package/ui/src/hooks/useAutosaveIndicator.ts +72 -0
  601. package/ui/src/hooks/useCompanyPageMemory.test.ts +90 -0
  602. package/ui/src/hooks/useCompanyPageMemory.ts +79 -0
  603. package/ui/src/hooks/useDateRange.ts +120 -0
  604. package/ui/src/hooks/useInboxBadge.ts +132 -0
  605. package/ui/src/hooks/useKeyboardShortcuts.ts +40 -0
  606. package/ui/src/hooks/useProjectOrder.ts +106 -0
  607. package/ui/src/index.css +770 -0
  608. package/ui/src/lib/agent-icons.ts +99 -0
  609. package/ui/src/lib/agent-order.ts +107 -0
  610. package/ui/src/lib/agent-skills-state.test.ts +90 -0
  611. package/ui/src/lib/agent-skills-state.ts +41 -0
  612. package/ui/src/lib/assignees.test.ts +92 -0
  613. package/ui/src/lib/assignees.ts +82 -0
  614. package/ui/src/lib/color-contrast.ts +107 -0
  615. package/ui/src/lib/company-export-selection.test.ts +41 -0
  616. package/ui/src/lib/company-export-selection.ts +57 -0
  617. package/ui/src/lib/company-page-memory.ts +65 -0
  618. package/ui/src/lib/company-portability-sidebar.test.ts +101 -0
  619. package/ui/src/lib/company-portability-sidebar.ts +62 -0
  620. package/ui/src/lib/company-routes.ts +88 -0
  621. package/ui/src/lib/company-selection.test.ts +34 -0
  622. package/ui/src/lib/company-selection.ts +18 -0
  623. package/ui/src/lib/groupBy.ts +11 -0
  624. package/ui/src/lib/inbox.test.ts +404 -0
  625. package/ui/src/lib/inbox.ts +292 -0
  626. package/ui/src/lib/instance-settings.test.ts +26 -0
  627. package/ui/src/lib/instance-settings.ts +25 -0
  628. package/ui/src/lib/issueDetailBreadcrumb.ts +24 -0
  629. package/ui/src/lib/legacy-agent-config.test.ts +40 -0
  630. package/ui/src/lib/legacy-agent-config.ts +17 -0
  631. package/ui/src/lib/mention-aware-link-node.test.ts +50 -0
  632. package/ui/src/lib/mention-aware-link-node.ts +67 -0
  633. package/ui/src/lib/mention-chips.ts +168 -0
  634. package/ui/src/lib/mention-deletion.test.ts +87 -0
  635. package/ui/src/lib/mention-deletion.ts +143 -0
  636. package/ui/src/lib/model-utils.ts +16 -0
  637. package/ui/src/lib/onboarding-goal.test.ts +22 -0
  638. package/ui/src/lib/onboarding-goal.ts +18 -0
  639. package/ui/src/lib/onboarding-launch.test.ts +131 -0
  640. package/ui/src/lib/onboarding-launch.ts +54 -0
  641. package/ui/src/lib/onboarding-route.test.ts +80 -0
  642. package/ui/src/lib/onboarding-route.ts +51 -0
  643. package/ui/src/lib/portable-files.ts +42 -0
  644. package/ui/src/lib/project-order.ts +71 -0
  645. package/ui/src/lib/queryKeys.ts +140 -0
  646. package/ui/src/lib/recent-assignees.ts +36 -0
  647. package/ui/src/lib/router.tsx +76 -0
  648. package/ui/src/lib/routine-trigger-patch.test.ts +72 -0
  649. package/ui/src/lib/routine-trigger-patch.ts +31 -0
  650. package/ui/src/lib/status-colors.ts +108 -0
  651. package/ui/src/lib/timeAgo.ts +31 -0
  652. package/ui/src/lib/utils.ts +168 -0
  653. package/ui/src/lib/worktree-branding.ts +65 -0
  654. package/ui/src/lib/zip.test.ts +289 -0
  655. package/ui/src/lib/zip.ts +284 -0
  656. package/ui/src/main.tsx +67 -0
  657. package/ui/src/pages/Activity.tsx +141 -0
  658. package/ui/src/pages/AgentDetail.tsx +4053 -0
  659. package/ui/src/pages/Agents.tsx +415 -0
  660. package/ui/src/pages/ApprovalDetail.tsx +369 -0
  661. package/ui/src/pages/Approvals.tsx +132 -0
  662. package/ui/src/pages/Auth.tsx +180 -0
  663. package/ui/src/pages/BoardClaim.tsx +125 -0
  664. package/ui/src/pages/CliAuth.tsx +184 -0
  665. package/ui/src/pages/Companies.tsx +297 -0
  666. package/ui/src/pages/CompanyExport.tsx +1019 -0
  667. package/ui/src/pages/CompanyImport.tsx +1355 -0
  668. package/ui/src/pages/CompanySettings.tsx +661 -0
  669. package/ui/src/pages/CompanySkills.tsx +1171 -0
  670. package/ui/src/pages/Costs.tsx +1103 -0
  671. package/ui/src/pages/Dashboard.tsx +388 -0
  672. package/ui/src/pages/DesignGuide.tsx +1330 -0
  673. package/ui/src/pages/ExecutionWorkspaceDetail.tsx +82 -0
  674. package/ui/src/pages/GoalDetail.tsx +197 -0
  675. package/ui/src/pages/Goals.tsx +63 -0
  676. package/ui/src/pages/Inbox.tsx +1291 -0
  677. package/ui/src/pages/InstanceExperimentalSettings.tsx +139 -0
  678. package/ui/src/pages/InstanceGeneralSettings.tsx +104 -0
  679. package/ui/src/pages/InstanceSettings.tsx +284 -0
  680. package/ui/src/pages/InviteLanding.tsx +320 -0
  681. package/ui/src/pages/IssueDetail.tsx +1201 -0
  682. package/ui/src/pages/Issues.tsx +116 -0
  683. package/ui/src/pages/MyIssues.tsx +72 -0
  684. package/ui/src/pages/NewAgent.tsx +353 -0
  685. package/ui/src/pages/NotFound.tsx +66 -0
  686. package/ui/src/pages/Org.tsx +132 -0
  687. package/ui/src/pages/OrgChart.tsx +447 -0
  688. package/ui/src/pages/PluginManager.tsx +510 -0
  689. package/ui/src/pages/PluginPage.tsx +156 -0
  690. package/ui/src/pages/PluginSettings.tsx +836 -0
  691. package/ui/src/pages/ProjectDetail.tsx +633 -0
  692. package/ui/src/pages/Projects.tsx +87 -0
  693. package/ui/src/pages/RoutineDetail.tsx +1022 -0
  694. package/ui/src/pages/Routines.tsx +661 -0
  695. package/ui/src/pages/RunTranscriptUxLab.tsx +334 -0
  696. package/ui/src/pages/VisualOffice.tsx +243 -0
  697. package/ui/src/plugins/bridge-init.ts +69 -0
  698. package/ui/src/plugins/bridge.ts +476 -0
  699. package/ui/src/plugins/launchers.tsx +834 -0
  700. package/ui/src/plugins/slots.tsx +855 -0
  701. package/ui/tsconfig.json +21 -0
  702. package/ui/vite.config.ts +23 -0
  703. package/ui/vitest.config.ts +14 -0
  704. package/vitest.config.ts +11 -0
@@ -0,0 +1,822 @@
1
+ /**
2
+ * PluginLifecycleManager — state-machine controller for plugin status
3
+ * transitions and worker process coordination.
4
+ *
5
+ * Each plugin moves through a well-defined state machine:
6
+ *
7
+ * ```
8
+ * installed ──→ ready ──→ disabled
9
+ * │ │ │
10
+ * │ ├──→ error│
11
+ * │ ↓ │
12
+ * │ upgrade_pending │
13
+ * │ │ │
14
+ * ↓ ↓ ↓
15
+ * uninstalled
16
+ * ```
17
+ *
18
+ * The lifecycle manager:
19
+ *
20
+ * 1. **Validates transitions** — Only transitions defined in
21
+ * `VALID_TRANSITIONS` are allowed; invalid transitions throw.
22
+ *
23
+ * 2. **Coordinates workers** — When a plugin moves to `ready`, its
24
+ * worker process is started. When it moves out of `ready`, the
25
+ * worker is stopped gracefully.
26
+ *
27
+ * 3. **Emits events** — `plugin.loaded`, `plugin.enabled`,
28
+ * `plugin.disabled`, `plugin.unloaded`, `plugin.status_changed`
29
+ * events are emitted so that other services (job coordinator,
30
+ * tool dispatcher, event bus) can react accordingly.
31
+ *
32
+ * 4. **Persists state** — Status changes are written to the database
33
+ * through the plugin registry service.
34
+ *
35
+ * @see PLUGIN_SPEC.md §12 — Process Model
36
+ * @see PLUGIN_SPEC.md §12.5 — Graceful Shutdown Policy
37
+ */
38
+ import { EventEmitter } from "node:events";
39
+ import type { Db } from "@corporateai/db";
40
+ import type {
41
+ PluginStatus,
42
+ PluginRecord,
43
+ PaperclipPluginManifestV1,
44
+ } from "@corporateai/shared";
45
+ import { pluginRegistryService } from "./plugin-registry.js";
46
+ import { pluginLoader, type PluginLoader } from "./plugin-loader.js";
47
+ import type { PluginWorkerManager, WorkerStartOptions } from "./plugin-worker-manager.js";
48
+ import { badRequest, notFound } from "../errors.js";
49
+ import { logger } from "../middleware/logger.js";
50
+
51
+ // ---------------------------------------------------------------------------
52
+ // Lifecycle state machine
53
+ // ---------------------------------------------------------------------------
54
+
55
+ /**
56
+ * Valid state transitions for the plugin lifecycle.
57
+ *
58
+ * installed → ready (initial load succeeds)
59
+ * installed → error (initial load fails)
60
+ * installed → uninstalled (abort installation)
61
+ *
62
+ * ready → disabled (operator disables plugin)
63
+ * ready → error (runtime failure)
64
+ * ready → upgrade_pending (upgrade with new capabilities)
65
+ * ready → uninstalled (uninstall)
66
+ *
67
+ * disabled → ready (operator re-enables plugin)
68
+ * disabled → uninstalled (uninstall while disabled)
69
+ *
70
+ * error → ready (retry / recovery)
71
+ * error → uninstalled (give up and uninstall)
72
+ *
73
+ * upgrade_pending → ready (operator approves new capabilities)
74
+ * upgrade_pending → error (upgrade worker fails)
75
+ * upgrade_pending → uninstalled (reject upgrade and uninstall)
76
+ *
77
+ * uninstalled → installed (reinstall)
78
+ */
79
+ const VALID_TRANSITIONS: Record<string, readonly PluginStatus[]> = {
80
+ installed: ["ready", "error", "uninstalled"],
81
+ ready: ["ready", "disabled", "error", "upgrade_pending", "uninstalled"],
82
+ disabled: ["ready", "uninstalled"],
83
+ error: ["ready", "uninstalled"],
84
+ upgrade_pending: ["ready", "error", "uninstalled"],
85
+ uninstalled: ["installed"], // reinstall
86
+ };
87
+
88
+ /**
89
+ * Check whether a transition from `from` → `to` is valid.
90
+ */
91
+ function isValidTransition(from: PluginStatus, to: PluginStatus): boolean {
92
+ return VALID_TRANSITIONS[from]?.includes(to) ?? false;
93
+ }
94
+
95
+ // ---------------------------------------------------------------------------
96
+ // Lifecycle events
97
+ // ---------------------------------------------------------------------------
98
+
99
+ /**
100
+ * Events emitted by the PluginLifecycleManager.
101
+ * Consumers can subscribe to these for routing-table updates, UI refresh
102
+ * notifications, and observability.
103
+ */
104
+ export interface PluginLifecycleEvents {
105
+ /** Emitted after a plugin is loaded (installed → ready). */
106
+ "plugin.loaded": { pluginId: string; pluginKey: string };
107
+ /** Emitted after a plugin transitions to ready (enabled). */
108
+ "plugin.enabled": { pluginId: string; pluginKey: string };
109
+ /** Emitted after a plugin is disabled (ready → disabled). */
110
+ "plugin.disabled": { pluginId: string; pluginKey: string; reason?: string };
111
+ /** Emitted after a plugin is unloaded (any → uninstalled). */
112
+ "plugin.unloaded": { pluginId: string; pluginKey: string; removeData: boolean };
113
+ /** Emitted on any status change. */
114
+ "plugin.status_changed": {
115
+ pluginId: string;
116
+ pluginKey: string;
117
+ previousStatus: PluginStatus;
118
+ newStatus: PluginStatus;
119
+ };
120
+ /** Emitted when a plugin enters an error state. */
121
+ "plugin.error": { pluginId: string; pluginKey: string; error: string };
122
+ /** Emitted when a plugin enters upgrade_pending. */
123
+ "plugin.upgrade_pending": { pluginId: string; pluginKey: string };
124
+ /** Emitted when a plugin worker process has been started. */
125
+ "plugin.worker_started": { pluginId: string; pluginKey: string };
126
+ /** Emitted when a plugin worker process has been stopped. */
127
+ "plugin.worker_stopped": { pluginId: string; pluginKey: string };
128
+ }
129
+
130
+ type LifecycleEventName = keyof PluginLifecycleEvents;
131
+ type LifecycleEventPayload<K extends LifecycleEventName> = PluginLifecycleEvents[K];
132
+
133
+ // ---------------------------------------------------------------------------
134
+ // PluginLifecycleManager
135
+ // ---------------------------------------------------------------------------
136
+
137
+ export interface PluginLifecycleManager {
138
+ /**
139
+ * Load a newly installed plugin – transitions `installed` → `ready`.
140
+ *
141
+ * This is called after the registry has persisted the initial install record.
142
+ * The caller should have already spawned the worker and performed health
143
+ * checks before calling this. If the worker fails, call `markError` instead.
144
+ */
145
+ load(pluginId: string): Promise<PluginRecord>;
146
+
147
+ /**
148
+ * Enable a plugin that is in `disabled`, `error`, or `upgrade_pending` state.
149
+ * Transitions → `ready`.
150
+ */
151
+ enable(pluginId: string): Promise<PluginRecord>;
152
+
153
+ /**
154
+ * Disable a running plugin.
155
+ * Transitions `ready` → `disabled`.
156
+ */
157
+ disable(pluginId: string, reason?: string): Promise<PluginRecord>;
158
+
159
+ /**
160
+ * Unload (uninstall) a plugin from any active state.
161
+ * Transitions → `uninstalled`.
162
+ *
163
+ * When `removeData` is true, the plugin row and cascaded config are
164
+ * hard-deleted. Otherwise a soft-delete sets status to `uninstalled`.
165
+ */
166
+ unload(pluginId: string, removeData?: boolean): Promise<PluginRecord | null>;
167
+
168
+ /**
169
+ * Mark a plugin as errored (e.g. worker crash, health-check failure).
170
+ * Transitions → `error`.
171
+ */
172
+ markError(pluginId: string, error: string): Promise<PluginRecord>;
173
+
174
+ /**
175
+ * Mark a plugin as requiring upgrade approval.
176
+ * Transitions `ready` → `upgrade_pending`.
177
+ */
178
+ markUpgradePending(pluginId: string): Promise<PluginRecord>;
179
+
180
+ /**
181
+ * Upgrade a plugin to a newer version.
182
+ * This is a placeholder that handles the lifecycle state transition.
183
+ * The actual package installation is handled by plugin-loader.
184
+ *
185
+ * If the upgrade adds new capabilities, transitions to `upgrade_pending`.
186
+ * Otherwise, transitions to `ready` directly.
187
+ */
188
+ upgrade(pluginId: string, version?: string): Promise<PluginRecord>;
189
+
190
+ /**
191
+ * Start the worker process for a plugin that is already in `ready` state.
192
+ *
193
+ * This is used by the server startup orchestration to start workers for
194
+ * plugins that were persisted as `ready`. It requires a `PluginWorkerManager`
195
+ * to have been provided at construction time.
196
+ *
197
+ * @param pluginId - The UUID of the plugin to start
198
+ * @param options - Worker start options (entrypoint path, config, etc.)
199
+ * @throws if no worker manager is configured or the plugin is not ready
200
+ */
201
+ startWorker(pluginId: string, options: WorkerStartOptions): Promise<void>;
202
+
203
+ /**
204
+ * Stop the worker process for a plugin without changing lifecycle state.
205
+ *
206
+ * This is used during server shutdown to gracefully stop all workers.
207
+ * It does not transition the plugin state — plugins remain in their
208
+ * current status so they can be restarted on next server boot.
209
+ *
210
+ * @param pluginId - The UUID of the plugin to stop
211
+ */
212
+ stopWorker(pluginId: string): Promise<void>;
213
+
214
+ /**
215
+ * Restart the worker process for a running plugin.
216
+ *
217
+ * Stops and re-starts the worker process. The plugin remains in `ready`
218
+ * state throughout. This is typically called after a config change.
219
+ *
220
+ * @param pluginId - The UUID of the plugin to restart
221
+ * @throws if no worker manager is configured or the plugin is not ready
222
+ */
223
+ restartWorker(pluginId: string): Promise<void>;
224
+
225
+ /**
226
+ * Get the current lifecycle state for a plugin.
227
+ */
228
+ getStatus(pluginId: string): Promise<PluginStatus | null>;
229
+
230
+ /**
231
+ * Check whether a transition is allowed from the plugin's current state.
232
+ */
233
+ canTransition(pluginId: string, to: PluginStatus): Promise<boolean>;
234
+
235
+ /**
236
+ * Subscribe to lifecycle events.
237
+ */
238
+ on<K extends LifecycleEventName>(
239
+ event: K,
240
+ listener: (payload: LifecycleEventPayload<K>) => void,
241
+ ): void;
242
+
243
+ /**
244
+ * Unsubscribe from lifecycle events.
245
+ */
246
+ off<K extends LifecycleEventName>(
247
+ event: K,
248
+ listener: (payload: LifecycleEventPayload<K>) => void,
249
+ ): void;
250
+
251
+ /**
252
+ * Subscribe to a lifecycle event once.
253
+ */
254
+ once<K extends LifecycleEventName>(
255
+ event: K,
256
+ listener: (payload: LifecycleEventPayload<K>) => void,
257
+ ): void;
258
+ }
259
+
260
+ // ---------------------------------------------------------------------------
261
+ // Factory
262
+ // ---------------------------------------------------------------------------
263
+
264
+ /**
265
+ * Options for constructing a PluginLifecycleManager.
266
+ */
267
+ export interface PluginLifecycleManagerOptions {
268
+ /** Plugin loader instance. Falls back to the default if omitted. */
269
+ loader?: PluginLoader;
270
+
271
+ /**
272
+ * Worker process manager. When provided, lifecycle transitions that bring
273
+ * a plugin online (load, enable, upgrade-to-ready) will start the worker
274
+ * process, and transitions that take a plugin offline (disable, unload,
275
+ * markError) will stop it.
276
+ *
277
+ * When omitted the lifecycle manager operates in state-only mode — the
278
+ * caller is responsible for managing worker processes externally.
279
+ */
280
+ workerManager?: PluginWorkerManager;
281
+ }
282
+
283
+ /**
284
+ * Create a PluginLifecycleManager.
285
+ *
286
+ * This service orchestrates plugin state transitions on top of the
287
+ * `pluginRegistryService` (which handles raw DB persistence). It enforces
288
+ * the lifecycle state machine, emits events for downstream consumers
289
+ * (routing tables, UI, observability), and manages worker processes via
290
+ * the `PluginWorkerManager` when one is provided.
291
+ *
292
+ * Usage:
293
+ * ```ts
294
+ * const lifecycle = pluginLifecycleManager(db, {
295
+ * workerManager: createPluginWorkerManager(),
296
+ * });
297
+ * lifecycle.on("plugin.enabled", ({ pluginId }) => { ... });
298
+ * await lifecycle.load(pluginId);
299
+ * ```
300
+ *
301
+ * @see PLUGIN_SPEC.md §21.3 — `plugins.status` column
302
+ * @see PLUGIN_SPEC.md §12 — Process Model
303
+ */
304
+ export function pluginLifecycleManager(
305
+ db: Db,
306
+ options?: PluginLoader | PluginLifecycleManagerOptions,
307
+ ): PluginLifecycleManager {
308
+ // Support the legacy signature: pluginLifecycleManager(db, loader)
309
+ // as well as the new options object form.
310
+ let loaderArg: PluginLoader | undefined;
311
+ let workerManager: PluginWorkerManager | undefined;
312
+
313
+ if (options && typeof options === "object" && "discoverAll" in options) {
314
+ // Legacy: second arg is a PluginLoader directly
315
+ loaderArg = options as PluginLoader;
316
+ } else if (options && typeof options === "object") {
317
+ const opts = options as PluginLifecycleManagerOptions;
318
+ loaderArg = opts.loader;
319
+ workerManager = opts.workerManager;
320
+ }
321
+
322
+ const registry = pluginRegistryService(db);
323
+ const pluginLoaderInstance = loaderArg ?? pluginLoader(db);
324
+ const emitter = new EventEmitter();
325
+ emitter.setMaxListeners(100); // plugins may have many listeners; 100 is a safe upper bound
326
+
327
+ const log = logger.child({ service: "plugin-lifecycle" });
328
+
329
+ // -----------------------------------------------------------------------
330
+ // Internal helpers
331
+ // -----------------------------------------------------------------------
332
+
333
+ async function requirePlugin(pluginId: string): Promise<PluginRecord> {
334
+ const plugin = await registry.getById(pluginId);
335
+ if (!plugin) throw notFound(`Plugin not found: ${pluginId}`);
336
+ return plugin as PluginRecord;
337
+ }
338
+
339
+ function assertTransition(plugin: PluginRecord, to: PluginStatus): void {
340
+ if (!isValidTransition(plugin.status, to)) {
341
+ throw badRequest(
342
+ `Invalid lifecycle transition: ${plugin.status} → ${to} for plugin ${plugin.pluginKey}`,
343
+ );
344
+ }
345
+ }
346
+
347
+ async function transition(
348
+ pluginId: string,
349
+ to: PluginStatus,
350
+ lastError: string | null = null,
351
+ existingPlugin?: PluginRecord,
352
+ ): Promise<PluginRecord> {
353
+ const plugin = existingPlugin ?? await requirePlugin(pluginId);
354
+ assertTransition(plugin, to);
355
+
356
+ const previousStatus = plugin.status;
357
+
358
+ const updated = await registry.updateStatus(pluginId, {
359
+ status: to,
360
+ lastError,
361
+ });
362
+
363
+ if (!updated) throw notFound(`Plugin not found after status update: ${pluginId}`);
364
+ const result = updated as PluginRecord;
365
+
366
+ log.info(
367
+ { pluginId, pluginKey: result.pluginKey, from: previousStatus, to },
368
+ `plugin lifecycle: ${previousStatus} → ${to}`,
369
+ );
370
+
371
+ // Emit the generic status_changed event
372
+ emitter.emit("plugin.status_changed", {
373
+ pluginId,
374
+ pluginKey: result.pluginKey,
375
+ previousStatus,
376
+ newStatus: to,
377
+ });
378
+
379
+ return result;
380
+ }
381
+
382
+ function emitDomain(
383
+ event: LifecycleEventName,
384
+ payload: PluginLifecycleEvents[LifecycleEventName],
385
+ ): void {
386
+ emitter.emit(event, payload);
387
+ }
388
+
389
+ // -----------------------------------------------------------------------
390
+ // Worker management helpers
391
+ // -----------------------------------------------------------------------
392
+
393
+ /**
394
+ * Stop the worker for a plugin if one is running.
395
+ * This is a best-effort operation — if no worker manager is configured
396
+ * or no worker is running, it silently succeeds.
397
+ */
398
+ async function stopWorkerIfRunning(
399
+ pluginId: string,
400
+ pluginKey: string,
401
+ ): Promise<void> {
402
+ if (!workerManager) return;
403
+ if (!workerManager.isRunning(pluginId) && !workerManager.getWorker(pluginId)) return;
404
+
405
+ try {
406
+ await workerManager.stopWorker(pluginId);
407
+ log.info({ pluginId, pluginKey }, "plugin lifecycle: worker stopped");
408
+ emitDomain("plugin.worker_stopped", { pluginId, pluginKey });
409
+ } catch (err) {
410
+ log.warn(
411
+ { pluginId, pluginKey, err: err instanceof Error ? err.message : String(err) },
412
+ "plugin lifecycle: failed to stop worker (best-effort)",
413
+ );
414
+ }
415
+ }
416
+
417
+ async function activateReadyPlugin(pluginId: string): Promise<void> {
418
+ const supportsRuntimeActivation =
419
+ typeof pluginLoaderInstance.hasRuntimeServices === "function"
420
+ && typeof pluginLoaderInstance.loadSingle === "function";
421
+ if (!supportsRuntimeActivation || !pluginLoaderInstance.hasRuntimeServices()) {
422
+ return;
423
+ }
424
+
425
+ const loadResult = await pluginLoaderInstance.loadSingle(pluginId);
426
+ if (!loadResult.success) {
427
+ throw new Error(
428
+ loadResult.error
429
+ ?? `Failed to activate plugin ${loadResult.plugin.pluginKey}`,
430
+ );
431
+ }
432
+ }
433
+
434
+ async function deactivatePluginRuntime(
435
+ pluginId: string,
436
+ pluginKey: string,
437
+ ): Promise<void> {
438
+ const supportsRuntimeDeactivation =
439
+ typeof pluginLoaderInstance.hasRuntimeServices === "function"
440
+ && typeof pluginLoaderInstance.unloadSingle === "function";
441
+
442
+ if (supportsRuntimeDeactivation && pluginLoaderInstance.hasRuntimeServices()) {
443
+ await pluginLoaderInstance.unloadSingle(pluginId, pluginKey);
444
+ return;
445
+ }
446
+
447
+ await stopWorkerIfRunning(pluginId, pluginKey);
448
+ }
449
+
450
+ // -----------------------------------------------------------------------
451
+ // Public API
452
+ // -----------------------------------------------------------------------
453
+
454
+ return {
455
+ // -- load -------------------------------------------------------------
456
+ /**
457
+ * load — Transitions a plugin to 'ready' status and starts its worker.
458
+ *
459
+ * This method is called after a plugin has been successfully installed and
460
+ * validated. It marks the plugin as ready in the database and immediately
461
+ * triggers the plugin loader to start the worker process.
462
+ *
463
+ * @param pluginId - The UUID of the plugin to load.
464
+ * @returns The updated plugin record.
465
+ */
466
+ async load(pluginId: string): Promise<PluginRecord> {
467
+ const result = await transition(pluginId, "ready");
468
+ await activateReadyPlugin(pluginId);
469
+
470
+ emitDomain("plugin.loaded", {
471
+ pluginId,
472
+ pluginKey: result.pluginKey,
473
+ });
474
+ emitDomain("plugin.enabled", {
475
+ pluginId,
476
+ pluginKey: result.pluginKey,
477
+ });
478
+ return result;
479
+ },
480
+
481
+ // -- enable -----------------------------------------------------------
482
+ /**
483
+ * enable — Re-enables a plugin that was previously in an error or upgrade state.
484
+ *
485
+ * Similar to load(), this method transitions the plugin to 'ready' and starts
486
+ * its worker, but it specifically targets plugins that are currently disabled.
487
+ *
488
+ * @param pluginId - The UUID of the plugin to enable.
489
+ * @returns The updated plugin record.
490
+ */
491
+ async enable(pluginId: string): Promise<PluginRecord> {
492
+ const plugin = await requirePlugin(pluginId);
493
+
494
+ // Only allow enabling from disabled, error, or upgrade_pending states
495
+ if (plugin.status !== "disabled" && plugin.status !== "error" && plugin.status !== "upgrade_pending") {
496
+ throw badRequest(
497
+ `Cannot enable plugin in status '${plugin.status}'. ` +
498
+ `Plugin must be in 'disabled', 'error', or 'upgrade_pending' status to be enabled.`,
499
+ );
500
+ }
501
+
502
+ const result = await transition(pluginId, "ready", null, plugin);
503
+ await activateReadyPlugin(pluginId);
504
+ emitDomain("plugin.enabled", {
505
+ pluginId,
506
+ pluginKey: result.pluginKey,
507
+ });
508
+ return result;
509
+ },
510
+
511
+ // -- disable ----------------------------------------------------------
512
+ async disable(pluginId: string, reason?: string): Promise<PluginRecord> {
513
+ const plugin = await requirePlugin(pluginId);
514
+
515
+ // Only allow disabling from ready state
516
+ if (plugin.status !== "ready") {
517
+ throw badRequest(
518
+ `Cannot disable plugin in status '${plugin.status}'. ` +
519
+ `Plugin must be in 'ready' status to be disabled.`,
520
+ );
521
+ }
522
+
523
+ await deactivatePluginRuntime(pluginId, plugin.pluginKey);
524
+
525
+ const result = await transition(pluginId, "disabled", reason ?? null, plugin);
526
+ emitDomain("plugin.disabled", {
527
+ pluginId,
528
+ pluginKey: result.pluginKey,
529
+ reason,
530
+ });
531
+ return result;
532
+ },
533
+
534
+ // -- unload -----------------------------------------------------------
535
+ async unload(
536
+ pluginId: string,
537
+ removeData = false,
538
+ ): Promise<PluginRecord | null> {
539
+ const plugin = await requirePlugin(pluginId);
540
+
541
+ // If already uninstalled and removeData, hard-delete
542
+ if (plugin.status === "uninstalled") {
543
+ if (removeData) {
544
+ await pluginLoaderInstance.cleanupInstallArtifacts(plugin);
545
+ const deleted = await registry.uninstall(pluginId, true);
546
+ log.info(
547
+ { pluginId, pluginKey: plugin.pluginKey },
548
+ "plugin lifecycle: hard-deleted already-uninstalled plugin",
549
+ );
550
+ emitDomain("plugin.unloaded", {
551
+ pluginId,
552
+ pluginKey: plugin.pluginKey,
553
+ removeData: true,
554
+ });
555
+ return deleted as PluginRecord | null;
556
+ }
557
+ throw badRequest(
558
+ `Plugin ${plugin.pluginKey} is already uninstalled. ` +
559
+ `Use removeData=true to permanently delete it.`,
560
+ );
561
+ }
562
+
563
+ await deactivatePluginRuntime(pluginId, plugin.pluginKey);
564
+ await pluginLoaderInstance.cleanupInstallArtifacts(plugin);
565
+
566
+ // Perform the uninstall via registry (handles soft/hard delete)
567
+ const result = await registry.uninstall(pluginId, removeData);
568
+
569
+ log.info(
570
+ { pluginId, pluginKey: plugin.pluginKey, removeData },
571
+ `plugin lifecycle: ${plugin.status} → uninstalled${removeData ? " (hard delete)" : ""}`,
572
+ );
573
+
574
+ emitter.emit("plugin.status_changed", {
575
+ pluginId,
576
+ pluginKey: plugin.pluginKey,
577
+ previousStatus: plugin.status,
578
+ newStatus: "uninstalled" as PluginStatus,
579
+ });
580
+
581
+ emitDomain("plugin.unloaded", {
582
+ pluginId,
583
+ pluginKey: plugin.pluginKey,
584
+ removeData,
585
+ });
586
+
587
+ return result as PluginRecord | null;
588
+ },
589
+
590
+ // -- markError --------------------------------------------------------
591
+ async markError(pluginId: string, error: string): Promise<PluginRecord> {
592
+ // Stop the worker — the plugin is in an error state and should not
593
+ // continue running. The worker manager's auto-restart is disabled
594
+ // because we are intentionally taking the plugin offline.
595
+ const plugin = await requirePlugin(pluginId);
596
+ await deactivatePluginRuntime(pluginId, plugin.pluginKey);
597
+
598
+ const result = await transition(pluginId, "error", error, plugin);
599
+ emitDomain("plugin.error", {
600
+ pluginId,
601
+ pluginKey: result.pluginKey,
602
+ error,
603
+ });
604
+ return result;
605
+ },
606
+
607
+ // -- markUpgradePending -----------------------------------------------
608
+ async markUpgradePending(pluginId: string): Promise<PluginRecord> {
609
+ const plugin = await requirePlugin(pluginId);
610
+ await deactivatePluginRuntime(pluginId, plugin.pluginKey);
611
+
612
+ const result = await transition(pluginId, "upgrade_pending", null, plugin);
613
+ emitDomain("plugin.upgrade_pending", {
614
+ pluginId,
615
+ pluginKey: result.pluginKey,
616
+ });
617
+ return result;
618
+ },
619
+
620
+ // -- upgrade ----------------------------------------------------------
621
+ /**
622
+ * Upgrade a plugin to a newer version by performing a package update and
623
+ * managing the lifecycle state transition.
624
+ *
625
+ * Following PLUGIN_SPEC.md §25.3, the upgrade process:
626
+ * 1. Stops the current worker process (if running).
627
+ * 2. Fetches and validates the new plugin package via the `PluginLoader`.
628
+ * 3. Compares the capabilities declared in the new manifest against the old one.
629
+ * 4. If new capabilities are added, transitions the plugin to `upgrade_pending`
630
+ * to await operator approval (worker stays stopped).
631
+ * 5. If no new capabilities are added, transitions the plugin back to `ready`
632
+ * with the updated version and manifest metadata.
633
+ *
634
+ * @param pluginId - The UUID of the plugin to upgrade.
635
+ * @param version - Optional target version specifier.
636
+ * @returns The updated `PluginRecord`.
637
+ * @throws {BadRequest} If the plugin is not in a ready or upgrade_pending state.
638
+ */
639
+ async upgrade(pluginId: string, version?: string): Promise<PluginRecord> {
640
+ const plugin = await requirePlugin(pluginId);
641
+
642
+ // Can only upgrade plugins that are ready or already in upgrade_pending
643
+ if (plugin.status !== "ready" && plugin.status !== "upgrade_pending") {
644
+ throw badRequest(
645
+ `Cannot upgrade plugin in status '${plugin.status}'. ` +
646
+ `Plugin must be in 'ready' or 'upgrade_pending' status to be upgraded.`,
647
+ );
648
+ }
649
+
650
+ log.info(
651
+ { pluginId, pluginKey: plugin.pluginKey, targetVersion: version },
652
+ "plugin lifecycle: upgrade requested",
653
+ );
654
+
655
+ await deactivatePluginRuntime(pluginId, plugin.pluginKey);
656
+
657
+ // 1. Download and validate new package via loader
658
+ const { oldManifest, newManifest, discovered } =
659
+ await pluginLoaderInstance.upgradePlugin(pluginId, { version });
660
+
661
+ log.info(
662
+ {
663
+ pluginId,
664
+ pluginKey: plugin.pluginKey,
665
+ oldVersion: oldManifest.version,
666
+ newVersion: newManifest.version,
667
+ },
668
+ "plugin lifecycle: package upgraded on disk",
669
+ );
670
+
671
+ // 2. Compare capabilities
672
+ const addedCaps = newManifest.capabilities.filter(
673
+ (cap) => !oldManifest.capabilities.includes(cap),
674
+ );
675
+
676
+ // 3. Transition state
677
+ if (addedCaps.length > 0) {
678
+ // New capabilities require operator approval — worker stays stopped
679
+ log.info(
680
+ { pluginId, pluginKey: plugin.pluginKey, addedCaps },
681
+ "plugin lifecycle: new capabilities detected, transitioning to upgrade_pending",
682
+ );
683
+ // Skip the inner stopWorkerIfRunning since we already stopped above
684
+ const result = await transition(pluginId, "upgrade_pending", null, plugin);
685
+ emitDomain("plugin.upgrade_pending", {
686
+ pluginId,
687
+ pluginKey: result.pluginKey,
688
+ });
689
+ return result;
690
+ } else {
691
+ const result = await transition(pluginId, "ready", null, {
692
+ ...plugin,
693
+ version: discovered.version,
694
+ manifestJson: newManifest,
695
+ } as PluginRecord);
696
+ await activateReadyPlugin(pluginId);
697
+
698
+ emitDomain("plugin.loaded", {
699
+ pluginId,
700
+ pluginKey: result.pluginKey,
701
+ });
702
+ emitDomain("plugin.enabled", {
703
+ pluginId,
704
+ pluginKey: result.pluginKey,
705
+ });
706
+
707
+ return result;
708
+ }
709
+ },
710
+
711
+ // -- startWorker ------------------------------------------------------
712
+ async startWorker(
713
+ pluginId: string,
714
+ options: WorkerStartOptions,
715
+ ): Promise<void> {
716
+ if (!workerManager) {
717
+ throw badRequest(
718
+ "Cannot start worker: no PluginWorkerManager is configured. " +
719
+ "Provide a workerManager option when constructing the lifecycle manager.",
720
+ );
721
+ }
722
+
723
+ const plugin = await requirePlugin(pluginId);
724
+ if (plugin.status !== "ready") {
725
+ throw badRequest(
726
+ `Cannot start worker for plugin in status '${plugin.status}'. ` +
727
+ `Plugin must be in 'ready' status.`,
728
+ );
729
+ }
730
+
731
+ log.info(
732
+ { pluginId, pluginKey: plugin.pluginKey },
733
+ "plugin lifecycle: starting worker",
734
+ );
735
+
736
+ await workerManager.startWorker(pluginId, options);
737
+ emitDomain("plugin.worker_started", {
738
+ pluginId,
739
+ pluginKey: plugin.pluginKey,
740
+ });
741
+
742
+ log.info(
743
+ { pluginId, pluginKey: plugin.pluginKey },
744
+ "plugin lifecycle: worker started",
745
+ );
746
+ },
747
+
748
+ // -- stopWorker -------------------------------------------------------
749
+ async stopWorker(pluginId: string): Promise<void> {
750
+ if (!workerManager) return; // No worker manager — nothing to stop
751
+
752
+ const plugin = await requirePlugin(pluginId);
753
+ await stopWorkerIfRunning(pluginId, plugin.pluginKey);
754
+ },
755
+
756
+ // -- restartWorker ----------------------------------------------------
757
+ async restartWorker(pluginId: string): Promise<void> {
758
+ if (!workerManager) {
759
+ throw badRequest(
760
+ "Cannot restart worker: no PluginWorkerManager is configured.",
761
+ );
762
+ }
763
+
764
+ const plugin = await requirePlugin(pluginId);
765
+ if (plugin.status !== "ready") {
766
+ throw badRequest(
767
+ `Cannot restart worker for plugin in status '${plugin.status}'. ` +
768
+ `Plugin must be in 'ready' status.`,
769
+ );
770
+ }
771
+
772
+ const handle = workerManager.getWorker(pluginId);
773
+ if (!handle) {
774
+ throw badRequest(
775
+ `Cannot restart worker for plugin "${plugin.pluginKey}": no worker is running.`,
776
+ );
777
+ }
778
+
779
+ log.info(
780
+ { pluginId, pluginKey: plugin.pluginKey },
781
+ "plugin lifecycle: restarting worker",
782
+ );
783
+
784
+ await handle.restart();
785
+
786
+ emitDomain("plugin.worker_stopped", { pluginId, pluginKey: plugin.pluginKey });
787
+ emitDomain("plugin.worker_started", { pluginId, pluginKey: plugin.pluginKey });
788
+
789
+ log.info(
790
+ { pluginId, pluginKey: plugin.pluginKey },
791
+ "plugin lifecycle: worker restarted",
792
+ );
793
+ },
794
+
795
+ // -- getStatus --------------------------------------------------------
796
+ async getStatus(pluginId: string): Promise<PluginStatus | null> {
797
+ const plugin = await registry.getById(pluginId);
798
+ return plugin?.status ?? null;
799
+ },
800
+
801
+ // -- canTransition ----------------------------------------------------
802
+ async canTransition(pluginId: string, to: PluginStatus): Promise<boolean> {
803
+ const plugin = await registry.getById(pluginId);
804
+ if (!plugin) return false;
805
+ return isValidTransition(plugin.status, to);
806
+ },
807
+
808
+ // -- Event subscriptions ----------------------------------------------
809
+ on(event, listener) {
810
+ emitter.on(event, listener);
811
+ },
812
+
813
+ off(event, listener) {
814
+ emitter.off(event, listener);
815
+ },
816
+
817
+ once(event, listener) {
818
+ emitter.once(event, listener);
819
+ },
820
+ };
821
+ }
822
+