slackhive 0.1.37 → 0.1.39

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 (542) hide show
  1. package/.dockerignore +14 -0
  2. package/.env.example +44 -0
  3. package/.github/ISSUE_TEMPLATE/bug_report.yml +65 -0
  4. package/.github/ISSUE_TEMPLATE/config.yml +5 -0
  5. package/.github/ISSUE_TEMPLATE/feature_request.yml +38 -0
  6. package/.github/PULL_REQUEST_TEMPLATE.md +27 -0
  7. package/.github/dependabot.yml +20 -0
  8. package/.github/workflows/audit.yml +149 -0
  9. package/.github/workflows/ci.yml +135 -0
  10. package/CHANGELOG.md +52 -0
  11. package/CODE_OF_CONDUCT.md +37 -0
  12. package/CONTRIBUTING.md +204 -0
  13. package/LICENSE +21 -0
  14. package/README.md +19 -0
  15. package/SECURITY.md +47 -0
  16. package/apps/runner/Dockerfile +33 -0
  17. package/apps/runner/dist/__tests__/channel-restrictions.test.d.ts +8 -0
  18. package/apps/runner/dist/__tests__/channel-restrictions.test.js +63 -0
  19. package/apps/runner/dist/__tests__/channel-restrictions.test.js.map +1 -0
  20. package/apps/runner/dist/__tests__/claude-handler-resolve.test.d.ts +20 -0
  21. package/apps/runner/dist/__tests__/claude-handler-resolve.test.js +178 -0
  22. package/apps/runner/dist/__tests__/claude-handler-resolve.test.js.map +1 -0
  23. package/apps/runner/dist/__tests__/compile-claude-md.test.d.ts +13 -0
  24. package/apps/runner/dist/__tests__/compile-claude-md.test.js +144 -0
  25. package/apps/runner/dist/__tests__/compile-claude-md.test.js.map +1 -0
  26. package/apps/runner/dist/__tests__/memory-sync.test.d.ts +11 -0
  27. package/apps/runner/dist/__tests__/memory-sync.test.js +56 -0
  28. package/apps/runner/dist/__tests__/memory-sync.test.js.map +1 -0
  29. package/apps/runner/dist/__tests__/slack-file-support.test.d.ts +9 -0
  30. package/apps/runner/dist/__tests__/slack-file-support.test.js +271 -0
  31. package/apps/runner/dist/__tests__/slack-file-support.test.js.map +1 -0
  32. package/apps/runner/dist/__tests__/slack-formatting.test.d.ts +12 -0
  33. package/apps/runner/dist/__tests__/slack-formatting.test.js +400 -0
  34. package/apps/runner/dist/__tests__/slack-formatting.test.js.map +1 -0
  35. package/apps/runner/dist/__tests__/thread-context.test.d.ts +12 -0
  36. package/apps/runner/dist/__tests__/thread-context.test.js +182 -0
  37. package/apps/runner/dist/__tests__/thread-context.test.js.map +1 -0
  38. package/apps/runner/dist/agent-runner.d.ts +118 -0
  39. package/apps/runner/dist/agent-runner.js +352 -0
  40. package/apps/runner/dist/agent-runner.js.map +1 -0
  41. package/apps/runner/dist/claude-handler.d.ts +122 -0
  42. package/apps/runner/dist/claude-handler.js +402 -0
  43. package/apps/runner/dist/claude-handler.js.map +1 -0
  44. package/apps/runner/dist/compile-claude-md.d.ts +59 -0
  45. package/apps/runner/dist/compile-claude-md.js +291 -0
  46. package/apps/runner/dist/compile-claude-md.js.map +1 -0
  47. package/apps/runner/dist/correction-handler.d.ts +46 -0
  48. package/apps/runner/dist/correction-handler.js +162 -0
  49. package/apps/runner/dist/correction-handler.js.map +1 -0
  50. package/apps/runner/dist/correction-manager.d.ts +53 -0
  51. package/apps/runner/dist/correction-manager.js +241 -0
  52. package/apps/runner/dist/correction-manager.js.map +1 -0
  53. package/apps/runner/dist/db.d.ts +193 -0
  54. package/apps/runner/dist/db.js +492 -0
  55. package/apps/runner/dist/db.js.map +1 -0
  56. package/apps/runner/dist/index.d.ts +9 -0
  57. package/apps/runner/dist/index.js +43 -0
  58. package/apps/runner/dist/index.js.map +1 -0
  59. package/apps/runner/dist/job-scheduler.d.ts +57 -0
  60. package/apps/runner/dist/job-scheduler.js +150 -0
  61. package/apps/runner/dist/job-scheduler.js.map +1 -0
  62. package/apps/runner/dist/logger.d.ts +32 -0
  63. package/apps/runner/dist/logger.js +52 -0
  64. package/apps/runner/dist/logger.js.map +1 -0
  65. package/apps/runner/dist/mcp-process-manager.d.ts +38 -0
  66. package/apps/runner/dist/mcp-process-manager.js +189 -0
  67. package/apps/runner/dist/mcp-process-manager.js.map +1 -0
  68. package/apps/runner/dist/memory-mcp.d.ts +14 -0
  69. package/apps/runner/dist/memory-mcp.js +88 -0
  70. package/apps/runner/dist/memory-mcp.js.map +1 -0
  71. package/apps/runner/dist/memory-watcher.d.ts +78 -0
  72. package/apps/runner/dist/memory-watcher.js +220 -0
  73. package/apps/runner/dist/memory-watcher.js.map +1 -0
  74. package/apps/runner/dist/slack-handler.d.ts +120 -0
  75. package/apps/runner/dist/slack-handler.js +843 -0
  76. package/apps/runner/dist/slack-handler.js.map +1 -0
  77. package/apps/runner/node_modules/.vite/vitest/da39a3ee5e6b4b0d3255bfef95601890afd80709/results.json +1 -0
  78. package/apps/runner/package.json +42 -0
  79. package/apps/runner/src/__tests__/channel-restrictions.test.ts +75 -0
  80. package/apps/runner/src/__tests__/claude-handler-resolve.test.ts +160 -0
  81. package/apps/runner/src/__tests__/compile-claude-md.test.ts +139 -0
  82. package/apps/runner/src/__tests__/memory-sync.test.ts +59 -0
  83. package/apps/runner/src/__tests__/slack-file-support.test.ts +376 -0
  84. package/apps/runner/src/__tests__/slack-formatting.test.ts +495 -0
  85. package/apps/runner/src/__tests__/thread-context.test.ts +215 -0
  86. package/apps/runner/src/agent-runner.ts +397 -0
  87. package/apps/runner/src/claude-handler.ts +475 -0
  88. package/apps/runner/src/compile-claude-md.ts +283 -0
  89. package/apps/runner/src/correction-handler.ts +191 -0
  90. package/apps/runner/src/correction-manager.ts +285 -0
  91. package/apps/runner/src/db.ts +604 -0
  92. package/apps/runner/src/index.ts +46 -0
  93. package/apps/runner/src/job-scheduler.ts +165 -0
  94. package/apps/runner/src/logger.ts +49 -0
  95. package/apps/runner/src/mcp-process-manager.ts +195 -0
  96. package/apps/runner/src/memory-mcp.ts +85 -0
  97. package/apps/runner/src/memory-watcher.ts +215 -0
  98. package/apps/runner/src/slack-handler.ts +929 -0
  99. package/apps/runner/tsconfig.json +17 -0
  100. package/apps/runner/vitest.config.mts +17 -0
  101. package/apps/web/.eslintrc.json +3 -0
  102. package/apps/web/.next/app-build-manifest.json +323 -0
  103. package/apps/web/.next/app-path-routes-manifest.json +46 -0
  104. package/apps/web/.next/build-manifest.json +33 -0
  105. package/apps/web/.next/cache/.previewinfo +1 -0
  106. package/apps/web/.next/cache/.rscinfo +1 -0
  107. package/apps/web/.next/cache/webpack/client-production/0.pack +0 -0
  108. package/apps/web/.next/cache/webpack/client-production/1.pack +0 -0
  109. package/apps/web/.next/cache/webpack/client-production/2.pack +0 -0
  110. package/apps/web/.next/cache/webpack/client-production/3.pack +0 -0
  111. package/apps/web/.next/cache/webpack/client-production/4.pack +0 -0
  112. package/apps/web/.next/cache/webpack/client-production/index.pack +0 -0
  113. package/apps/web/.next/cache/webpack/client-production/index.pack.old +0 -0
  114. package/apps/web/.next/cache/webpack/edge-server-production/0.pack +0 -0
  115. package/apps/web/.next/cache/webpack/edge-server-production/1.pack +0 -0
  116. package/apps/web/.next/cache/webpack/edge-server-production/index.pack +0 -0
  117. package/apps/web/.next/cache/webpack/edge-server-production/index.pack.old +0 -0
  118. package/apps/web/.next/cache/webpack/server-production/0.pack +0 -0
  119. package/apps/web/.next/cache/webpack/server-production/1.pack +0 -0
  120. package/apps/web/.next/cache/webpack/server-production/2.pack +0 -0
  121. package/apps/web/.next/cache/webpack/server-production/index.pack +0 -0
  122. package/apps/web/.next/cache/webpack/server-production/index.pack.old +0 -0
  123. package/apps/web/.next/diagnostics/build-diagnostics.json +6 -0
  124. package/apps/web/.next/diagnostics/framework.json +1 -0
  125. package/apps/web/.next/package.json +1 -0
  126. package/apps/web/.next/react-loadable-manifest.json +1 -0
  127. package/apps/web/.next/server/app/_not-found/page.js +2 -0
  128. package/apps/web/.next/server/app/_not-found/page.js.nft.json +1 -0
  129. package/apps/web/.next/server/app/_not-found/page_client-reference-manifest.js +1 -0
  130. package/apps/web/.next/server/app/agents/[slug]/page.js +4 -0
  131. package/apps/web/.next/server/app/agents/[slug]/page.js.nft.json +1 -0
  132. package/apps/web/.next/server/app/agents/[slug]/page_client-reference-manifest.js +1 -0
  133. package/apps/web/.next/server/app/agents/new/page.js +2 -0
  134. package/apps/web/.next/server/app/agents/new/page.js.nft.json +1 -0
  135. package/apps/web/.next/server/app/agents/new/page_client-reference-manifest.js +1 -0
  136. package/apps/web/.next/server/app/api/agents/[id]/access/route.js +1 -0
  137. package/apps/web/.next/server/app/api/agents/[id]/access/route.js.nft.json +1 -0
  138. package/apps/web/.next/server/app/api/agents/[id]/access/route_client-reference-manifest.js +1 -0
  139. package/apps/web/.next/server/app/api/agents/[id]/claude-md/route.js +6 -0
  140. package/apps/web/.next/server/app/api/agents/[id]/claude-md/route.js.nft.json +1 -0
  141. package/apps/web/.next/server/app/api/agents/[id]/claude-md/route_client-reference-manifest.js +1 -0
  142. package/apps/web/.next/server/app/api/agents/[id]/logs/route.js +3 -0
  143. package/apps/web/.next/server/app/api/agents/[id]/logs/route.js.nft.json +1 -0
  144. package/apps/web/.next/server/app/api/agents/[id]/logs/route_client-reference-manifest.js +1 -0
  145. package/apps/web/.next/server/app/api/agents/[id]/manifest/route.js +1 -0
  146. package/apps/web/.next/server/app/api/agents/[id]/manifest/route.js.nft.json +1 -0
  147. package/apps/web/.next/server/app/api/agents/[id]/manifest/route_client-reference-manifest.js +1 -0
  148. package/apps/web/.next/server/app/api/agents/[id]/mcps/route.js +1 -0
  149. package/apps/web/.next/server/app/api/agents/[id]/mcps/route.js.nft.json +1 -0
  150. package/apps/web/.next/server/app/api/agents/[id]/mcps/route_client-reference-manifest.js +1 -0
  151. package/apps/web/.next/server/app/api/agents/[id]/memories/[memId]/route.js +1 -0
  152. package/apps/web/.next/server/app/api/agents/[id]/memories/[memId]/route.js.nft.json +1 -0
  153. package/apps/web/.next/server/app/api/agents/[id]/memories/[memId]/route_client-reference-manifest.js +1 -0
  154. package/apps/web/.next/server/app/api/agents/[id]/memories/route.js +1 -0
  155. package/apps/web/.next/server/app/api/agents/[id]/memories/route.js.nft.json +1 -0
  156. package/apps/web/.next/server/app/api/agents/[id]/memories/route_client-reference-manifest.js +1 -0
  157. package/apps/web/.next/server/app/api/agents/[id]/permissions/route.js +1 -0
  158. package/apps/web/.next/server/app/api/agents/[id]/permissions/route.js.nft.json +1 -0
  159. package/apps/web/.next/server/app/api/agents/[id]/permissions/route_client-reference-manifest.js +1 -0
  160. package/apps/web/.next/server/app/api/agents/[id]/reload/route.js +1 -0
  161. package/apps/web/.next/server/app/api/agents/[id]/reload/route.js.nft.json +1 -0
  162. package/apps/web/.next/server/app/api/agents/[id]/reload/route_client-reference-manifest.js +1 -0
  163. package/apps/web/.next/server/app/api/agents/[id]/restrictions/route.js +1 -0
  164. package/apps/web/.next/server/app/api/agents/[id]/restrictions/route.js.nft.json +1 -0
  165. package/apps/web/.next/server/app/api/agents/[id]/restrictions/route_client-reference-manifest.js +1 -0
  166. package/apps/web/.next/server/app/api/agents/[id]/route.js +33 -0
  167. package/apps/web/.next/server/app/api/agents/[id]/route.js.nft.json +1 -0
  168. package/apps/web/.next/server/app/api/agents/[id]/route_client-reference-manifest.js +1 -0
  169. package/apps/web/.next/server/app/api/agents/[id]/skills/[skillId]/route.js +1 -0
  170. package/apps/web/.next/server/app/api/agents/[id]/skills/[skillId]/route.js.nft.json +1 -0
  171. package/apps/web/.next/server/app/api/agents/[id]/skills/[skillId]/route_client-reference-manifest.js +1 -0
  172. package/apps/web/.next/server/app/api/agents/[id]/skills/route.js +1 -0
  173. package/apps/web/.next/server/app/api/agents/[id]/skills/route.js.nft.json +1 -0
  174. package/apps/web/.next/server/app/api/agents/[id]/skills/route_client-reference-manifest.js +1 -0
  175. package/apps/web/.next/server/app/api/agents/[id]/slack-info/route.js +1 -0
  176. package/apps/web/.next/server/app/api/agents/[id]/slack-info/route.js.nft.json +1 -0
  177. package/apps/web/.next/server/app/api/agents/[id]/slack-info/route_client-reference-manifest.js +1 -0
  178. package/apps/web/.next/server/app/api/agents/[id]/snapshots/[sid]/restore/route.js +1 -0
  179. package/apps/web/.next/server/app/api/agents/[id]/snapshots/[sid]/restore/route.js.nft.json +1 -0
  180. package/apps/web/.next/server/app/api/agents/[id]/snapshots/[sid]/restore/route_client-reference-manifest.js +1 -0
  181. package/apps/web/.next/server/app/api/agents/[id]/snapshots/[sid]/route.js +1 -0
  182. package/apps/web/.next/server/app/api/agents/[id]/snapshots/[sid]/route.js.nft.json +1 -0
  183. package/apps/web/.next/server/app/api/agents/[id]/snapshots/[sid]/route_client-reference-manifest.js +1 -0
  184. package/apps/web/.next/server/app/api/agents/[id]/snapshots/route.js +1 -0
  185. package/apps/web/.next/server/app/api/agents/[id]/snapshots/route.js.nft.json +1 -0
  186. package/apps/web/.next/server/app/api/agents/[id]/snapshots/route_client-reference-manifest.js +1 -0
  187. package/apps/web/.next/server/app/api/agents/[id]/start/route.js +1 -0
  188. package/apps/web/.next/server/app/api/agents/[id]/start/route.js.nft.json +1 -0
  189. package/apps/web/.next/server/app/api/agents/[id]/start/route_client-reference-manifest.js +1 -0
  190. package/apps/web/.next/server/app/api/agents/[id]/stop/route.js +1 -0
  191. package/apps/web/.next/server/app/api/agents/[id]/stop/route.js.nft.json +1 -0
  192. package/apps/web/.next/server/app/api/agents/[id]/stop/route_client-reference-manifest.js +1 -0
  193. package/apps/web/.next/server/app/api/agents/route.js +91 -0
  194. package/apps/web/.next/server/app/api/agents/route.js.nft.json +1 -0
  195. package/apps/web/.next/server/app/api/agents/route_client-reference-manifest.js +1 -0
  196. package/apps/web/.next/server/app/api/auth/login/route.js +1 -0
  197. package/apps/web/.next/server/app/api/auth/login/route.js.nft.json +1 -0
  198. package/apps/web/.next/server/app/api/auth/login/route_client-reference-manifest.js +1 -0
  199. package/apps/web/.next/server/app/api/auth/logout/route.js +1 -0
  200. package/apps/web/.next/server/app/api/auth/logout/route.js.nft.json +1 -0
  201. package/apps/web/.next/server/app/api/auth/logout/route_client-reference-manifest.js +1 -0
  202. package/apps/web/.next/server/app/api/auth/me/route.js +1 -0
  203. package/apps/web/.next/server/app/api/auth/me/route.js.nft.json +1 -0
  204. package/apps/web/.next/server/app/api/auth/me/route_client-reference-manifest.js +1 -0
  205. package/apps/web/.next/server/app/api/auth/users/[id]/route.js +1 -0
  206. package/apps/web/.next/server/app/api/auth/users/[id]/route.js.nft.json +1 -0
  207. package/apps/web/.next/server/app/api/auth/users/[id]/route_client-reference-manifest.js +1 -0
  208. package/apps/web/.next/server/app/api/auth/users/route.js +1 -0
  209. package/apps/web/.next/server/app/api/auth/users/route.js.nft.json +1 -0
  210. package/apps/web/.next/server/app/api/auth/users/route_client-reference-manifest.js +1 -0
  211. package/apps/web/.next/server/app/api/env-vars/[key]/route.js +1 -0
  212. package/apps/web/.next/server/app/api/env-vars/[key]/route.js.nft.json +1 -0
  213. package/apps/web/.next/server/app/api/env-vars/[key]/route_client-reference-manifest.js +1 -0
  214. package/apps/web/.next/server/app/api/env-vars/route.js +1 -0
  215. package/apps/web/.next/server/app/api/env-vars/route.js.nft.json +1 -0
  216. package/apps/web/.next/server/app/api/env-vars/route_client-reference-manifest.js +1 -0
  217. package/apps/web/.next/server/app/api/jobs/[id]/route.js +1 -0
  218. package/apps/web/.next/server/app/api/jobs/[id]/route.js.nft.json +1 -0
  219. package/apps/web/.next/server/app/api/jobs/[id]/route_client-reference-manifest.js +1 -0
  220. package/apps/web/.next/server/app/api/jobs/[id]/runs/route.js +1 -0
  221. package/apps/web/.next/server/app/api/jobs/[id]/runs/route.js.nft.json +1 -0
  222. package/apps/web/.next/server/app/api/jobs/[id]/runs/route_client-reference-manifest.js +1 -0
  223. package/apps/web/.next/server/app/api/jobs/route.js +1 -0
  224. package/apps/web/.next/server/app/api/jobs/route.js.nft.json +1 -0
  225. package/apps/web/.next/server/app/api/jobs/route_client-reference-manifest.js +1 -0
  226. package/apps/web/.next/server/app/api/mcps/[id]/route.js +1 -0
  227. package/apps/web/.next/server/app/api/mcps/[id]/route.js.nft.json +1 -0
  228. package/apps/web/.next/server/app/api/mcps/[id]/route_client-reference-manifest.js +1 -0
  229. package/apps/web/.next/server/app/api/mcps/[id]/test/route.js +1 -0
  230. package/apps/web/.next/server/app/api/mcps/[id]/test/route.js.nft.json +1 -0
  231. package/apps/web/.next/server/app/api/mcps/[id]/test/route_client-reference-manifest.js +1 -0
  232. package/apps/web/.next/server/app/api/mcps/route.js +1 -0
  233. package/apps/web/.next/server/app/api/mcps/route.js.nft.json +1 -0
  234. package/apps/web/.next/server/app/api/mcps/route_client-reference-manifest.js +1 -0
  235. package/apps/web/.next/server/app/api/settings/route.js +1 -0
  236. package/apps/web/.next/server/app/api/settings/route.js.nft.json +1 -0
  237. package/apps/web/.next/server/app/api/settings/route_client-reference-manifest.js +1 -0
  238. package/apps/web/.next/server/app/icon.svg/route.js +1 -0
  239. package/apps/web/.next/server/app/icon.svg/route.js.nft.json +1 -0
  240. package/apps/web/.next/server/app/jobs/page.js +2 -0
  241. package/apps/web/.next/server/app/jobs/page.js.nft.json +1 -0
  242. package/apps/web/.next/server/app/jobs/page_client-reference-manifest.js +1 -0
  243. package/apps/web/.next/server/app/login/page.js +2 -0
  244. package/apps/web/.next/server/app/login/page.js.nft.json +1 -0
  245. package/apps/web/.next/server/app/login/page_client-reference-manifest.js +1 -0
  246. package/apps/web/.next/server/app/page.js +2 -0
  247. package/apps/web/.next/server/app/page.js.nft.json +1 -0
  248. package/apps/web/.next/server/app/page_client-reference-manifest.js +1 -0
  249. package/apps/web/.next/server/app/settings/env-vars/page.js +2 -0
  250. package/apps/web/.next/server/app/settings/env-vars/page.js.nft.json +1 -0
  251. package/apps/web/.next/server/app/settings/env-vars/page_client-reference-manifest.js +1 -0
  252. package/apps/web/.next/server/app/settings/mcps/page.js +2 -0
  253. package/apps/web/.next/server/app/settings/mcps/page.js.nft.json +1 -0
  254. package/apps/web/.next/server/app/settings/mcps/page_client-reference-manifest.js +1 -0
  255. package/apps/web/.next/server/app/settings/page.js +2 -0
  256. package/apps/web/.next/server/app/settings/page.js.nft.json +1 -0
  257. package/apps/web/.next/server/app/settings/page_client-reference-manifest.js +1 -0
  258. package/apps/web/.next/server/app-paths-manifest.json +46 -0
  259. package/apps/web/.next/server/chunks/1157.js +9 -0
  260. package/apps/web/.next/server/chunks/2287.js +1 -0
  261. package/apps/web/.next/server/chunks/3444.js +1 -0
  262. package/apps/web/.next/server/chunks/383.js +6 -0
  263. package/apps/web/.next/server/chunks/4012.js +58 -0
  264. package/apps/web/.next/server/chunks/6791.js +1 -0
  265. package/apps/web/.next/server/chunks/7171.js +1 -0
  266. package/apps/web/.next/server/chunks/8819.js +22 -0
  267. package/apps/web/.next/server/edge-runtime-webpack.js +2 -0
  268. package/apps/web/.next/server/edge-runtime-webpack.js.map +1 -0
  269. package/apps/web/.next/server/interception-route-rewrite-manifest.js +1 -0
  270. package/apps/web/.next/server/middleware-build-manifest.js +1 -0
  271. package/apps/web/.next/server/middleware-manifest.json +32 -0
  272. package/apps/web/.next/server/middleware-react-loadable-manifest.js +1 -0
  273. package/apps/web/.next/server/next-font-manifest.js +1 -0
  274. package/apps/web/.next/server/next-font-manifest.json +1 -0
  275. package/apps/web/.next/server/pages/_app.js +1 -0
  276. package/apps/web/.next/server/pages/_app.js.nft.json +1 -0
  277. package/apps/web/.next/server/pages/_document.js +1 -0
  278. package/apps/web/.next/server/pages/_document.js.nft.json +1 -0
  279. package/apps/web/.next/server/pages/_error.js +19 -0
  280. package/apps/web/.next/server/pages/_error.js.nft.json +1 -0
  281. package/apps/web/.next/server/pages-manifest.json +5 -0
  282. package/apps/web/.next/server/server-reference-manifest.js +1 -0
  283. package/apps/web/.next/server/server-reference-manifest.json +1 -0
  284. package/apps/web/.next/server/src/middleware.js +14 -0
  285. package/apps/web/.next/server/src/middleware.js.map +1 -0
  286. package/apps/web/.next/server/webpack-runtime.js +1 -0
  287. package/apps/web/.next/static/chunks/18-90b700ea37b686a2.js +1 -0
  288. package/apps/web/.next/static/chunks/87c73c54-24122e7b92478d00.js +1 -0
  289. package/apps/web/.next/static/chunks/9664-af80478aa73ba424.js +1 -0
  290. package/apps/web/.next/static/chunks/app/_not-found/page-b9cee17ed89ca24a.js +1 -0
  291. package/apps/web/.next/static/chunks/app/agents/[slug]/page-18369fc3fe1a9a7b.js +1 -0
  292. package/apps/web/.next/static/chunks/app/agents/new/page-bf11cf8901c7e2cd.js +1 -0
  293. package/apps/web/.next/static/chunks/app/api/agents/[id]/access/route-07f0f73ac9839899.js +1 -0
  294. package/apps/web/.next/static/chunks/app/api/agents/[id]/claude-md/route-07f0f73ac9839899.js +1 -0
  295. package/apps/web/.next/static/chunks/app/api/agents/[id]/logs/route-07f0f73ac9839899.js +1 -0
  296. package/apps/web/.next/static/chunks/app/api/agents/[id]/manifest/route-07f0f73ac9839899.js +1 -0
  297. package/apps/web/.next/static/chunks/app/api/agents/[id]/mcps/route-07f0f73ac9839899.js +1 -0
  298. package/apps/web/.next/static/chunks/app/api/agents/[id]/memories/[memId]/route-07f0f73ac9839899.js +1 -0
  299. package/apps/web/.next/static/chunks/app/api/agents/[id]/memories/route-07f0f73ac9839899.js +1 -0
  300. package/apps/web/.next/static/chunks/app/api/agents/[id]/permissions/route-07f0f73ac9839899.js +1 -0
  301. package/apps/web/.next/static/chunks/app/api/agents/[id]/reload/route-07f0f73ac9839899.js +1 -0
  302. package/apps/web/.next/static/chunks/app/api/agents/[id]/restrictions/route-07f0f73ac9839899.js +1 -0
  303. package/apps/web/.next/static/chunks/app/api/agents/[id]/route-07f0f73ac9839899.js +1 -0
  304. package/apps/web/.next/static/chunks/app/api/agents/[id]/skills/[skillId]/route-07f0f73ac9839899.js +1 -0
  305. package/apps/web/.next/static/chunks/app/api/agents/[id]/skills/route-07f0f73ac9839899.js +1 -0
  306. package/apps/web/.next/static/chunks/app/api/agents/[id]/slack-info/route-07f0f73ac9839899.js +1 -0
  307. package/apps/web/.next/static/chunks/app/api/agents/[id]/snapshots/[sid]/restore/route-07f0f73ac9839899.js +1 -0
  308. package/apps/web/.next/static/chunks/app/api/agents/[id]/snapshots/[sid]/route-07f0f73ac9839899.js +1 -0
  309. package/apps/web/.next/static/chunks/app/api/agents/[id]/snapshots/route-07f0f73ac9839899.js +1 -0
  310. package/apps/web/.next/static/chunks/app/api/agents/[id]/start/route-07f0f73ac9839899.js +1 -0
  311. package/apps/web/.next/static/chunks/app/api/agents/[id]/stop/route-07f0f73ac9839899.js +1 -0
  312. package/apps/web/.next/static/chunks/app/api/agents/route-07f0f73ac9839899.js +1 -0
  313. package/apps/web/.next/static/chunks/app/api/auth/login/route-07f0f73ac9839899.js +1 -0
  314. package/apps/web/.next/static/chunks/app/api/auth/logout/route-07f0f73ac9839899.js +1 -0
  315. package/apps/web/.next/static/chunks/app/api/auth/me/route-07f0f73ac9839899.js +1 -0
  316. package/apps/web/.next/static/chunks/app/api/auth/users/[id]/route-07f0f73ac9839899.js +1 -0
  317. package/apps/web/.next/static/chunks/app/api/auth/users/route-07f0f73ac9839899.js +1 -0
  318. package/apps/web/.next/static/chunks/app/api/env-vars/[key]/route-07f0f73ac9839899.js +1 -0
  319. package/apps/web/.next/static/chunks/app/api/env-vars/route-07f0f73ac9839899.js +1 -0
  320. package/apps/web/.next/static/chunks/app/api/jobs/[id]/route-07f0f73ac9839899.js +1 -0
  321. package/apps/web/.next/static/chunks/app/api/jobs/[id]/runs/route-07f0f73ac9839899.js +1 -0
  322. package/apps/web/.next/static/chunks/app/api/jobs/route-07f0f73ac9839899.js +1 -0
  323. package/apps/web/.next/static/chunks/app/api/mcps/[id]/route-07f0f73ac9839899.js +1 -0
  324. package/apps/web/.next/static/chunks/app/api/mcps/[id]/test/route-07f0f73ac9839899.js +1 -0
  325. package/apps/web/.next/static/chunks/app/api/mcps/route-07f0f73ac9839899.js +1 -0
  326. package/apps/web/.next/static/chunks/app/api/settings/route-07f0f73ac9839899.js +1 -0
  327. package/apps/web/.next/static/chunks/app/jobs/page-f5aa89a47c50efd8.js +1 -0
  328. package/apps/web/.next/static/chunks/app/layout-2079f4964aa7314e.js +1 -0
  329. package/apps/web/.next/static/chunks/app/login/layout-07f0f73ac9839899.js +1 -0
  330. package/apps/web/.next/static/chunks/app/login/page-aa259283dc38e8f9.js +1 -0
  331. package/apps/web/.next/static/chunks/app/page-e83437b608104dff.js +1 -0
  332. package/apps/web/.next/static/chunks/app/settings/env-vars/page-06479dbdfb78b76b.js +1 -0
  333. package/apps/web/.next/static/chunks/app/settings/mcps/page-75650686ed6490c7.js +1 -0
  334. package/apps/web/.next/static/chunks/app/settings/page-e1e62fc41ff6cddd.js +1 -0
  335. package/apps/web/.next/static/chunks/framework-811407f832a33072.js +1 -0
  336. package/apps/web/.next/static/chunks/main-3f1cddbdd67b1546.js +1 -0
  337. package/apps/web/.next/static/chunks/main-app-cebd8a6a5ccbf72d.js +1 -0
  338. package/apps/web/.next/static/chunks/pages/_app-50fa07b56b2d29ac.js +1 -0
  339. package/apps/web/.next/static/chunks/pages/_error-fed8688bdd23f211.js +1 -0
  340. package/apps/web/.next/static/chunks/polyfills-42372ed130431b0a.js +1 -0
  341. package/apps/web/.next/static/chunks/webpack-6c05566dba553c97.js +1 -0
  342. package/apps/web/.next/static/css/15371687405525e2.css +5 -0
  343. package/apps/web/.next/static/ikfNbLhuw7jntn35bz0lk/_buildManifest.js +1 -0
  344. package/apps/web/.next/static/ikfNbLhuw7jntn35bz0lk/_ssgManifest.js +1 -0
  345. package/apps/web/.next/trace +5 -0
  346. package/apps/web/.next/types/app/agents/[slug]/page.ts +84 -0
  347. package/apps/web/.next/types/app/agents/new/page.ts +84 -0
  348. package/apps/web/.next/types/app/api/agents/[id]/access/route.ts +347 -0
  349. package/apps/web/.next/types/app/api/agents/[id]/claude-md/route.ts +347 -0
  350. package/apps/web/.next/types/app/api/agents/[id]/logs/route.ts +347 -0
  351. package/apps/web/.next/types/app/api/agents/[id]/manifest/route.ts +347 -0
  352. package/apps/web/.next/types/app/api/agents/[id]/mcps/route.ts +347 -0
  353. package/apps/web/.next/types/app/api/agents/[id]/memories/[memId]/route.ts +347 -0
  354. package/apps/web/.next/types/app/api/agents/[id]/memories/route.ts +347 -0
  355. package/apps/web/.next/types/app/api/agents/[id]/permissions/route.ts +347 -0
  356. package/apps/web/.next/types/app/api/agents/[id]/reload/route.ts +347 -0
  357. package/apps/web/.next/types/app/api/agents/[id]/restrictions/route.ts +347 -0
  358. package/apps/web/.next/types/app/api/agents/[id]/route.ts +347 -0
  359. package/apps/web/.next/types/app/api/agents/[id]/skills/[skillId]/route.ts +347 -0
  360. package/apps/web/.next/types/app/api/agents/[id]/skills/route.ts +347 -0
  361. package/apps/web/.next/types/app/api/agents/[id]/slack-info/route.ts +347 -0
  362. package/apps/web/.next/types/app/api/agents/[id]/snapshots/[sid]/restore/route.ts +347 -0
  363. package/apps/web/.next/types/app/api/agents/[id]/snapshots/[sid]/route.ts +347 -0
  364. package/apps/web/.next/types/app/api/agents/[id]/snapshots/route.ts +347 -0
  365. package/apps/web/.next/types/app/api/agents/[id]/start/route.ts +347 -0
  366. package/apps/web/.next/types/app/api/agents/[id]/stop/route.ts +347 -0
  367. package/apps/web/.next/types/app/api/agents/route.ts +347 -0
  368. package/apps/web/.next/types/app/api/auth/login/route.ts +347 -0
  369. package/apps/web/.next/types/app/api/auth/logout/route.ts +347 -0
  370. package/apps/web/.next/types/app/api/auth/me/route.ts +347 -0
  371. package/apps/web/.next/types/app/api/auth/users/[id]/route.ts +347 -0
  372. package/apps/web/.next/types/app/api/auth/users/route.ts +347 -0
  373. package/apps/web/.next/types/app/api/env-vars/[key]/route.ts +347 -0
  374. package/apps/web/.next/types/app/api/env-vars/route.ts +347 -0
  375. package/apps/web/.next/types/app/api/jobs/[id]/route.ts +347 -0
  376. package/apps/web/.next/types/app/api/jobs/[id]/runs/route.ts +347 -0
  377. package/apps/web/.next/types/app/api/jobs/route.ts +347 -0
  378. package/apps/web/.next/types/app/api/mcps/[id]/route.ts +347 -0
  379. package/apps/web/.next/types/app/api/mcps/[id]/test/route.ts +347 -0
  380. package/apps/web/.next/types/app/api/mcps/route.ts +347 -0
  381. package/apps/web/.next/types/app/api/settings/route.ts +347 -0
  382. package/apps/web/.next/types/app/jobs/page.ts +84 -0
  383. package/apps/web/.next/types/app/login/layout.ts +84 -0
  384. package/apps/web/.next/types/app/login/page.ts +84 -0
  385. package/apps/web/.next/types/app/page.ts +84 -0
  386. package/apps/web/.next/types/app/settings/env-vars/page.ts +84 -0
  387. package/apps/web/.next/types/app/settings/mcps/page.ts +84 -0
  388. package/apps/web/.next/types/app/settings/page.ts +84 -0
  389. package/apps/web/.next/types/cache-life.d.ts +141 -0
  390. package/apps/web/.next/types/package.json +1 -0
  391. package/apps/web/.next/types/routes.d.ts +114 -0
  392. package/apps/web/.next/types/validator.ts +448 -0
  393. package/apps/web/Dockerfile +37 -0
  394. package/apps/web/next-env.d.ts +6 -0
  395. package/apps/web/next.config.js +6 -0
  396. package/apps/web/node_modules/.vite/vitest/da39a3ee5e6b4b0d3255bfef95601890afd80709/results.json +1 -0
  397. package/apps/web/package.json +48 -0
  398. package/apps/web/postcss.config.js +3 -0
  399. package/apps/web/public/logo.svg +17 -0
  400. package/apps/web/src/app/agents/[slug]/page.tsx +2235 -0
  401. package/apps/web/src/app/agents/new/page.tsx +1161 -0
  402. package/apps/web/src/app/api/agents/[id]/access/route.ts +76 -0
  403. package/apps/web/src/app/api/agents/[id]/claude-md/route.ts +111 -0
  404. package/apps/web/src/app/api/agents/[id]/logs/route.ts +84 -0
  405. package/apps/web/src/app/api/agents/[id]/manifest/route.ts +32 -0
  406. package/apps/web/src/app/api/agents/[id]/mcps/route.ts +73 -0
  407. package/apps/web/src/app/api/agents/[id]/memories/[memId]/route.ts +31 -0
  408. package/apps/web/src/app/api/agents/[id]/memories/route.ts +56 -0
  409. package/apps/web/src/app/api/agents/[id]/permissions/route.ts +74 -0
  410. package/apps/web/src/app/api/agents/[id]/reload/route.ts +33 -0
  411. package/apps/web/src/app/api/agents/[id]/restrictions/route.ts +85 -0
  412. package/apps/web/src/app/api/agents/[id]/route.ts +81 -0
  413. package/apps/web/src/app/api/agents/[id]/skills/[skillId]/route.ts +52 -0
  414. package/apps/web/src/app/api/agents/[id]/skills/route.ts +80 -0
  415. package/apps/web/src/app/api/agents/[id]/slack-info/route.ts +38 -0
  416. package/apps/web/src/app/api/agents/[id]/snapshots/[sid]/restore/route.ts +61 -0
  417. package/apps/web/src/app/api/agents/[id]/snapshots/[sid]/route.ts +53 -0
  418. package/apps/web/src/app/api/agents/[id]/snapshots/route.ts +84 -0
  419. package/apps/web/src/app/api/agents/[id]/start/route.ts +35 -0
  420. package/apps/web/src/app/api/agents/[id]/stop/route.ts +35 -0
  421. package/apps/web/src/app/api/agents/route.ts +99 -0
  422. package/apps/web/src/app/api/auth/login/route.ts +39 -0
  423. package/apps/web/src/app/api/auth/logout/route.ts +21 -0
  424. package/apps/web/src/app/api/auth/me/route.ts +24 -0
  425. package/apps/web/src/app/api/auth/users/[id]/route.ts +48 -0
  426. package/apps/web/src/app/api/auth/users/route.ts +63 -0
  427. package/apps/web/src/app/api/env-vars/[key]/route.ts +66 -0
  428. package/apps/web/src/app/api/env-vars/route.ts +59 -0
  429. package/apps/web/src/app/api/jobs/[id]/route.ts +51 -0
  430. package/apps/web/src/app/api/jobs/[id]/runs/route.ts +24 -0
  431. package/apps/web/src/app/api/jobs/route.ts +42 -0
  432. package/apps/web/src/app/api/mcps/[id]/route.ts +60 -0
  433. package/apps/web/src/app/api/mcps/[id]/test/route.ts +195 -0
  434. package/apps/web/src/app/api/mcps/route.ts +72 -0
  435. package/apps/web/src/app/api/settings/route.ts +42 -0
  436. package/apps/web/src/app/globals.css +124 -0
  437. package/apps/web/src/app/icon.svg +17 -0
  438. package/apps/web/src/app/jobs/page.tsx +543 -0
  439. package/apps/web/src/app/layout-shell.tsx +89 -0
  440. package/apps/web/src/app/layout.tsx +18 -0
  441. package/apps/web/src/app/login/layout.tsx +9 -0
  442. package/apps/web/src/app/login/page.tsx +150 -0
  443. package/apps/web/src/app/page.tsx +573 -0
  444. package/apps/web/src/app/settings/env-vars/page.tsx +216 -0
  445. package/apps/web/src/app/settings/mcps/page.tsx +763 -0
  446. package/apps/web/src/app/settings/page.tsx +528 -0
  447. package/apps/web/src/app/sidebar.tsx +345 -0
  448. package/apps/web/src/lib/__tests__/api-guard.test.ts +189 -0
  449. package/apps/web/src/lib/__tests__/auth.test.ts +262 -0
  450. package/apps/web/src/lib/__tests__/boss-registry.test.ts +323 -0
  451. package/apps/web/src/lib/__tests__/compile.test.ts +161 -0
  452. package/apps/web/src/lib/__tests__/db-agent-hierarchy.test.ts +136 -0
  453. package/apps/web/src/lib/__tests__/db-env-vars.test.ts +216 -0
  454. package/apps/web/src/lib/__tests__/db-restrictions.test.ts +117 -0
  455. package/apps/web/src/lib/__tests__/db.integration.test.ts +271 -0
  456. package/apps/web/src/lib/__tests__/diff.test.ts +102 -0
  457. package/apps/web/src/lib/__tests__/mcp-mask.test.ts +274 -0
  458. package/apps/web/src/lib/__tests__/skill-templates.test.ts +237 -0
  459. package/apps/web/src/lib/__tests__/slack-manifest.test.ts +105 -0
  460. package/apps/web/src/lib/api-guard.ts +68 -0
  461. package/apps/web/src/lib/auth-context.tsx +71 -0
  462. package/apps/web/src/lib/auth.ts +128 -0
  463. package/apps/web/src/lib/boss-registry.ts +90 -0
  464. package/apps/web/src/lib/compile.ts +51 -0
  465. package/apps/web/src/lib/db.ts +1196 -0
  466. package/apps/web/src/lib/diff.ts +43 -0
  467. package/apps/web/src/lib/mcp-mask.ts +91 -0
  468. package/apps/web/src/lib/portal.tsx +23 -0
  469. package/apps/web/src/lib/skill-templates.ts +148 -0
  470. package/apps/web/src/lib/slack-manifest.ts +85 -0
  471. package/apps/web/src/middleware.ts +68 -0
  472. package/apps/web/tailwind.config.js +6 -0
  473. package/apps/web/tsconfig.json +23 -0
  474. package/apps/web/vitest.config.mts +21 -0
  475. package/cli/.claude/settings.local.json +6 -0
  476. package/cli/README.md +281 -0
  477. package/cli/node_modules/.package-lock.json +427 -0
  478. package/cli/node_modules/commander/LICENSE +22 -0
  479. package/cli/node_modules/commander/Readme.md +1157 -0
  480. package/cli/node_modules/commander/esm.mjs +16 -0
  481. package/cli/node_modules/commander/index.js +24 -0
  482. package/cli/node_modules/commander/lib/argument.js +149 -0
  483. package/cli/node_modules/commander/lib/command.js +2509 -0
  484. package/cli/node_modules/commander/lib/error.js +39 -0
  485. package/cli/node_modules/commander/lib/help.js +520 -0
  486. package/cli/node_modules/commander/lib/option.js +330 -0
  487. package/cli/node_modules/commander/lib/suggestSimilar.js +101 -0
  488. package/cli/node_modules/commander/package-support.json +16 -0
  489. package/cli/node_modules/commander/package.json +84 -0
  490. package/cli/node_modules/commander/typings/esm.d.mts +3 -0
  491. package/cli/node_modules/commander/typings/index.d.ts +969 -0
  492. package/cli/package-lock.json +449 -0
  493. package/cli/package.json +44 -0
  494. package/cli/src/commands/init.ts +514 -0
  495. package/cli/src/commands/manage.ts +115 -0
  496. package/cli/src/index.ts +63 -0
  497. package/cli/tsconfig.json +14 -0
  498. package/docker-compose.yml +122 -0
  499. package/docs/agents/boss-agents.mdx +108 -0
  500. package/docs/agents/creating-agents.mdx +132 -0
  501. package/docs/agents/memory.mdx +113 -0
  502. package/docs/agents/tools.mdx +103 -0
  503. package/docs/configuration/env-vars.mdx +166 -0
  504. package/docs/configuration/mcp-servers.mdx +203 -0
  505. package/docs/configuration/slack-app.mdx +175 -0
  506. package/docs/docs.json +79 -0
  507. package/docs/favicon.svg +17 -0
  508. package/docs/features/history.mdx +60 -0
  509. package/docs/features/import-export.mdx +77 -0
  510. package/docs/features/logs.mdx +131 -0
  511. package/docs/features/multi-workspace.mdx +90 -0
  512. package/docs/features/scheduled-jobs.mdx +231 -0
  513. package/docs/features/users.mdx +92 -0
  514. package/docs/introduction.mdx +160 -0
  515. package/docs/logo/dark.svg +17 -0
  516. package/docs/logo/light.svg +17 -0
  517. package/docs/logo/wide-dark.svg +12 -0
  518. package/docs/logo/wide-light.svg +12 -0
  519. package/docs/quickstart.mdx +270 -0
  520. package/docs/self-hosting/docker.mdx +151 -0
  521. package/docs/self-hosting/production.mdx +176 -0
  522. package/package.json +20 -36
  523. package/packages/shared/dist/index.d.ts +8 -0
  524. package/packages/shared/dist/index.d.ts.map +1 -0
  525. package/packages/shared/dist/index.js +24 -0
  526. package/packages/shared/dist/index.js.map +1 -0
  527. package/packages/shared/dist/types.d.ts +584 -0
  528. package/packages/shared/dist/types.d.ts.map +1 -0
  529. package/packages/shared/dist/types.js +39 -0
  530. package/packages/shared/dist/types.js.map +1 -0
  531. package/packages/shared/package.json +15 -0
  532. package/packages/shared/src/db/schema.sql +354 -0
  533. package/packages/shared/src/index.ts +8 -0
  534. package/packages/shared/src/types.ts +683 -0
  535. package/packages/shared/tsconfig.json +17 -0
  536. package/scripts/dev.sh +45 -0
  537. /package/{dist → cli/dist}/commands/init.d.ts +0 -0
  538. /package/{dist → cli/dist}/commands/init.js +0 -0
  539. /package/{dist → cli/dist}/commands/manage.d.ts +0 -0
  540. /package/{dist → cli/dist}/commands/manage.js +0 -0
  541. /package/{dist → cli/dist}/index.d.ts +0 -0
  542. /package/{dist → cli/dist}/index.js +0 -0
@@ -0,0 +1,543 @@
1
+ 'use client';
2
+
3
+ /**
4
+ * @fileoverview Scheduled jobs page — create, manage, and monitor recurring tasks.
5
+ *
6
+ * Jobs are executed by the boss agent on a cron schedule.
7
+ * Results are posted to a Slack channel or DM.
8
+ *
9
+ * @module web/app/jobs
10
+ */
11
+
12
+ import React, { useEffect, useState } from 'react';
13
+ import { useAuth } from '@/lib/auth-context';
14
+ import { Portal } from '@/lib/portal';
15
+ import { Hash, MessageSquare, CalendarClock } from 'lucide-react';
16
+
17
+ interface JobRun {
18
+ id: string;
19
+ jobId: string;
20
+ startedAt: string;
21
+ finishedAt?: string;
22
+ status: 'running' | 'success' | 'error';
23
+ output?: string;
24
+ error?: string;
25
+ }
26
+
27
+ interface Job {
28
+ id: string;
29
+ agentId: string;
30
+ name: string;
31
+ prompt: string;
32
+ cronSchedule: string;
33
+ targetType: 'channel' | 'dm';
34
+ targetId: string;
35
+ enabled: boolean;
36
+ createdAt: string;
37
+ updatedAt: string;
38
+ lastRun?: JobRun;
39
+ }
40
+
41
+ const PRESETS = [
42
+ { label: 'Every hour', cron: '0 * * * *' },
43
+ { label: 'Every 6 hours', cron: '0 */6 * * *' },
44
+ { label: 'Daily at 8 AM', cron: '0 8 * * *' },
45
+ { label: 'Daily at 9 AM', cron: '0 9 * * *' },
46
+ { label: 'Mon-Fri at 9 AM', cron: '0 9 * * 1-5' },
47
+ { label: 'Weekly Mon 9 AM', cron: '0 9 * * 1' },
48
+ ];
49
+
50
+ /**
51
+ * Converts a cron expression to a human-readable string.
52
+ * Simple implementation — covers common patterns.
53
+ */
54
+ function cronToHuman(cron: string): string {
55
+ const parts = cron.split(' ');
56
+ if (parts.length !== 5) return cron;
57
+ const [min, hour, dom, mon, dow] = parts;
58
+
59
+ if (min === '0' && hour === '*') return 'Every hour';
60
+ if (min === '0' && hour.startsWith('*/')) return `Every ${hour.slice(2)} hours`;
61
+ if (min === '0' && dow === '1-5' && dom === '*' && mon === '*') return `Weekdays at ${hour}:00`;
62
+ if (min === '0' && dow === '1' && dom === '*' && mon === '*') return `Mondays at ${hour}:00`;
63
+ if (min === '0' && dow === '*' && dom === '*' && mon === '*') return `Daily at ${hour}:00`;
64
+ if (dom !== '*' && mon === '*') return `Day ${dom} at ${hour}:${min.padStart(2, '0')}`;
65
+ return cron;
66
+ }
67
+
68
+ /**
69
+ * Scheduled jobs management page.
70
+ *
71
+ * @returns {JSX.Element}
72
+ */
73
+ interface AgentOption { id: string; name: string; slug: string; isBoss: boolean; }
74
+
75
+ export default function JobsPage() {
76
+ const { canEdit } = useAuth();
77
+ const [jobs, setJobs] = useState<Job[]>([]);
78
+ const [agents, setAgents] = useState<AgentOption[]>([]);
79
+ const [loading, setLoading] = useState(true);
80
+ const [showForm, setShowForm] = useState(false);
81
+ const [editingJob, setEditingJob] = useState<Job | null>(null);
82
+ const [expandedRuns, setExpandedRuns] = useState<Set<string>>(new Set());
83
+ const [runs, setRuns] = useState<Record<string, JobRun[]>>({});
84
+
85
+ const load = () => {
86
+ setLoading(true);
87
+ Promise.all([
88
+ fetch('/api/jobs').then(r => r.json()),
89
+ fetch('/api/agents').then(r => r.json()),
90
+ ]).then(([j, a]) => { setJobs(j); setAgents(a); }).catch(() => {}).finally(() => setLoading(false));
91
+ };
92
+ useEffect(() => { load(); }, []);
93
+
94
+ const toggleRuns = async (jobId: string) => {
95
+ const next = new Set(expandedRuns);
96
+ if (next.has(jobId)) {
97
+ next.delete(jobId);
98
+ } else {
99
+ next.add(jobId);
100
+ if (!runs[jobId]) {
101
+ const r = await fetch(`/api/jobs/${jobId}/runs`);
102
+ const data = await r.json();
103
+ setRuns(prev => ({ ...prev, [jobId]: data }));
104
+ }
105
+ }
106
+ setExpandedRuns(next);
107
+ };
108
+
109
+ const toggleEnabled = async (job: Job) => {
110
+ await fetch(`/api/jobs/${job.id}`, {
111
+ method: 'PATCH',
112
+ headers: { 'Content-Type': 'application/json' },
113
+ body: JSON.stringify({ enabled: !job.enabled }),
114
+ });
115
+ load();
116
+ };
117
+
118
+ const deleteJob = async (job: Job) => {
119
+ if (!confirm(`Delete job "${job.name}"?`)) return;
120
+ await fetch(`/api/jobs/${job.id}`, { method: 'DELETE' });
121
+ load();
122
+ };
123
+
124
+ const openEdit = (job: Job) => { setEditingJob(job); setShowForm(true); };
125
+ const openCreate = () => { setEditingJob(null); setShowForm(true); };
126
+
127
+ return (
128
+ <div className="fade-up" style={{ padding: '36px 40px' }}>
129
+ {/* Header */}
130
+ <div style={{ display: 'flex', alignItems: 'center', justifyContent: 'space-between', marginBottom: 24 }}>
131
+ <div>
132
+ <h1 style={{ margin: 0, fontSize: 22, fontWeight: 700, color: 'var(--text)', letterSpacing: '-0.02em' }}>
133
+ Scheduled Jobs
134
+ </h1>
135
+ <p style={{ margin: '4px 0 0', fontSize: 13, color: 'var(--muted)' }}>
136
+ Recurring tasks executed by any agent on a cron schedule.
137
+ </p>
138
+ </div>
139
+ {canEdit && (
140
+ <button onClick={openCreate} style={{
141
+ display: 'inline-flex', alignItems: 'center', gap: 6,
142
+ background: 'var(--accent)', color: '#fff',
143
+ padding: '9px 18px', borderRadius: 8,
144
+ fontSize: 13, fontWeight: 500, border: 'none', cursor: 'pointer',
145
+ fontFamily: 'var(--font-sans)',
146
+ }}>
147
+ <svg width="13" height="13" viewBox="0 0 16 16" fill="none">
148
+ <path d="M8 3v10M3 8h10" stroke="currentColor" strokeWidth="1.6" strokeLinecap="round"/>
149
+ </svg>
150
+ New Job
151
+ </button>
152
+ )}
153
+ </div>
154
+
155
+ {/* Job list */}
156
+ {loading ? (
157
+ <p style={{ color: 'var(--muted)', fontSize: 13 }}>Loading...</p>
158
+ ) : jobs.length === 0 ? (
159
+ <div style={{
160
+ textAlign: 'center', padding: '60px 20px',
161
+ border: '1px solid var(--border)', borderRadius: 12, background: '#fff',
162
+ }}>
163
+ <CalendarClock size={32} style={{ marginBottom: 12, color: 'var(--border-2)' }} />
164
+ <p style={{ margin: '0 0 4px', fontSize: 16, fontWeight: 600, color: 'var(--text)' }}>No scheduled jobs</p>
165
+ <p style={{ margin: 0, fontSize: 13, color: 'var(--muted)', maxWidth: 300, marginInline: 'auto' }}>
166
+ {canEdit ? 'Create a recurring task for the boss agent to execute on a schedule.' : 'No jobs have been configured yet.'}
167
+ </p>
168
+ </div>
169
+ ) : (
170
+ <div style={{ display: 'flex', flexDirection: 'column', gap: 12 }}>
171
+ {jobs.map(job => (
172
+ <div key={job.id} style={{
173
+ background: '#fff', border: '1px solid var(--border)',
174
+ borderRadius: 12, boxShadow: 'var(--shadow-sm)', overflow: 'hidden',
175
+ }}>
176
+ {/* Job row */}
177
+ <div style={{ display: 'flex', alignItems: 'center', gap: 14, padding: '16px 20px' }}>
178
+ {/* Status indicator */}
179
+ <div style={{
180
+ width: 8, height: 8, borderRadius: '50%', flexShrink: 0,
181
+ background: !job.enabled ? 'var(--border-2)'
182
+ : job.lastRun?.status === 'error' ? '#dc2626'
183
+ : job.lastRun?.status === 'success' ? '#059669'
184
+ : job.lastRun?.status === 'running' ? '#2563eb'
185
+ : 'var(--border-2)',
186
+ }} />
187
+
188
+ {/* Info */}
189
+ <div style={{ flex: 1, minWidth: 0 }}>
190
+ <div style={{ display: 'flex', alignItems: 'center', gap: 8 }}>
191
+ <span style={{ fontSize: 14, fontWeight: 600, color: 'var(--text)' }}>{job.name}</span>
192
+ {!job.enabled && (
193
+ <span style={{
194
+ fontSize: 10, fontWeight: 600, color: 'var(--subtle)',
195
+ background: 'var(--surface-2)', padding: '1px 6px', borderRadius: 4,
196
+ textTransform: 'uppercase', letterSpacing: '0.04em',
197
+ }}>Paused</span>
198
+ )}
199
+ </div>
200
+ <div style={{ fontSize: 12, color: 'var(--muted)', marginTop: 2, display: 'flex', gap: 12, flexWrap: 'wrap' }}>
201
+ <span>{cronToHuman(job.cronSchedule)}</span>
202
+ {job.agentId && (() => {
203
+ const a = agents.find(x => x.id === job.agentId);
204
+ return a ? (
205
+ <span style={{ display: 'inline-flex', alignItems: 'center', gap: 4 }}>
206
+ <span style={{ width: 6, height: 6, borderRadius: '50%', background: a.isBoss ? 'var(--accent)' : 'var(--muted)', display: 'inline-block' }} />
207
+ {a.name}
208
+ </span>
209
+ ) : null;
210
+ })()}
211
+ <span style={{ display: 'inline-flex', alignItems: 'center', gap: 3 }}>
212
+ {job.targetType === 'dm' ? <MessageSquare size={11} /> : <Hash size={11} />}
213
+ {job.targetType === 'dm' ? 'DM' : 'Channel'}: <code style={{ fontFamily: 'var(--font-mono)', fontSize: 11 }}>{job.targetId}</code>
214
+ </span>
215
+ </div>
216
+ </div>
217
+
218
+ {/* Last run */}
219
+ <div style={{ textAlign: 'right', flexShrink: 0 }}>
220
+ {job.lastRun ? (
221
+ <>
222
+ <div style={{
223
+ fontSize: 11, fontWeight: 600,
224
+ color: job.lastRun.status === 'success' ? '#059669'
225
+ : job.lastRun.status === 'error' ? '#dc2626'
226
+ : '#2563eb',
227
+ textTransform: 'uppercase',
228
+ }}>{job.lastRun.status}</div>
229
+ <div style={{ fontSize: 11, color: 'var(--subtle)' }}>
230
+ {new Date(job.lastRun.startedAt).toLocaleString()}
231
+ </div>
232
+ </>
233
+ ) : (
234
+ <div style={{ fontSize: 11, color: 'var(--subtle)' }}>Never run</div>
235
+ )}
236
+ </div>
237
+
238
+ {/* Actions */}
239
+ <div style={{ display: 'flex', gap: 6, flexShrink: 0 }}>
240
+ {/* History toggle */}
241
+ <button onClick={() => toggleRuns(job.id)} title="Run history" style={{
242
+ background: 'none', border: '1px solid var(--border)', borderRadius: 6,
243
+ width: 30, height: 30, display: 'flex', alignItems: 'center', justifyContent: 'center',
244
+ cursor: 'pointer', color: 'var(--muted)', transition: 'color 0.12s, border-color 0.12s',
245
+ }}
246
+ onMouseEnter={e => { (e.currentTarget as HTMLElement).style.color = 'var(--text)'; (e.currentTarget as HTMLElement).style.borderColor = 'var(--border-2)'; }}
247
+ onMouseLeave={e => { (e.currentTarget as HTMLElement).style.color = 'var(--muted)'; (e.currentTarget as HTMLElement).style.borderColor = 'var(--border)'; }}
248
+ >
249
+ <svg width="14" height="14" viewBox="0 0 16 16" fill="none">
250
+ <path d="M2 3h12M2 7h12M2 11h8" stroke="currentColor" strokeWidth="1.3" strokeLinecap="round"/>
251
+ </svg>
252
+ </button>
253
+ {canEdit && (
254
+ <>
255
+ {/* Toggle enabled */}
256
+ <button onClick={() => toggleEnabled(job)} title={job.enabled ? 'Pause' : 'Enable'} style={{
257
+ background: 'none', border: '1px solid var(--border)', borderRadius: 6,
258
+ width: 30, height: 30, display: 'flex', alignItems: 'center', justifyContent: 'center',
259
+ cursor: 'pointer', color: job.enabled ? '#059669' : 'var(--subtle)',
260
+ transition: 'color 0.12s',
261
+ }}>
262
+ {job.enabled ? (
263
+ <svg width="14" height="14" viewBox="0 0 16 16" fill="none"><path d="M6 3v10M10 3v10" stroke="currentColor" strokeWidth="1.5" strokeLinecap="round"/></svg>
264
+ ) : (
265
+ <svg width="14" height="14" viewBox="0 0 16 16" fill="none"><path d="M4 3l9 5-9 5V3z" fill="currentColor"/></svg>
266
+ )}
267
+ </button>
268
+ <button onClick={() => openEdit(job)} title="Edit" style={{
269
+ background: 'none', border: '1px solid var(--border)', borderRadius: 6,
270
+ width: 30, height: 30, display: 'flex', alignItems: 'center', justifyContent: 'center',
271
+ cursor: 'pointer', color: 'var(--muted)',
272
+ }}>
273
+ <svg width="13" height="13" viewBox="0 0 16 16" fill="none"><path d="M11.5 1.5l3 3L5 14H2v-3L11.5 1.5z" stroke="currentColor" strokeWidth="1.3" strokeLinejoin="round"/></svg>
274
+ </button>
275
+ <button onClick={() => deleteJob(job)} title="Delete" style={{
276
+ background: 'none', border: '1px solid var(--border)', borderRadius: 6,
277
+ width: 30, height: 30, display: 'flex', alignItems: 'center', justifyContent: 'center',
278
+ cursor: 'pointer', color: '#dc2626', opacity: 0.6,
279
+ }}>
280
+ <svg width="13" height="13" viewBox="0 0 16 16" fill="none"><path d="M3 4h10M6 4V2h4v2M5 4v9h6V4" stroke="currentColor" strokeWidth="1.2" strokeLinejoin="round"/></svg>
281
+ </button>
282
+ </>
283
+ )}
284
+ </div>
285
+ </div>
286
+
287
+ {/* Prompt preview */}
288
+ <div style={{
289
+ padding: '0 20px 14px', fontSize: 12, color: 'var(--muted)',
290
+ overflow: 'hidden', textOverflow: 'ellipsis', whiteSpace: 'nowrap',
291
+ }}>
292
+ <span style={{ color: 'var(--subtle)', fontSize: 11 }}>Prompt:</span> {job.prompt}
293
+ </div>
294
+
295
+ {/* Run history (expanded) */}
296
+ {expandedRuns.has(job.id) && (
297
+ <div style={{ borderTop: '1px solid var(--border)', background: 'var(--surface-2)' }}>
298
+ <div style={{ padding: '10px 20px', fontSize: 11, fontWeight: 600, color: 'var(--muted)', letterSpacing: '0.04em', textTransform: 'uppercase' }}>
299
+ Run History
300
+ </div>
301
+ {(runs[job.id] ?? []).length === 0 ? (
302
+ <div style={{ padding: '12px 20px', fontSize: 12, color: 'var(--subtle)' }}>No runs yet</div>
303
+ ) : (
304
+ (runs[job.id] ?? []).slice(0, 10).map(run => (
305
+ <div key={run.id} style={{
306
+ display: 'flex', alignItems: 'flex-start', gap: 10,
307
+ padding: '10px 20px', borderTop: '1px solid var(--border)',
308
+ fontSize: 12,
309
+ }}>
310
+ <div style={{
311
+ width: 6, height: 6, borderRadius: '50%', marginTop: 5, flexShrink: 0,
312
+ background: run.status === 'success' ? '#059669' : run.status === 'error' ? '#dc2626' : '#2563eb',
313
+ }} />
314
+ <div style={{ flex: 1, minWidth: 0 }}>
315
+ <div style={{ display: 'flex', gap: 10, color: 'var(--muted)' }}>
316
+ <span style={{
317
+ fontWeight: 600, textTransform: 'uppercase',
318
+ color: run.status === 'success' ? '#059669' : run.status === 'error' ? '#dc2626' : '#2563eb',
319
+ }}>{run.status}</span>
320
+ <span>{new Date(run.startedAt).toLocaleString()}</span>
321
+ {run.finishedAt && (
322
+ <span style={{ color: 'var(--subtle)' }}>
323
+ ({Math.round((new Date(run.finishedAt).getTime() - new Date(run.startedAt).getTime()) / 1000)}s)
324
+ </span>
325
+ )}
326
+ </div>
327
+ {run.output && (
328
+ <pre style={{
329
+ margin: '4px 0 0', fontSize: 11, color: 'var(--text)',
330
+ whiteSpace: 'pre-wrap', wordBreak: 'break-word',
331
+ fontFamily: 'var(--font-mono)', maxHeight: 100, overflow: 'auto',
332
+ background: '#fff', padding: '6px 8px', borderRadius: 6,
333
+ border: '1px solid var(--border)',
334
+ }}>{run.output.slice(0, 500)}</pre>
335
+ )}
336
+ {run.error && (
337
+ <div style={{ margin: '4px 0 0', fontSize: 11, color: '#dc2626' }}>
338
+ {run.error}
339
+ </div>
340
+ )}
341
+ </div>
342
+ </div>
343
+ ))
344
+ )}
345
+ </div>
346
+ )}
347
+ </div>
348
+ ))}
349
+ </div>
350
+ )}
351
+
352
+ {/* Create/edit modal */}
353
+ {showForm && (
354
+ <JobFormModal
355
+ job={editingJob}
356
+ agents={agents}
357
+ onClose={() => { setShowForm(false); setEditingJob(null); }}
358
+ onSaved={() => { setShowForm(false); setEditingJob(null); load(); }}
359
+ />
360
+ )}
361
+ </div>
362
+ );
363
+ }
364
+
365
+ // =============================================================================
366
+ // Job form modal
367
+ // =============================================================================
368
+
369
+ function JobFormModal({ job, agents, onClose, onSaved }: {
370
+ job: Job | null; agents: AgentOption[]; onClose: () => void; onSaved: () => void;
371
+ }) {
372
+ const isEdit = !!job;
373
+ const [agentId, setAgentId] = useState(job?.agentId ?? agents[0]?.id ?? '');
374
+ const [name, setName] = useState(job?.name ?? '');
375
+ const [prompt, setPrompt] = useState(job?.prompt ?? '');
376
+ const [cronSchedule, setCronSchedule] = useState(job?.cronSchedule ?? '0 8 * * *');
377
+ const [targetType, setTargetType] = useState<'channel' | 'dm'>(job?.targetType ?? 'channel');
378
+ const [targetId, setTargetId] = useState(job?.targetId ?? '');
379
+ const [enabled, setEnabled] = useState(job?.enabled ?? true);
380
+ const [saving, setSaving] = useState(false);
381
+ const [error, setError] = useState('');
382
+
383
+ const save = async () => {
384
+ if (!agentId || !name || !prompt || !cronSchedule || !targetId) {
385
+ setError('All fields are required');
386
+ return;
387
+ }
388
+ setSaving(true);
389
+ setError('');
390
+ try {
391
+ const body = { agentId, name, prompt, cronSchedule, targetType, targetId, enabled };
392
+ const r = isEdit
393
+ ? await fetch(`/api/jobs/${job!.id}`, { method: 'PATCH', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(body) })
394
+ : await fetch('/api/jobs', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(body) });
395
+ if (!r.ok) {
396
+ const data = await r.json();
397
+ setError(data.error || 'Failed');
398
+ return;
399
+ }
400
+ onSaved();
401
+ } finally {
402
+ setSaving(false);
403
+ }
404
+ };
405
+
406
+ return (
407
+ <Portal>
408
+ <div style={{
409
+ position: 'fixed', top: 0, left: 0, right: 0, bottom: 0,
410
+ background: 'rgba(0,0,0,0.4)',
411
+ display: 'flex', alignItems: 'flex-start', justifyContent: 'center',
412
+ zIndex: 9999, padding: '40px 16px',
413
+ overflowY: 'auto',
414
+ backdropFilter: 'blur(2px)',
415
+ }}>
416
+ <div style={{
417
+ background: '#fff', borderRadius: 14, border: '1px solid var(--border)',
418
+ padding: 28, width: 480, maxWidth: '100%', boxShadow: 'var(--shadow-lg)',
419
+ display: 'flex', flexDirection: 'column', gap: 16,
420
+ flexShrink: 0,
421
+ }}>
422
+ <div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
423
+ <h3 style={{ margin: 0, fontSize: 16, fontWeight: 600, color: 'var(--text)' }}>
424
+ {isEdit ? 'Edit Job' : 'New Scheduled Job'}
425
+ </h3>
426
+ <button onClick={onClose} style={{ background: 'none', border: 'none', color: 'var(--muted)', fontSize: 18, cursor: 'pointer' }}>&times;</button>
427
+ </div>
428
+
429
+ {/* Agent */}
430
+ <div>
431
+ <label style={labelStyle}>Agent</label>
432
+ <select value={agentId} onChange={e => setAgentId(e.target.value)} style={inputStyle}>
433
+ {agents.length === 0 && <option value="">No agents available</option>}
434
+ {agents.map(a => (
435
+ <option key={a.id} value={a.id}>{a.name}{a.isBoss ? ' (Boss)' : ''}</option>
436
+ ))}
437
+ </select>
438
+ <p style={{ margin: '4px 0 0', fontSize: 11, color: 'var(--subtle)' }}>
439
+ The agent that will receive and execute this prompt.
440
+ </p>
441
+ </div>
442
+
443
+ {/* Name */}
444
+ <div>
445
+ <label style={labelStyle}>Name</label>
446
+ <input type="text" value={name} onChange={e => setName(e.target.value)} placeholder="e.g. Daily Booking Report"
447
+ style={inputStyle} />
448
+ </div>
449
+
450
+ {/* Prompt */}
451
+ <div>
452
+ <label style={labelStyle}>Prompt</label>
453
+ <textarea value={prompt} onChange={e => setPrompt(e.target.value)}
454
+ placeholder="What should this agent do? e.g. Generate a summary of yesterday's bookings with key metrics"
455
+ rows={3} style={{ ...inputStyle, resize: 'vertical', fontFamily: 'var(--font-sans)' }} />
456
+ <p style={{ margin: '4px 0 0', fontSize: 11, color: 'var(--subtle)' }}>
457
+ Sent to the agent on each scheduled run.
458
+ </p>
459
+ </div>
460
+
461
+ {/* Schedule */}
462
+ <div>
463
+ <label style={labelStyle}>Schedule</label>
464
+ <div style={{ display: 'flex', flexWrap: 'wrap', gap: 6, marginBottom: 8 }}>
465
+ {PRESETS.map(p => (
466
+ <button key={p.cron} onClick={() => setCronSchedule(p.cron)}
467
+ style={{
468
+ padding: '4px 10px', borderRadius: 6, fontSize: 11, fontWeight: 500,
469
+ border: cronSchedule === p.cron ? '1px solid var(--accent)' : '1px solid var(--border)',
470
+ background: cronSchedule === p.cron ? 'rgba(23,23,23,0.06)' : '#fff',
471
+ color: cronSchedule === p.cron ? 'var(--text)' : 'var(--muted)',
472
+ cursor: 'pointer', fontFamily: 'var(--font-sans)',
473
+ }}>{p.label}</button>
474
+ ))}
475
+ </div>
476
+ <input type="text" value={cronSchedule} onChange={e => setCronSchedule(e.target.value)}
477
+ placeholder="0 8 * * *" style={{ ...inputStyle, fontFamily: 'var(--font-mono)' }} />
478
+ <p style={{ margin: '4px 0 0', fontSize: 11, color: 'var(--subtle)' }}>
479
+ Cron expression: minute hour day month weekday — <span style={{ color: 'var(--text)' }}>{cronToHuman(cronSchedule)}</span>
480
+ </p>
481
+ </div>
482
+
483
+ {/* Target */}
484
+ <div>
485
+ <label style={labelStyle}>Deliver to</label>
486
+ <div style={{ display: 'flex', gap: 8, marginBottom: 8 }}>
487
+ {(['channel', 'dm'] as const).map(t => (
488
+ <label key={t} style={{
489
+ display: 'flex', alignItems: 'center', gap: 6, padding: '6px 12px',
490
+ border: targetType === t ? '1px solid var(--accent)' : '1px solid var(--border)',
491
+ borderRadius: 7, cursor: 'pointer', fontSize: 13,
492
+ background: targetType === t ? 'rgba(23,23,23,0.06)' : '#fff',
493
+ color: targetType === t ? 'var(--text)' : 'var(--muted)',
494
+ }}>
495
+ <input type="radio" name="targetType" checked={targetType === t} onChange={() => setTargetType(t)} style={{ display: 'none' }} />
496
+ <span style={{ display: 'inline-flex', alignItems: 'center', gap: 5 }}>
497
+ {t === 'channel' ? <Hash size={13} /> : <MessageSquare size={13} />}
498
+ {t === 'channel' ? 'Channel' : 'DM'}
499
+ </span>
500
+ </label>
501
+ ))}
502
+ </div>
503
+ <input type="text" value={targetId} onChange={e => setTargetId(e.target.value)}
504
+ placeholder={targetType === 'channel' ? 'Channel ID (e.g. C0ANTCQ918U)' : 'User ID (e.g. U095GQAM6PL)'}
505
+ style={inputStyle} />
506
+ </div>
507
+
508
+ {/* Enabled */}
509
+ <label style={{ display: 'flex', alignItems: 'center', gap: 8, cursor: 'pointer' }}>
510
+ <input type="checkbox" checked={enabled} onChange={e => setEnabled(e.target.checked)}
511
+ style={{ accentColor: 'var(--accent)', width: 14, height: 14 }} />
512
+ <span style={{ fontSize: 13, color: 'var(--text)' }}>Enabled</span>
513
+ </label>
514
+
515
+ {error && <div style={{ fontSize: 12, color: '#dc2626', background: 'rgba(220,38,38,0.06)', padding: '6px 10px', borderRadius: 6 }}>{error}</div>}
516
+
517
+ <div style={{ display: 'flex', gap: 8, justifyContent: 'flex-end' }}>
518
+ <button onClick={onClose} style={{
519
+ padding: '8px 16px', borderRadius: 7, border: '1px solid var(--border)',
520
+ background: '#fff', fontSize: 13, cursor: 'pointer', fontFamily: 'var(--font-sans)',
521
+ }}>Cancel</button>
522
+ <button onClick={save} disabled={saving} style={{
523
+ padding: '8px 18px', borderRadius: 7, border: 'none',
524
+ background: 'var(--accent)', color: '#fff', fontSize: 13, fontWeight: 500,
525
+ cursor: 'pointer', fontFamily: 'var(--font-sans)',
526
+ }}>{saving ? 'Saving...' : isEdit ? 'Save Changes' : 'Create Job'}</button>
527
+ </div>
528
+ </div>
529
+ </div>
530
+ </Portal>
531
+ );
532
+ }
533
+
534
+ const labelStyle: React.CSSProperties = {
535
+ display: 'block', fontSize: 12, fontWeight: 500, color: 'var(--muted)', marginBottom: 5,
536
+ };
537
+
538
+ const inputStyle: React.CSSProperties = {
539
+ width: '100%', padding: '8px 12px', borderRadius: 7,
540
+ border: '1px solid var(--border)', fontSize: 13, outline: 'none',
541
+ boxSizing: 'border-box' as const, fontFamily: 'var(--font-sans)',
542
+ color: 'var(--text)',
543
+ };
@@ -0,0 +1,89 @@
1
+ 'use client';
2
+
3
+ /**
4
+ * @fileoverview Client layout shell — auth provider + responsive sidebar + main.
5
+ *
6
+ * On mobile (<768px), sidebar is hidden by default and shown as an overlay.
7
+ * A hamburger button is shown in the top-left of the main area.
8
+ *
9
+ * @module web/app/layout-shell
10
+ */
11
+
12
+ import { useContext, useState, useEffect } from 'react';
13
+ import { usePathname } from 'next/navigation';
14
+ import { Sidebar, SidebarContext } from './sidebar';
15
+ import { AuthProvider } from '@/lib/auth-context';
16
+
17
+ /**
18
+ * Renders auth provider + sidebar + main content.
19
+ * Login page gets no sidebar.
20
+ *
21
+ * @param {{ children: React.ReactNode }} props
22
+ */
23
+ export function LayoutShell({ children }: { children: React.ReactNode }) {
24
+ const pathname = usePathname();
25
+
26
+ if (pathname === '/login') {
27
+ return <>{children}</>;
28
+ }
29
+
30
+ return (
31
+ <AuthProvider>
32
+ <ResponsiveLayout>{children}</ResponsiveLayout>
33
+ </AuthProvider>
34
+ );
35
+ }
36
+
37
+ function ResponsiveLayout({ children }: { children: React.ReactNode }) {
38
+ const [mobileOpen, setMobileOpen] = useState(false);
39
+ const [isMobile, setIsMobile] = useState(false);
40
+ const pathname = usePathname();
41
+
42
+ useEffect(() => {
43
+ const check = () => setIsMobile(window.innerWidth <= 768);
44
+ check();
45
+ window.addEventListener('resize', check);
46
+ return () => window.removeEventListener('resize', check);
47
+ }, []);
48
+
49
+ // Close mobile sidebar on navigation
50
+ useEffect(() => { setMobileOpen(false); }, [pathname]);
51
+
52
+ return (
53
+ <Sidebar mobileOpen={mobileOpen} onMobileClose={() => setMobileOpen(false)}>
54
+ <Main isMobile={isMobile} onHamburger={() => setMobileOpen(true)}>
55
+ {children}
56
+ </Main>
57
+ </Sidebar>
58
+ );
59
+ }
60
+
61
+ function Main({ children, isMobile, onHamburger }: { children: React.ReactNode; isMobile: boolean; onHamburger: () => void }) {
62
+ const { width } = useContext(SidebarContext);
63
+ return (
64
+ <main style={{
65
+ marginLeft: isMobile ? 0 : width,
66
+ transition: 'margin-left 0.25s cubic-bezier(0.16,1,0.3,1)',
67
+ }}>
68
+ {/* Mobile hamburger */}
69
+ {isMobile && (
70
+ <button
71
+ onClick={onHamburger}
72
+ style={{
73
+ position: 'fixed', top: 12, left: 12, zIndex: 48,
74
+ width: 36, height: 36, borderRadius: 8,
75
+ background: '#fff', border: '1px solid var(--border)',
76
+ boxShadow: 'var(--shadow-md)',
77
+ display: 'flex', alignItems: 'center', justifyContent: 'center',
78
+ cursor: 'pointer', color: 'var(--text)',
79
+ }}
80
+ >
81
+ <svg width="18" height="18" viewBox="0 0 16 16" fill="none">
82
+ <path d="M2 4h12M2 8h12M2 12h12" stroke="currentColor" strokeWidth="1.4" strokeLinecap="round"/>
83
+ </svg>
84
+ </button>
85
+ )}
86
+ {children}
87
+ </main>
88
+ );
89
+ }
@@ -0,0 +1,18 @@
1
+ import type { Metadata } from 'next';
2
+ import './globals.css';
3
+ import { LayoutShell } from './layout-shell';
4
+
5
+ export const metadata: Metadata = {
6
+ title: 'SlackHive',
7
+ description: 'Claude Code agent control plane',
8
+ };
9
+
10
+ export default function RootLayout({ children }: { children: React.ReactNode }) {
11
+ return (
12
+ <html lang="en">
13
+ <body style={{ margin: 0 }}>
14
+ <LayoutShell>{children}</LayoutShell>
15
+ </body>
16
+ </html>
17
+ );
18
+ }
@@ -0,0 +1,9 @@
1
+ /**
2
+ * @fileoverview Login layout — no sidebar, just centered content.
3
+ *
4
+ * @module web/app/login/layout
5
+ */
6
+
7
+ export default function LoginLayout({ children }: { children: React.ReactNode }) {
8
+ return <>{children}</>;
9
+ }