@jlongo78/agent-spaces 0.7.7 → 0.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (797) hide show
  1. package/.next/standalone/.next/BUILD_ID +1 -1
  2. package/.next/standalone/.next/app-path-routes-manifest.json +6 -0
  3. package/.next/standalone/.next/build-manifest.json +2 -2
  4. package/.next/standalone/.next/prerender-manifest.json +3 -3
  5. package/.next/standalone/.next/required-server-files.json +19 -19
  6. package/.next/standalone/.next/routes-manifest.json +38 -0
  7. package/.next/standalone/.next/server/app/(desktop)/admin/analytics/page_client-reference-manifest.js +1 -1
  8. package/.next/standalone/.next/server/app/(desktop)/admin/users/page_client-reference-manifest.js +1 -1
  9. package/.next/standalone/.next/server/app/(desktop)/analytics/page_client-reference-manifest.js +1 -1
  10. package/.next/standalone/.next/server/app/(desktop)/cortex/page/react-loadable-manifest.json +1 -1
  11. package/.next/standalone/.next/server/app/(desktop)/cortex/page.js.nft.json +1 -1
  12. package/.next/standalone/.next/server/app/(desktop)/cortex/page_client-reference-manifest.js +1 -1
  13. package/.next/standalone/.next/server/app/(desktop)/network/page_client-reference-manifest.js +1 -1
  14. package/.next/standalone/.next/server/app/(desktop)/page_client-reference-manifest.js +1 -1
  15. package/.next/standalone/.next/server/app/(desktop)/projects/page_client-reference-manifest.js +1 -1
  16. package/.next/standalone/.next/server/app/(desktop)/sessions/[id]/page.js.nft.json +1 -1
  17. package/.next/standalone/.next/server/app/(desktop)/sessions/[id]/page_client-reference-manifest.js +1 -1
  18. package/.next/standalone/.next/server/app/(desktop)/sessions/page_client-reference-manifest.js +1 -1
  19. package/.next/standalone/.next/server/app/(desktop)/settings/page_client-reference-manifest.js +1 -1
  20. package/.next/standalone/.next/server/app/(desktop)/terminal/page.js.nft.json +1 -1
  21. package/.next/standalone/.next/server/app/(desktop)/terminal/page_client-reference-manifest.js +1 -1
  22. package/.next/standalone/.next/server/app/(desktop)/terminal/pane/[id]/page.js.nft.json +1 -1
  23. package/.next/standalone/.next/server/app/(desktop)/terminal/pane/[id]/page_client-reference-manifest.js +1 -1
  24. package/.next/standalone/.next/server/app/(desktop)/terminal/remote/[nodeId]/[workspaceId]/page.js.nft.json +1 -1
  25. package/.next/standalone/.next/server/app/(desktop)/terminal/remote/[nodeId]/[workspaceId]/page_client-reference-manifest.js +1 -1
  26. package/.next/standalone/.next/server/app/(desktop)/workspaces/page_client-reference-manifest.js +1 -1
  27. package/.next/standalone/.next/server/app/_global-error.html +2 -2
  28. package/.next/standalone/.next/server/app/_global-error.rsc +1 -1
  29. package/.next/standalone/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
  30. package/.next/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  31. package/.next/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  32. package/.next/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  33. package/.next/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  34. package/.next/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  35. package/.next/standalone/.next/server/app/_not-found.html +1 -1
  36. package/.next/standalone/.next/server/app/_not-found.rsc +2 -2
  37. package/.next/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
  38. package/.next/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  39. package/.next/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
  40. package/.next/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  41. package/.next/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  42. package/.next/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
  43. package/.next/standalone/.next/server/app/admin/analytics.html +1 -1
  44. package/.next/standalone/.next/server/app/admin/analytics.rsc +6 -7
  45. package/.next/standalone/.next/server/app/admin/analytics.segments/!KGRlc2t0b3Ap/admin/analytics/__PAGE__.segment.rsc +2 -2
  46. package/.next/standalone/.next/server/app/admin/analytics.segments/!KGRlc2t0b3Ap/admin/analytics.segment.rsc +1 -1
  47. package/.next/standalone/.next/server/app/admin/analytics.segments/!KGRlc2t0b3Ap/admin.segment.rsc +1 -1
  48. package/.next/standalone/.next/server/app/admin/analytics.segments/!KGRlc2t0b3Ap.segment.rsc +1 -1
  49. package/.next/standalone/.next/server/app/admin/analytics.segments/_full.segment.rsc +6 -7
  50. package/.next/standalone/.next/server/app/admin/analytics.segments/_head.segment.rsc +1 -1
  51. package/.next/standalone/.next/server/app/admin/analytics.segments/_index.segment.rsc +2 -2
  52. package/.next/standalone/.next/server/app/admin/analytics.segments/_tree.segment.rsc +2 -2
  53. package/.next/standalone/.next/server/app/admin/users.html +1 -1
  54. package/.next/standalone/.next/server/app/admin/users.rsc +2 -2
  55. package/.next/standalone/.next/server/app/admin/users.segments/!KGRlc2t0b3Ap/admin/users/__PAGE__.segment.rsc +1 -1
  56. package/.next/standalone/.next/server/app/admin/users.segments/!KGRlc2t0b3Ap/admin/users.segment.rsc +1 -1
  57. package/.next/standalone/.next/server/app/admin/users.segments/!KGRlc2t0b3Ap/admin.segment.rsc +1 -1
  58. package/.next/standalone/.next/server/app/admin/users.segments/!KGRlc2t0b3Ap.segment.rsc +1 -1
  59. package/.next/standalone/.next/server/app/admin/users.segments/_full.segment.rsc +2 -2
  60. package/.next/standalone/.next/server/app/admin/users.segments/_head.segment.rsc +1 -1
  61. package/.next/standalone/.next/server/app/admin/users.segments/_index.segment.rsc +2 -2
  62. package/.next/standalone/.next/server/app/admin/users.segments/_tree.segment.rsc +2 -2
  63. package/.next/standalone/.next/server/app/analytics.html +1 -1
  64. package/.next/standalone/.next/server/app/analytics.rsc +3 -3
  65. package/.next/standalone/.next/server/app/analytics.segments/!KGRlc2t0b3Ap/analytics/__PAGE__.segment.rsc +2 -2
  66. package/.next/standalone/.next/server/app/analytics.segments/!KGRlc2t0b3Ap/analytics.segment.rsc +1 -1
  67. package/.next/standalone/.next/server/app/analytics.segments/!KGRlc2t0b3Ap.segment.rsc +1 -1
  68. package/.next/standalone/.next/server/app/analytics.segments/_full.segment.rsc +3 -3
  69. package/.next/standalone/.next/server/app/analytics.segments/_head.segment.rsc +1 -1
  70. package/.next/standalone/.next/server/app/analytics.segments/_index.segment.rsc +2 -2
  71. package/.next/standalone/.next/server/app/analytics.segments/_tree.segment.rsc +2 -2
  72. package/.next/standalone/.next/server/app/api/analytics/overview/route.js.nft.json +1 -1
  73. package/.next/standalone/.next/server/app/api/benchmark/lobes/route/app-paths-manifest.json +3 -0
  74. package/.next/standalone/.next/server/app/api/benchmark/lobes/route/build-manifest.json +11 -0
  75. package/.next/standalone/.next/server/app/api/benchmark/lobes/route/server-reference-manifest.json +4 -0
  76. package/.next/standalone/.next/server/app/api/benchmark/lobes/route.js +7 -0
  77. package/.next/standalone/.next/server/app/api/benchmark/lobes/route.js.map +5 -0
  78. package/.next/standalone/.next/server/app/api/benchmark/lobes/route.js.nft.json +1 -0
  79. package/.next/standalone/.next/server/app/api/benchmark/lobes/route_client-reference-manifest.js +2 -0
  80. package/.next/standalone/.next/server/app/api/benchmark/run/route/app-paths-manifest.json +3 -0
  81. package/.next/standalone/.next/server/app/api/benchmark/run/route/build-manifest.json +11 -0
  82. package/.next/standalone/.next/server/app/api/benchmark/run/route/server-reference-manifest.json +4 -0
  83. package/.next/standalone/.next/server/app/api/benchmark/run/route.js +7 -0
  84. package/.next/standalone/.next/server/app/api/benchmark/run/route.js.map +5 -0
  85. package/.next/standalone/.next/server/app/api/benchmark/run/route.js.nft.json +1 -0
  86. package/.next/standalone/.next/server/app/api/benchmark/run/route_client-reference-manifest.js +2 -0
  87. package/.next/standalone/.next/server/app/api/benchmark/runs/[id]/route/app-paths-manifest.json +3 -0
  88. package/.next/standalone/.next/server/app/api/benchmark/runs/[id]/route/build-manifest.json +11 -0
  89. package/.next/standalone/.next/server/app/api/benchmark/runs/[id]/route/server-reference-manifest.json +4 -0
  90. package/.next/standalone/.next/server/app/api/benchmark/runs/[id]/route.js +7 -0
  91. package/.next/standalone/.next/server/app/api/benchmark/runs/[id]/route.js.map +5 -0
  92. package/.next/standalone/.next/server/app/api/benchmark/runs/[id]/route.js.nft.json +1 -0
  93. package/.next/standalone/.next/server/app/api/benchmark/runs/[id]/route_client-reference-manifest.js +2 -0
  94. package/.next/standalone/.next/server/app/api/benchmark/runs/route/app-paths-manifest.json +3 -0
  95. package/.next/standalone/.next/server/app/api/benchmark/runs/route/build-manifest.json +11 -0
  96. package/.next/standalone/.next/server/app/api/benchmark/runs/route/server-reference-manifest.json +4 -0
  97. package/.next/standalone/.next/server/app/api/benchmark/runs/route.js +7 -0
  98. package/.next/standalone/.next/server/app/api/benchmark/runs/route.js.map +5 -0
  99. package/.next/standalone/.next/server/app/api/benchmark/runs/route.js.nft.json +1 -0
  100. package/.next/standalone/.next/server/app/api/benchmark/runs/route_client-reference-manifest.js +2 -0
  101. package/.next/standalone/.next/server/app/api/benchmark/status/route/app-paths-manifest.json +3 -0
  102. package/.next/standalone/.next/server/app/api/benchmark/status/route/build-manifest.json +11 -0
  103. package/.next/standalone/.next/server/app/api/benchmark/status/route/server-reference-manifest.json +4 -0
  104. package/.next/standalone/.next/server/app/api/benchmark/status/route.js +7 -0
  105. package/.next/standalone/.next/server/app/api/benchmark/status/route.js.map +5 -0
  106. package/.next/standalone/.next/server/app/api/benchmark/status/route.js.nft.json +1 -0
  107. package/.next/standalone/.next/server/app/api/benchmark/status/route_client-reference-manifest.js +2 -0
  108. package/.next/standalone/.next/server/app/api/bulk/route.js.nft.json +1 -1
  109. package/.next/standalone/.next/server/app/api/config/route.js.nft.json +1 -1
  110. package/.next/standalone/.next/server/app/api/cortex/context/route.js.nft.json +1 -1
  111. package/.next/standalone/.next/server/app/api/cortex/curation/assess/route.js.nft.json +1 -1
  112. package/.next/standalone/.next/server/app/api/cortex/curation/publish/route.js.nft.json +1 -1
  113. package/.next/standalone/.next/server/app/api/cortex/curation/refine/route.js.nft.json +1 -1
  114. package/.next/standalone/.next/server/app/api/cortex/curation/review/route.js.nft.json +1 -1
  115. package/.next/standalone/.next/server/app/api/cortex/curation/seed/route.js.nft.json +1 -1
  116. package/.next/standalone/.next/server/app/api/cortex/export/route.js.nft.json +1 -1
  117. package/.next/standalone/.next/server/app/api/cortex/federation/pending/route.js.nft.json +1 -1
  118. package/.next/standalone/.next/server/app/api/cortex/federation/resolve/route.js.nft.json +1 -1
  119. package/.next/standalone/.next/server/app/api/cortex/federation/search/route.js.nft.json +1 -1
  120. package/.next/standalone/.next/server/app/api/cortex/federation/teach/route.js.nft.json +1 -1
  121. package/.next/standalone/.next/server/app/api/cortex/graph/edges/route.js.nft.json +1 -1
  122. package/.next/standalone/.next/server/app/api/cortex/graph/entities/[id]/route.js.nft.json +1 -1
  123. package/.next/standalone/.next/server/app/api/cortex/graph/entities/route.js.nft.json +1 -1
  124. package/.next/standalone/.next/server/app/api/cortex/graph/populate/route.js.nft.json +1 -1
  125. package/.next/standalone/.next/server/app/api/cortex/import/route.js.nft.json +1 -1
  126. package/.next/standalone/.next/server/app/api/cortex/import/status/route.js.nft.json +1 -1
  127. package/.next/standalone/.next/server/app/api/cortex/ingest/bootstrap/route.js.nft.json +1 -1
  128. package/.next/standalone/.next/server/app/api/cortex/ingest/status/route.js.nft.json +1 -1
  129. package/.next/standalone/.next/server/app/api/cortex/knowledge/[id]/route.js.nft.json +1 -1
  130. package/.next/standalone/.next/server/app/api/cortex/knowledge/route.js.nft.json +1 -1
  131. package/.next/standalone/.next/server/app/api/cortex/lobes/[id]/route.js.nft.json +1 -1
  132. package/.next/standalone/.next/server/app/api/cortex/lobes/route.js.nft.json +1 -1
  133. package/.next/standalone/.next/server/app/api/cortex/lobes/share/route.js.nft.json +1 -1
  134. package/.next/standalone/.next/server/app/api/cortex/marketplace/browse/route.js.nft.json +1 -1
  135. package/.next/standalone/.next/server/app/api/cortex/marketplace/preview/route.js.nft.json +1 -1
  136. package/.next/standalone/.next/server/app/api/cortex/mcp/call/route.js.nft.json +1 -1
  137. package/.next/standalone/.next/server/app/api/cortex/mcp/tools/route.js.nft.json +1 -1
  138. package/.next/standalone/.next/server/app/api/cortex/search/route.js.nft.json +1 -1
  139. package/.next/standalone/.next/server/app/api/cortex/settings/route.js.nft.json +1 -1
  140. package/.next/standalone/.next/server/app/api/cortex/status/route.js.nft.json +1 -1
  141. package/.next/standalone/.next/server/app/api/cortex/timeline/route.js.nft.json +1 -1
  142. package/.next/standalone/.next/server/app/api/cortex/usage/route.js.nft.json +1 -1
  143. package/.next/standalone/.next/server/app/api/cortex/workspace/[id]/context/route.js.nft.json +1 -1
  144. package/.next/standalone/.next/server/app/api/events/route.js.nft.json +1 -1
  145. package/.next/standalone/.next/server/app/api/files/route.js +1 -1
  146. package/.next/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  147. package/.next/standalone/.next/server/app/api/folders/route.js.nft.json +1 -1
  148. package/.next/standalone/.next/server/app/api/network/handshake/route.js.nft.json +1 -1
  149. package/.next/standalone/.next/server/app/api/network/projects/route.js.nft.json +1 -1
  150. package/.next/standalone/.next/server/app/api/network/search/route.js.nft.json +1 -1
  151. package/.next/standalone/.next/server/app/api/network/sessions/[id]/messages/route.js.nft.json +1 -1
  152. package/.next/standalone/.next/server/app/api/network/sessions/[id]/route.js.nft.json +1 -1
  153. package/.next/standalone/.next/server/app/api/network/sessions/route.js.nft.json +1 -1
  154. package/.next/standalone/.next/server/app/api/network/workspaces/[id]/route.js.nft.json +1 -1
  155. package/.next/standalone/.next/server/app/api/network/workspaces/route.js.nft.json +1 -1
  156. package/.next/standalone/.next/server/app/api/panes/[id]/route.js.nft.json +1 -1
  157. package/.next/standalone/.next/server/app/api/panes/route.js.nft.json +1 -1
  158. package/.next/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  159. package/.next/standalone/.next/server/app/api/search/route.js.nft.json +1 -1
  160. package/.next/standalone/.next/server/app/api/sessions/[id]/chat/route.js.nft.json +1 -1
  161. package/.next/standalone/.next/server/app/api/sessions/[id]/messages/route.js.nft.json +1 -1
  162. package/.next/standalone/.next/server/app/api/sessions/[id]/route.js.nft.json +1 -1
  163. package/.next/standalone/.next/server/app/api/sessions/route.js.nft.json +1 -1
  164. package/.next/standalone/.next/server/app/api/sync/route.js.nft.json +1 -1
  165. package/.next/standalone/.next/server/app/api/tags/route.js.nft.json +1 -1
  166. package/.next/standalone/.next/server/app/api/tier/route.js.nft.json +1 -1
  167. package/.next/standalone/.next/server/app/api/whisper/route/app-paths-manifest.json +3 -0
  168. package/.next/standalone/.next/server/app/api/whisper/route/build-manifest.json +11 -0
  169. package/.next/standalone/.next/server/app/api/whisper/route/server-reference-manifest.json +4 -0
  170. package/.next/standalone/.next/server/app/api/whisper/route.js +7 -0
  171. package/.next/standalone/.next/server/app/api/whisper/route.js.map +5 -0
  172. package/.next/standalone/.next/server/app/api/whisper/route.js.nft.json +1 -0
  173. package/.next/standalone/.next/server/app/api/whisper/route_client-reference-manifest.js +2 -0
  174. package/.next/standalone/.next/server/app/api/workspaces/[id]/context/[key]/route.js.nft.json +1 -1
  175. package/.next/standalone/.next/server/app/api/workspaces/[id]/context/route.js.nft.json +1 -1
  176. package/.next/standalone/.next/server/app/api/workspaces/[id]/messages/[msgId]/route.js.nft.json +1 -1
  177. package/.next/standalone/.next/server/app/api/workspaces/[id]/messages/route.js.nft.json +1 -1
  178. package/.next/standalone/.next/server/app/api/workspaces/[id]/route.js.nft.json +1 -1
  179. package/.next/standalone/.next/server/app/api/workspaces/[id]/sessions/route.js.nft.json +1 -1
  180. package/.next/standalone/.next/server/app/api/workspaces/route.js.nft.json +1 -1
  181. package/.next/standalone/.next/server/app/cortex.html +1 -1
  182. package/.next/standalone/.next/server/app/cortex.rsc +3 -3
  183. package/.next/standalone/.next/server/app/cortex.segments/!KGRlc2t0b3Ap/cortex/__PAGE__.segment.rsc +2 -2
  184. package/.next/standalone/.next/server/app/cortex.segments/!KGRlc2t0b3Ap/cortex.segment.rsc +1 -1
  185. package/.next/standalone/.next/server/app/cortex.segments/!KGRlc2t0b3Ap.segment.rsc +1 -1
  186. package/.next/standalone/.next/server/app/cortex.segments/_full.segment.rsc +3 -3
  187. package/.next/standalone/.next/server/app/cortex.segments/_head.segment.rsc +1 -1
  188. package/.next/standalone/.next/server/app/cortex.segments/_index.segment.rsc +2 -2
  189. package/.next/standalone/.next/server/app/cortex.segments/_tree.segment.rsc +2 -2
  190. package/.next/standalone/.next/server/app/login/page_client-reference-manifest.js +1 -1
  191. package/.next/standalone/.next/server/app/login.html +1 -1
  192. package/.next/standalone/.next/server/app/login.rsc +2 -2
  193. package/.next/standalone/.next/server/app/login.segments/_full.segment.rsc +2 -2
  194. package/.next/standalone/.next/server/app/login.segments/_head.segment.rsc +1 -1
  195. package/.next/standalone/.next/server/app/login.segments/_index.segment.rsc +2 -2
  196. package/.next/standalone/.next/server/app/login.segments/_tree.segment.rsc +2 -2
  197. package/.next/standalone/.next/server/app/login.segments/login/__PAGE__.segment.rsc +1 -1
  198. package/.next/standalone/.next/server/app/login.segments/login.segment.rsc +1 -1
  199. package/.next/standalone/.next/server/app/m/page_client-reference-manifest.js +1 -1
  200. package/.next/standalone/.next/server/app/m/projects/page_client-reference-manifest.js +1 -1
  201. package/.next/standalone/.next/server/app/m/projects.html +1 -1
  202. package/.next/standalone/.next/server/app/m/projects.rsc +2 -2
  203. package/.next/standalone/.next/server/app/m/projects.segments/_full.segment.rsc +2 -2
  204. package/.next/standalone/.next/server/app/m/projects.segments/_head.segment.rsc +1 -1
  205. package/.next/standalone/.next/server/app/m/projects.segments/_index.segment.rsc +2 -2
  206. package/.next/standalone/.next/server/app/m/projects.segments/_tree.segment.rsc +2 -2
  207. package/.next/standalone/.next/server/app/m/projects.segments/m/projects/__PAGE__.segment.rsc +1 -1
  208. package/.next/standalone/.next/server/app/m/projects.segments/m/projects.segment.rsc +1 -1
  209. package/.next/standalone/.next/server/app/m/projects.segments/m.segment.rsc +1 -1
  210. package/.next/standalone/.next/server/app/m/sessions/[id]/page.js.nft.json +1 -1
  211. package/.next/standalone/.next/server/app/m/sessions/[id]/page_client-reference-manifest.js +1 -1
  212. package/.next/standalone/.next/server/app/m/sessions/page_client-reference-manifest.js +1 -1
  213. package/.next/standalone/.next/server/app/m/sessions.html +1 -1
  214. package/.next/standalone/.next/server/app/m/sessions.rsc +2 -2
  215. package/.next/standalone/.next/server/app/m/sessions.segments/_full.segment.rsc +2 -2
  216. package/.next/standalone/.next/server/app/m/sessions.segments/_head.segment.rsc +1 -1
  217. package/.next/standalone/.next/server/app/m/sessions.segments/_index.segment.rsc +2 -2
  218. package/.next/standalone/.next/server/app/m/sessions.segments/_tree.segment.rsc +2 -2
  219. package/.next/standalone/.next/server/app/m/sessions.segments/m/sessions/__PAGE__.segment.rsc +1 -1
  220. package/.next/standalone/.next/server/app/m/sessions.segments/m/sessions.segment.rsc +1 -1
  221. package/.next/standalone/.next/server/app/m/sessions.segments/m.segment.rsc +1 -1
  222. package/.next/standalone/.next/server/app/m/settings/page_client-reference-manifest.js +1 -1
  223. package/.next/standalone/.next/server/app/m/settings.html +1 -1
  224. package/.next/standalone/.next/server/app/m/settings.rsc +2 -2
  225. package/.next/standalone/.next/server/app/m/settings.segments/_full.segment.rsc +2 -2
  226. package/.next/standalone/.next/server/app/m/settings.segments/_head.segment.rsc +1 -1
  227. package/.next/standalone/.next/server/app/m/settings.segments/_index.segment.rsc +2 -2
  228. package/.next/standalone/.next/server/app/m/settings.segments/_tree.segment.rsc +2 -2
  229. package/.next/standalone/.next/server/app/m/settings.segments/m/settings/__PAGE__.segment.rsc +1 -1
  230. package/.next/standalone/.next/server/app/m/settings.segments/m/settings.segment.rsc +1 -1
  231. package/.next/standalone/.next/server/app/m/settings.segments/m.segment.rsc +1 -1
  232. package/.next/standalone/.next/server/app/m/terminal/page.js.nft.json +1 -1
  233. package/.next/standalone/.next/server/app/m/terminal/page_client-reference-manifest.js +1 -1
  234. package/.next/standalone/.next/server/app/m/terminal.html +1 -1
  235. package/.next/standalone/.next/server/app/m/terminal.rsc +3 -3
  236. package/.next/standalone/.next/server/app/m/terminal.segments/_full.segment.rsc +3 -3
  237. package/.next/standalone/.next/server/app/m/terminal.segments/_head.segment.rsc +1 -1
  238. package/.next/standalone/.next/server/app/m/terminal.segments/_index.segment.rsc +2 -2
  239. package/.next/standalone/.next/server/app/m/terminal.segments/_tree.segment.rsc +2 -2
  240. package/.next/standalone/.next/server/app/m/terminal.segments/m/terminal/__PAGE__.segment.rsc +2 -2
  241. package/.next/standalone/.next/server/app/m/terminal.segments/m/terminal.segment.rsc +1 -1
  242. package/.next/standalone/.next/server/app/m/terminal.segments/m.segment.rsc +1 -1
  243. package/.next/standalone/.next/server/app/m.html +1 -1
  244. package/.next/standalone/.next/server/app/m.rsc +2 -2
  245. package/.next/standalone/.next/server/app/m.segments/_full.segment.rsc +2 -2
  246. package/.next/standalone/.next/server/app/m.segments/_head.segment.rsc +1 -1
  247. package/.next/standalone/.next/server/app/m.segments/_index.segment.rsc +2 -2
  248. package/.next/standalone/.next/server/app/m.segments/_tree.segment.rsc +2 -2
  249. package/.next/standalone/.next/server/app/m.segments/m/__PAGE__.segment.rsc +1 -1
  250. package/.next/standalone/.next/server/app/m.segments/m.segment.rsc +1 -1
  251. package/.next/standalone/.next/server/app/network.html +1 -1
  252. package/.next/standalone/.next/server/app/network.rsc +2 -2
  253. package/.next/standalone/.next/server/app/network.segments/!KGRlc2t0b3Ap/network/__PAGE__.segment.rsc +1 -1
  254. package/.next/standalone/.next/server/app/network.segments/!KGRlc2t0b3Ap/network.segment.rsc +1 -1
  255. package/.next/standalone/.next/server/app/network.segments/!KGRlc2t0b3Ap.segment.rsc +1 -1
  256. package/.next/standalone/.next/server/app/network.segments/_full.segment.rsc +2 -2
  257. package/.next/standalone/.next/server/app/network.segments/_head.segment.rsc +1 -1
  258. package/.next/standalone/.next/server/app/network.segments/_index.segment.rsc +2 -2
  259. package/.next/standalone/.next/server/app/network.segments/_tree.segment.rsc +2 -2
  260. package/.next/standalone/.next/server/app/projects.html +1 -1
  261. package/.next/standalone/.next/server/app/projects.rsc +2 -2
  262. package/.next/standalone/.next/server/app/projects.segments/!KGRlc2t0b3Ap/projects/__PAGE__.segment.rsc +1 -1
  263. package/.next/standalone/.next/server/app/projects.segments/!KGRlc2t0b3Ap/projects.segment.rsc +1 -1
  264. package/.next/standalone/.next/server/app/projects.segments/!KGRlc2t0b3Ap.segment.rsc +1 -1
  265. package/.next/standalone/.next/server/app/projects.segments/_full.segment.rsc +2 -2
  266. package/.next/standalone/.next/server/app/projects.segments/_head.segment.rsc +1 -1
  267. package/.next/standalone/.next/server/app/projects.segments/_index.segment.rsc +2 -2
  268. package/.next/standalone/.next/server/app/projects.segments/_tree.segment.rsc +2 -2
  269. package/.next/standalone/.next/server/app/sessions.html +1 -1
  270. package/.next/standalone/.next/server/app/sessions.rsc +2 -2
  271. package/.next/standalone/.next/server/app/sessions.segments/!KGRlc2t0b3Ap/sessions/__PAGE__.segment.rsc +1 -1
  272. package/.next/standalone/.next/server/app/sessions.segments/!KGRlc2t0b3Ap/sessions.segment.rsc +1 -1
  273. package/.next/standalone/.next/server/app/sessions.segments/!KGRlc2t0b3Ap.segment.rsc +1 -1
  274. package/.next/standalone/.next/server/app/sessions.segments/_full.segment.rsc +2 -2
  275. package/.next/standalone/.next/server/app/sessions.segments/_head.segment.rsc +1 -1
  276. package/.next/standalone/.next/server/app/sessions.segments/_index.segment.rsc +2 -2
  277. package/.next/standalone/.next/server/app/sessions.segments/_tree.segment.rsc +2 -2
  278. package/.next/standalone/.next/server/app/settings.html +1 -1
  279. package/.next/standalone/.next/server/app/settings.rsc +2 -2
  280. package/.next/standalone/.next/server/app/settings.segments/!KGRlc2t0b3Ap/settings/__PAGE__.segment.rsc +1 -1
  281. package/.next/standalone/.next/server/app/settings.segments/!KGRlc2t0b3Ap/settings.segment.rsc +1 -1
  282. package/.next/standalone/.next/server/app/settings.segments/!KGRlc2t0b3Ap.segment.rsc +1 -1
  283. package/.next/standalone/.next/server/app/settings.segments/_full.segment.rsc +2 -2
  284. package/.next/standalone/.next/server/app/settings.segments/_head.segment.rsc +1 -1
  285. package/.next/standalone/.next/server/app/settings.segments/_index.segment.rsc +2 -2
  286. package/.next/standalone/.next/server/app/settings.segments/_tree.segment.rsc +2 -2
  287. package/.next/standalone/.next/server/app/terminal.html +1 -1
  288. package/.next/standalone/.next/server/app/terminal.rsc +3 -3
  289. package/.next/standalone/.next/server/app/terminal.segments/!KGRlc2t0b3Ap/terminal/__PAGE__.segment.rsc +2 -2
  290. package/.next/standalone/.next/server/app/terminal.segments/!KGRlc2t0b3Ap/terminal.segment.rsc +1 -1
  291. package/.next/standalone/.next/server/app/terminal.segments/!KGRlc2t0b3Ap.segment.rsc +1 -1
  292. package/.next/standalone/.next/server/app/terminal.segments/_full.segment.rsc +3 -3
  293. package/.next/standalone/.next/server/app/terminal.segments/_head.segment.rsc +1 -1
  294. package/.next/standalone/.next/server/app/terminal.segments/_index.segment.rsc +2 -2
  295. package/.next/standalone/.next/server/app/terminal.segments/_tree.segment.rsc +2 -2
  296. package/.next/standalone/.next/server/app/vr/page/react-loadable-manifest.json +1 -1
  297. package/.next/standalone/.next/server/app/vr/page_client-reference-manifest.js +1 -1
  298. package/.next/standalone/.next/server/app/vr.html +1 -1
  299. package/.next/standalone/.next/server/app/vr.rsc +3 -3
  300. package/.next/standalone/.next/server/app/vr.segments/_full.segment.rsc +3 -3
  301. package/.next/standalone/.next/server/app/vr.segments/_head.segment.rsc +1 -1
  302. package/.next/standalone/.next/server/app/vr.segments/_index.segment.rsc +2 -2
  303. package/.next/standalone/.next/server/app/vr.segments/_tree.segment.rsc +2 -2
  304. package/.next/standalone/.next/server/app/vr.segments/vr/__PAGE__.segment.rsc +2 -2
  305. package/.next/standalone/.next/server/app/vr.segments/vr.segment.rsc +1 -1
  306. package/.next/standalone/.next/server/app/workspaces.html +1 -1
  307. package/.next/standalone/.next/server/app/workspaces.rsc +2 -2
  308. package/.next/standalone/.next/server/app/workspaces.segments/!KGRlc2t0b3Ap/workspaces/__PAGE__.segment.rsc +1 -1
  309. package/.next/standalone/.next/server/app/workspaces.segments/!KGRlc2t0b3Ap/workspaces.segment.rsc +1 -1
  310. package/.next/standalone/.next/server/app/workspaces.segments/!KGRlc2t0b3Ap.segment.rsc +1 -1
  311. package/.next/standalone/.next/server/app/workspaces.segments/_full.segment.rsc +2 -2
  312. package/.next/standalone/.next/server/app/workspaces.segments/_head.segment.rsc +1 -1
  313. package/.next/standalone/.next/server/app/workspaces.segments/_index.segment.rsc +2 -2
  314. package/.next/standalone/.next/server/app/workspaces.segments/_tree.segment.rsc +2 -2
  315. package/.next/standalone/.next/server/app-paths-manifest.json +6 -0
  316. package/.next/standalone/.next/server/chunks/[root-of-the-server]__0041efe4._.js +2 -2
  317. package/.next/standalone/.next/server/chunks/[root-of-the-server]__00bf0ace._.js +2 -2
  318. package/.next/standalone/.next/server/chunks/[root-of-the-server]__0a837dd9._.js +98 -0
  319. package/.next/standalone/.next/server/chunks/[root-of-the-server]__0e71d908._.js +3 -3
  320. package/.next/standalone/.next/server/chunks/[root-of-the-server]__0e9142f3._.js +2 -2
  321. package/.next/standalone/.next/server/chunks/[root-of-the-server]__10e47926._.js +1 -1
  322. package/.next/standalone/.next/server/chunks/[root-of-the-server]__1665dc78._.js +2 -2
  323. package/.next/standalone/.next/server/chunks/[root-of-the-server]__175cbabf._.js +2 -2
  324. package/.next/standalone/.next/server/chunks/[root-of-the-server]__1adae357._.js +2 -2
  325. package/.next/standalone/.next/server/chunks/[root-of-the-server]__1d359752._.js +2 -2
  326. package/.next/standalone/.next/server/chunks/[root-of-the-server]__1e8fabeb._.js +2 -2
  327. package/.next/standalone/.next/server/chunks/[root-of-the-server]__1f8deca0._.js +8 -8
  328. package/.next/standalone/.next/server/chunks/[root-of-the-server]__253fdda1._.js +2 -2
  329. package/.next/standalone/.next/server/chunks/[root-of-the-server]__28e6434f._.js +2 -2
  330. package/.next/standalone/.next/server/chunks/[root-of-the-server]__2a386564._.js +3 -3
  331. package/.next/standalone/.next/server/chunks/[root-of-the-server]__2c20fb38._.js +2 -2
  332. package/.next/standalone/.next/server/chunks/[root-of-the-server]__309132cd._.js +1 -1
  333. package/.next/standalone/.next/server/chunks/[root-of-the-server]__33fec964._.js +3 -3
  334. package/.next/standalone/.next/server/chunks/[root-of-the-server]__3786d8ae._.js +1 -1
  335. package/.next/standalone/.next/server/chunks/[root-of-the-server]__3ae92407._.js +2 -2
  336. package/.next/standalone/.next/server/chunks/[root-of-the-server]__3beda9fe._.js +2 -2
  337. package/.next/standalone/.next/server/chunks/[root-of-the-server]__4619e9bd._.js +1 -1
  338. package/.next/standalone/.next/server/chunks/[root-of-the-server]__4a051043._.js +1 -1
  339. package/.next/standalone/.next/server/chunks/[root-of-the-server]__508002e4._.js +2 -2
  340. package/.next/standalone/.next/server/chunks/[root-of-the-server]__5086c373._.js +2 -2
  341. package/.next/standalone/.next/server/chunks/[root-of-the-server]__5913e097._.js +2 -2
  342. package/.next/standalone/.next/server/chunks/[root-of-the-server]__5b5f68d2._.js +2 -2
  343. package/.next/standalone/.next/server/chunks/[root-of-the-server]__5c1f2459._.js +2 -2
  344. package/.next/standalone/.next/server/chunks/[root-of-the-server]__5ec8c977._.js +2 -2
  345. package/.next/standalone/.next/server/chunks/[root-of-the-server]__63cebc6c._.js +2 -2
  346. package/.next/standalone/.next/server/chunks/[root-of-the-server]__64d30d4d._.js +2 -2
  347. package/.next/standalone/.next/server/chunks/[root-of-the-server]__6c54fc2e._.js +2 -2
  348. package/.next/standalone/.next/server/chunks/[root-of-the-server]__6dc1fb7e._.js +1 -1
  349. package/.next/standalone/.next/server/chunks/[root-of-the-server]__6e568102._.js +2 -2
  350. package/.next/standalone/.next/server/chunks/[root-of-the-server]__6faa04c0._.js +2 -2
  351. package/.next/standalone/.next/server/chunks/[root-of-the-server]__74a34dc3._.js +2 -2
  352. package/.next/standalone/.next/server/chunks/[root-of-the-server]__7819e4cf._.js +98 -0
  353. package/.next/standalone/.next/server/chunks/[root-of-the-server]__7e7250a4._.js +2 -2
  354. package/.next/standalone/.next/server/chunks/[root-of-the-server]__8309e0a4._.js +2 -2
  355. package/.next/standalone/.next/server/chunks/[root-of-the-server]__86cc0e2b._.js +6 -6
  356. package/.next/standalone/.next/server/chunks/[root-of-the-server]__89c2565a._.js +2 -2
  357. package/.next/standalone/.next/server/chunks/[root-of-the-server]__8d178ad9._.js +2 -2
  358. package/.next/standalone/.next/server/chunks/[root-of-the-server]__93ee06f3._.js +2 -2
  359. package/.next/standalone/.next/server/chunks/[root-of-the-server]__9e1f0137._.js +98 -0
  360. package/.next/standalone/.next/server/chunks/[root-of-the-server]__9e4c154a._.js +2 -2
  361. package/.next/standalone/.next/server/chunks/[root-of-the-server]__a9d2e1d3._.js +2 -2
  362. package/.next/standalone/.next/server/chunks/[root-of-the-server]__ae53d343._.js +2 -2
  363. package/.next/standalone/.next/server/chunks/[root-of-the-server]__b3a04cef._.js +2 -2
  364. package/.next/standalone/.next/server/chunks/[root-of-the-server]__b4270b77._.js +1 -1
  365. package/.next/standalone/.next/server/chunks/[root-of-the-server]__b6b6ce60._.js +3 -3
  366. package/.next/standalone/.next/server/chunks/[root-of-the-server]__c0757773._.js +98 -0
  367. package/.next/standalone/.next/server/chunks/[root-of-the-server]__c24cfa91._.js +3 -0
  368. package/.next/standalone/.next/server/chunks/{[root-of-the-server]__eb8acb65._.js → [root-of-the-server]__c7c47529._.js} +2 -2
  369. package/.next/standalone/.next/server/chunks/[root-of-the-server]__c88b63f7._.js +2 -2
  370. package/.next/standalone/.next/server/chunks/[root-of-the-server]__cba5f007._.js +1 -1
  371. package/.next/standalone/.next/server/chunks/[root-of-the-server]__cbf4ceb0._.js +2 -2
  372. package/.next/standalone/.next/server/chunks/[root-of-the-server]__cefdba2f._.js +2 -2
  373. package/.next/standalone/.next/server/chunks/[root-of-the-server]__cf9e82bb._.js +2 -2
  374. package/.next/standalone/.next/server/chunks/[root-of-the-server]__d2897392._.js +2 -2
  375. package/.next/standalone/.next/server/chunks/[root-of-the-server]__d3b2d856._.js +2 -2
  376. package/.next/standalone/.next/server/chunks/[root-of-the-server]__d73273ca._.js +2 -2
  377. package/.next/standalone/.next/server/chunks/[root-of-the-server]__d8417eb6._.js +2 -2
  378. package/.next/standalone/.next/server/chunks/[root-of-the-server]__dc2a55de._.js +2 -2
  379. package/.next/standalone/.next/server/chunks/[root-of-the-server]__e0d4690b._.js +2 -2
  380. package/.next/standalone/.next/server/chunks/[root-of-the-server]__e3ea8547._.js +98 -0
  381. package/.next/standalone/.next/server/chunks/[root-of-the-server]__e678dd53._.js +1 -1
  382. package/.next/standalone/.next/server/chunks/[root-of-the-server]__e9223f55._.js +2 -2
  383. package/.next/standalone/.next/server/chunks/[root-of-the-server]__ea630076._.js +2 -2
  384. package/.next/standalone/.next/server/chunks/[root-of-the-server]__f26ca49d._.js +1 -1
  385. package/.next/standalone/.next/server/chunks/[root-of-the-server]__f33e1101._.js +1 -1
  386. package/.next/standalone/.next/server/chunks/[root-of-the-server]__f515f865._.js +2 -2
  387. package/.next/standalone/.next/server/chunks/[root-of-the-server]__fceb5d60._.js +2 -2
  388. package/.next/standalone/.next/server/chunks/[root-of-the-server]__fed41403._.js +2 -2
  389. package/.next/standalone/.next/server/chunks/[root-of-the-server]__ff2e98c2._.js +2 -2
  390. package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_benchmark_lobes_route_actions_ea7beadb.js +3 -0
  391. package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_benchmark_run_route_actions_9ed0ba41.js +3 -0
  392. package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_benchmark_runs_[id]_route_actions_39f90307.js +3 -0
  393. package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_benchmark_runs_route_actions_37cf958b.js +3 -0
  394. package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_benchmark_status_route_actions_009e2cba.js +3 -0
  395. package/.next/standalone/.next/server/chunks/_next-internal_server_app_api_whisper_route_actions_be9a633d.js +3 -0
  396. package/.next/standalone/.next/server/chunks/ssr/{_c1cfdd09._.js → [root-of-the-server]__16621ac5._.js} +2 -2
  397. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__66aca5d4._.js +1 -1
  398. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__9c81bd86._.js +3 -0
  399. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__e3bf6054._.js +5 -0
  400. package/.next/standalone/.next/server/chunks/ssr/{_81abf587._.js → _31ada310._.js} +2 -2
  401. package/.next/standalone/.next/server/chunks/ssr/_648a8a2d._.js +3 -0
  402. package/.next/standalone/.next/server/chunks/ssr/_999dae61._.js +3 -0
  403. package/.next/standalone/.next/server/chunks/ssr/_c48c91e5._.js +3 -0
  404. package/.next/standalone/.next/server/chunks/ssr/_f8959434._.js +3 -0
  405. package/.next/standalone/.next/server/chunks/ssr/src_app_(desktop)_cortex_page_tsx_0f33d8b3._.js +1 -1
  406. package/.next/standalone/.next/server/chunks/ssr/src_components_terminal_terminal-pane_tsx_803c5e2c._.js +9 -0
  407. package/.next/standalone/.next/server/edge/chunks/[root-of-the-server]__32a0045c._.js +1 -1
  408. package/.next/standalone/.next/server/edge/chunks/_d73df637._.js +1 -1
  409. package/.next/standalone/.next/server/middleware-manifest.json +5 -5
  410. package/.next/standalone/.next/server/pages/404.html +1 -1
  411. package/.next/standalone/.next/server/pages/500.html +2 -2
  412. package/.next/standalone/.next/server/server-reference-manifest.js +1 -1
  413. package/.next/standalone/.next/server/server-reference-manifest.json +1 -1
  414. package/.next/standalone/.next/static/chunks/157826a3253f8ccf.js +7 -0
  415. package/.next/standalone/.next/static/chunks/{69606d281c39f9b2.js → 2f9f09924ff9d447.js} +1 -1
  416. package/.next/standalone/.next/static/chunks/{84fe8d44deeeb74f.js → 4ced66ef12d91b38.js} +35 -12
  417. package/.next/standalone/.next/static/chunks/57d04d161b8a01b3.js +7 -0
  418. package/.next/standalone/.next/static/chunks/709c4608e5b935e8.js +1 -0
  419. package/.next/standalone/.next/static/chunks/{9cfa0291d55d8d2a.js → 7b14d4e609b55b9b.js} +1 -1
  420. package/.next/standalone/.next/static/chunks/7be37f4a56b8f575.js +1 -0
  421. package/.next/standalone/.next/static/chunks/8021d5a2269ff113.js +1 -0
  422. package/.next/standalone/.next/static/chunks/a3fd93a9dde3cacc.js +1 -0
  423. package/.next/standalone/.next/static/chunks/{412140a02893327a.js → afee31c7399daf2a.js} +1 -1
  424. package/.next/standalone/.next/static/chunks/b92fdbf858aeb0b3.js +1 -0
  425. package/.next/standalone/.next/static/chunks/c17274c2f95d4ba2.js +5 -0
  426. package/.next/standalone/.next/static/chunks/e82f4414650587cf.js +7 -0
  427. package/.next/standalone/.next/static/chunks/f6464729e7aa0da0.css +3 -0
  428. package/.next/standalone/LICENSE +661 -0
  429. package/.next/standalone/NOTICE +5 -0
  430. package/.next/standalone/README.md +131 -0
  431. package/.next/standalone/bin/cortex-hook.sh +62 -62
  432. package/.next/standalone/bin/cortex-mcp.js +60 -60
  433. package/.next/standalone/bin/fix-standalone-externals.js +79 -0
  434. package/.next/standalone/bin/lib/auto-setup.js +110 -0
  435. package/.next/standalone/bin/mdns-service.js +171 -0
  436. package/.next/standalone/bin/postinstall.js +35 -0
  437. package/.next/standalone/bin/setup-admin.js +195 -0
  438. package/.next/standalone/bin/spaces-dev.js +247 -0
  439. package/.next/standalone/bin/spaces-install.js +638 -0
  440. package/.next/standalone/bin/spaces-reset-totp.js +50 -0
  441. package/.next/standalone/bin/spaces-service.js +1020 -0
  442. package/.next/standalone/bin/spaces-setup.js +253 -0
  443. package/.next/standalone/bin/spaces.js +788 -0
  444. package/.next/standalone/bin/ssh-auth-keys.sh +68 -0
  445. package/.next/standalone/bin/terminal-server.js +1807 -0
  446. package/.next/standalone/docker-compose.yml +28 -0
  447. package/.next/standalone/docs/architecture.md +387 -0
  448. package/.next/standalone/docs/cortex.md +293 -0
  449. package/.next/standalone/docs/getting-started.md +96 -0
  450. package/.next/standalone/docs/plans/2026-02-24-multi-agent-sessions-design.md +133 -0
  451. package/.next/standalone/docs/plans/2026-02-24-multi-agent-sessions-plan.md +959 -0
  452. package/.next/standalone/docs/plans/2026-03-07-service-command-design.md +146 -0
  453. package/.next/standalone/docs/plans/2026-03-07-service-command-plan.md +254 -0
  454. package/.next/standalone/docs/server-install.md +564 -0
  455. package/.next/standalone/docs/social-card.html +150 -0
  456. package/.next/standalone/docs/superpowers/plans/2026-03-12-spaces-cortex.md +5270 -0
  457. package/.next/standalone/docs/superpowers/plans/2026-03-13-cortex-wiring.md +1387 -1387
  458. package/.next/standalone/docs/superpowers/plans/2026-03-14-cortex-v2-entity-graph.md +1923 -1923
  459. package/.next/standalone/docs/superpowers/plans/2026-03-14-cortex-v2-knowledge-evolution.md +1113 -1113
  460. package/.next/standalone/docs/superpowers/plans/2026-03-15-cortex-v2-boundary-engine.md +853 -853
  461. package/.next/standalone/docs/superpowers/plans/2026-03-15-cortex-v2-context-engine.md +1274 -1274
  462. package/.next/standalone/docs/superpowers/plans/2026-03-15-cortex-v2-signal-ingestion.md +933 -933
  463. package/.next/standalone/docs/superpowers/plans/2026-03-16-cortex-lobes.md +1080 -1080
  464. package/.next/standalone/docs/superpowers/plans/2026-03-16-cortex-v2-gravity-system.md +768 -768
  465. package/.next/standalone/docs/superpowers/plans/2026-03-16-cortex-v2-ui.md +1108 -1108
  466. package/.next/standalone/docs/superpowers/plans/2026-03-18-cortex-ui-integration.md +1846 -1846
  467. package/.next/standalone/docs/superpowers/plans/2026-03-19-vr-phase1-shell.md +1639 -0
  468. package/.next/standalone/docs/superpowers/specs/2026-03-11-universe-view-design.md +320 -0
  469. package/.next/standalone/docs/superpowers/specs/2026-03-12-spaces-brain-design.md +720 -0
  470. package/.next/standalone/docs/superpowers/specs/2026-03-13-cortex-wiring-design.md +268 -268
  471. package/.next/standalone/docs/superpowers/specs/2026-03-14-cortex-v2-design.md +623 -623
  472. package/.next/standalone/docs/superpowers/specs/2026-03-16-cortex-lobes-design.md +263 -263
  473. package/.next/standalone/docs/superpowers/specs/2026-03-16-cortex-v2-ui-design.md +240 -240
  474. package/.next/standalone/docs/superpowers/specs/2026-03-16-pane-ux-design.md +77 -0
  475. package/.next/standalone/docs/superpowers/specs/2026-03-18-cortex-ui-integration-design.md +341 -341
  476. package/.next/standalone/docs/superpowers/specs/2026-03-19-vr-phase1-shell-design.md +288 -0
  477. package/.next/standalone/docs/tiers.md +104 -0
  478. package/.next/standalone/eslint.config.mjs +18 -0
  479. package/.next/standalone/next.config.ts +20 -0
  480. package/.next/standalone/nginx.conf +53 -0
  481. package/.next/standalone/node_modules/@img/sharp-libvips-linux-x64/README.md +46 -0
  482. package/.next/standalone/node_modules/@img/sharp-libvips-linux-x64/lib/glib-2.0/include/glibconfig.h +221 -0
  483. package/.next/standalone/node_modules/@img/sharp-libvips-linux-x64/lib/index.js +1 -0
  484. package/.next/standalone/node_modules/@img/sharp-libvips-linux-x64/lib/libvips-cpp.so.8.17.3 +0 -0
  485. package/.next/standalone/node_modules/@img/sharp-libvips-linux-x64/package.json +42 -0
  486. package/.next/standalone/node_modules/@img/sharp-libvips-linuxmusl-x64/README.md +46 -0
  487. package/.next/standalone/node_modules/@img/sharp-libvips-linuxmusl-x64/lib/glib-2.0/include/glibconfig.h +221 -0
  488. package/.next/standalone/node_modules/@img/sharp-libvips-linuxmusl-x64/lib/index.js +1 -0
  489. package/.next/standalone/node_modules/@img/sharp-libvips-linuxmusl-x64/lib/libvips-cpp.so.8.17.3 +0 -0
  490. package/.next/standalone/node_modules/@img/sharp-libvips-linuxmusl-x64/package.json +42 -0
  491. package/.next/standalone/node_modules/@img/sharp-libvips-linuxmusl-x64/versions.json +30 -0
  492. package/.next/standalone/node_modules/@img/sharp-linux-x64/lib/sharp-linux-x64.node +0 -0
  493. package/.next/standalone/node_modules/@img/{sharp-win32-x64 → sharp-linux-x64}/package.json +46 -39
  494. package/.next/standalone/node_modules/@img/sharp-linuxmusl-x64/lib/sharp-linuxmusl-x64.node +0 -0
  495. package/.next/standalone/node_modules/@img/sharp-linuxmusl-x64/package.json +46 -0
  496. package/.next/standalone/package-lock.json +14154 -0
  497. package/.next/standalone/package.json +104 -104
  498. package/.next/standalone/postcss.config.mjs +7 -0
  499. package/.next/standalone/scripts/rebuild.cmd +65 -0
  500. package/.next/standalone/scripts/rebuild.sh +59 -0
  501. package/.next/standalone/server.js +1 -1
  502. package/.next/standalone/src/app/(desktop)/admin/analytics/page.tsx +266 -0
  503. package/.next/standalone/src/app/(desktop)/admin/users/page.tsx +399 -0
  504. package/.next/standalone/src/app/(desktop)/analytics/page.tsx +166 -0
  505. package/.next/standalone/src/app/(desktop)/cortex/page.tsx +81 -78
  506. package/.next/standalone/src/app/(desktop)/dashboard-client.tsx +56 -0
  507. package/.next/standalone/src/app/(desktop)/layout.tsx +18 -0
  508. package/.next/standalone/src/app/(desktop)/network/page.tsx +137 -0
  509. package/.next/standalone/src/app/(desktop)/page.tsx +17 -0
  510. package/.next/standalone/src/app/(desktop)/projects/page.tsx +68 -0
  511. package/.next/standalone/src/app/(desktop)/sessions/[id]/page.tsx +519 -0
  512. package/.next/standalone/src/app/(desktop)/sessions/page.tsx +145 -0
  513. package/.next/standalone/src/app/(desktop)/settings/page.tsx +446 -0
  514. package/.next/standalone/src/app/(desktop)/terminal/layout.tsx +7 -0
  515. package/.next/standalone/src/app/(desktop)/terminal/page.tsx +1151 -0
  516. package/.next/standalone/src/app/(desktop)/terminal/pane/[id]/page.tsx +211 -0
  517. package/.next/standalone/src/app/(desktop)/terminal/remote/[nodeId]/[workspaceId]/page.tsx +192 -0
  518. package/.next/standalone/src/app/(desktop)/workspaces/page.tsx +12 -0
  519. package/.next/standalone/src/app/api/admin/analytics/route.ts +10 -0
  520. package/.next/standalone/src/app/api/admin/users/[id]/route.ts +20 -0
  521. package/.next/standalone/src/app/api/admin/users/route.ts +15 -0
  522. package/.next/standalone/src/app/api/analytics/overview/route.ts +80 -0
  523. package/.next/standalone/src/app/api/auth/login/route.ts +10 -0
  524. package/.next/standalone/src/app/api/auth/logout/route.ts +9 -0
  525. package/.next/standalone/src/app/api/auth/me/route.ts +22 -0
  526. package/.next/standalone/src/app/api/auth/totp/setup/route.ts +10 -0
  527. package/.next/standalone/src/app/api/auth/totp/status/route.ts +10 -0
  528. package/.next/standalone/src/app/api/auth/totp/verify/route.ts +10 -0
  529. package/.next/standalone/src/app/api/benchmark/lobes/route.ts +16 -0
  530. package/.next/standalone/src/app/api/benchmark/run/route.ts +92 -0
  531. package/.next/standalone/src/app/api/benchmark/runs/[id]/route.ts +26 -0
  532. package/.next/standalone/src/app/api/benchmark/runs/route.ts +16 -0
  533. package/.next/standalone/src/app/api/benchmark/status/route.ts +35 -0
  534. package/.next/standalone/src/app/api/bulk/route.ts +34 -0
  535. package/.next/standalone/src/app/api/chat/route.ts +85 -0
  536. package/.next/standalone/src/app/api/config/route.ts +30 -0
  537. package/.next/standalone/src/app/api/cortex/context/route.ts +78 -78
  538. package/.next/standalone/src/app/api/cortex/curation/assess/route.ts +27 -27
  539. package/.next/standalone/src/app/api/cortex/curation/publish/route.ts +23 -23
  540. package/.next/standalone/src/app/api/cortex/curation/refine/route.ts +23 -23
  541. package/.next/standalone/src/app/api/cortex/curation/review/route.ts +29 -29
  542. package/.next/standalone/src/app/api/cortex/curation/seed/route.ts +23 -23
  543. package/.next/standalone/src/app/api/cortex/export/route.ts +40 -40
  544. package/.next/standalone/src/app/api/cortex/federation/pending/route.ts +20 -20
  545. package/.next/standalone/src/app/api/cortex/federation/resolve/route.ts +43 -43
  546. package/.next/standalone/src/app/api/cortex/federation/search/route.ts +35 -35
  547. package/.next/standalone/src/app/api/cortex/federation/teach/route.ts +76 -76
  548. package/.next/standalone/src/app/api/cortex/graph/edges/route.ts +112 -112
  549. package/.next/standalone/src/app/api/cortex/graph/entities/[id]/route.ts +73 -73
  550. package/.next/standalone/src/app/api/cortex/graph/entities/route.ts +75 -75
  551. package/.next/standalone/src/app/api/cortex/graph/populate/route.ts +203 -203
  552. package/.next/standalone/src/app/api/cortex/import/route.ts +75 -75
  553. package/.next/standalone/src/app/api/cortex/import/status/route.ts +15 -15
  554. package/.next/standalone/src/app/api/cortex/ingest/bootstrap/route.ts +29 -29
  555. package/.next/standalone/src/app/api/cortex/ingest/status/route.ts +15 -15
  556. package/.next/standalone/src/app/api/cortex/knowledge/[id]/route.ts +91 -91
  557. package/.next/standalone/src/app/api/cortex/knowledge/route.ts +93 -93
  558. package/.next/standalone/src/app/api/cortex/lobes/[id]/route.ts +67 -67
  559. package/.next/standalone/src/app/api/cortex/lobes/route.ts +22 -22
  560. package/.next/standalone/src/app/api/cortex/lobes/share/route.ts +80 -80
  561. package/.next/standalone/src/app/api/cortex/marketplace/browse/route.ts +43 -43
  562. package/.next/standalone/src/app/api/cortex/marketplace/preview/route.ts +46 -46
  563. package/.next/standalone/src/app/api/cortex/mcp/call/route.ts +11 -11
  564. package/.next/standalone/src/app/api/cortex/mcp/tools/route.ts +6 -6
  565. package/.next/standalone/src/app/api/cortex/search/route.ts +43 -43
  566. package/.next/standalone/src/app/api/cortex/settings/route.ts +33 -33
  567. package/.next/standalone/src/app/api/cortex/status/route.ts +169 -169
  568. package/.next/standalone/src/app/api/cortex/timeline/route.ts +42 -42
  569. package/.next/standalone/src/app/api/cortex/usage/route.ts +31 -31
  570. package/.next/standalone/src/app/api/cortex/workspace/[id]/context/route.ts +41 -41
  571. package/.next/standalone/src/app/api/events/route.ts +40 -0
  572. package/.next/standalone/src/app/api/files/route.ts +187 -0
  573. package/.next/standalone/src/app/api/folders/route.ts +97 -0
  574. package/.next/standalone/src/app/api/network/connect-callback/route.ts +11 -0
  575. package/.next/standalone/src/app/api/network/connect-request/[id]/route.ts +11 -0
  576. package/.next/standalone/src/app/api/network/connect-request/route.ts +17 -0
  577. package/.next/standalone/src/app/api/network/discovered/route.ts +9 -0
  578. package/.next/standalone/src/app/api/network/handshake/route.ts +25 -0
  579. package/.next/standalone/src/app/api/network/health/route.ts +10 -0
  580. package/.next/standalone/src/app/api/network/identity/route.ts +15 -0
  581. package/.next/standalone/src/app/api/network/keys/[id]/route.ts +10 -0
  582. package/.next/standalone/src/app/api/network/keys/route.ts +15 -0
  583. package/.next/standalone/src/app/api/network/nodes/[id]/route.ts +15 -0
  584. package/.next/standalone/src/app/api/network/nodes/check/route.ts +9 -0
  585. package/.next/standalone/src/app/api/network/nodes/route.ts +15 -0
  586. package/.next/standalone/src/app/api/network/projects/route.ts +25 -0
  587. package/.next/standalone/src/app/api/network/proxy/[nodeId]/[...path]/route.ts +15 -0
  588. package/.next/standalone/src/app/api/network/search/route.ts +38 -0
  589. package/.next/standalone/src/app/api/network/sessions/[id]/messages/route.ts +36 -0
  590. package/.next/standalone/src/app/api/network/sessions/[id]/route.ts +34 -0
  591. package/.next/standalone/src/app/api/network/sessions/route.ts +43 -0
  592. package/.next/standalone/src/app/api/network/terminal/token/route.ts +10 -0
  593. package/.next/standalone/src/app/api/network/workspaces/[id]/route.ts +34 -0
  594. package/.next/standalone/src/app/api/network/workspaces/route.ts +61 -0
  595. package/.next/standalone/src/app/api/panes/[id]/route.ts +60 -0
  596. package/.next/standalone/src/app/api/panes/route.ts +39 -0
  597. package/.next/standalone/src/app/api/projects/route.ts +13 -0
  598. package/.next/standalone/src/app/api/search/route.ts +47 -0
  599. package/.next/standalone/src/app/api/sessions/[id]/chat/route.ts +120 -0
  600. package/.next/standalone/src/app/api/sessions/[id]/messages/route.ts +28 -0
  601. package/.next/standalone/src/app/api/sessions/[id]/route.ts +73 -0
  602. package/.next/standalone/src/app/api/sessions/route.ts +64 -0
  603. package/.next/standalone/src/app/api/sync/route.ts +24 -0
  604. package/.next/standalone/src/app/api/tags/route.ts +35 -0
  605. package/.next/standalone/src/app/api/tier/route.ts +16 -0
  606. package/.next/standalone/src/app/api/updates/route.ts +53 -0
  607. package/.next/standalone/src/app/api/whisper/route.ts +90 -0
  608. package/.next/standalone/src/app/api/workspaces/[id]/context/[key]/route.ts +39 -0
  609. package/.next/standalone/src/app/api/workspaces/[id]/context/route.ts +28 -0
  610. package/.next/standalone/src/app/api/workspaces/[id]/messages/[msgId]/route.ts +17 -0
  611. package/.next/standalone/src/app/api/workspaces/[id]/messages/route.ts +39 -0
  612. package/.next/standalone/src/app/api/workspaces/[id]/route.ts +47 -0
  613. package/.next/standalone/src/app/api/workspaces/[id]/sessions/route.ts +62 -0
  614. package/.next/standalone/src/app/api/workspaces/route.ts +79 -0
  615. package/.next/standalone/src/app/globals.css +85 -0
  616. package/.next/standalone/src/app/icon.png +0 -0
  617. package/.next/standalone/src/app/layout.tsx +33 -0
  618. package/.next/standalone/src/app/login/layout.tsx +7 -0
  619. package/.next/standalone/src/app/login/page.tsx +315 -0
  620. package/.next/standalone/src/app/m/layout.tsx +16 -0
  621. package/.next/standalone/src/app/m/page.tsx +118 -0
  622. package/.next/standalone/src/app/m/projects/page.tsx +64 -0
  623. package/.next/standalone/src/app/m/sessions/[id]/page.tsx +168 -0
  624. package/.next/standalone/src/app/m/sessions/page.tsx +177 -0
  625. package/.next/standalone/src/app/m/settings/page.tsx +230 -0
  626. package/.next/standalone/src/app/m/terminal/page.tsx +413 -0
  627. package/.next/standalone/src/app/vr/page.tsx +21 -0
  628. package/.next/standalone/src/app/vr/vr-app.tsx +163 -0
  629. package/.next/standalone/src/app/vr/vr-controls.tsx +139 -0
  630. package/.next/standalone/src/app/vr/vr-door.tsx +82 -0
  631. package/.next/standalone/src/app/vr/vr-environment.tsx +71 -0
  632. package/.next/standalone/src/app/vr/vr-gaze.tsx +89 -0
  633. package/.next/standalone/src/app/vr/vr-layout.ts +49 -0
  634. package/.next/standalone/src/app/vr/vr-lobby.tsx +97 -0
  635. package/.next/standalone/src/app/vr/vr-pane.tsx +195 -0
  636. package/.next/standalone/src/app/vr/vr-room.tsx +79 -0
  637. package/.next/standalone/src/app/vr/vr-terminal.tsx +303 -0
  638. package/.next/standalone/src/components/auth/totp-gate.tsx +183 -0
  639. package/.next/standalone/src/components/bus/activity-panel.tsx +261 -0
  640. package/.next/standalone/src/components/common/color-picker.tsx +35 -0
  641. package/.next/standalone/src/components/common/dev-directory-picker.tsx +339 -0
  642. package/.next/standalone/src/components/common/folder-picker.tsx +200 -0
  643. package/.next/standalone/src/components/common/tag-picker.tsx +190 -0
  644. package/.next/standalone/src/components/common/workspace-picker.tsx +113 -0
  645. package/.next/standalone/src/components/cortex/benchmark-tab.tsx +880 -0
  646. package/.next/standalone/src/components/cortex/constants.ts +29 -29
  647. package/.next/standalone/src/components/cortex/cortex-dashboard.tsx +304 -304
  648. package/.next/standalone/src/components/cortex/cortex-indicator.tsx +44 -44
  649. package/.next/standalone/src/components/cortex/cortex-panel.tsx +140 -140
  650. package/.next/standalone/src/components/cortex/cortex-settings.tsx +221 -221
  651. package/.next/standalone/src/components/cortex/curation-tab.tsx +810 -810
  652. package/.next/standalone/src/components/cortex/entity-detail.tsx +101 -101
  653. package/.next/standalone/src/components/cortex/entity-graph.tsx +382 -382
  654. package/.next/standalone/src/components/cortex/import-dialog.tsx +212 -212
  655. package/.next/standalone/src/components/cortex/injection-badge.tsx +72 -72
  656. package/.next/standalone/src/components/cortex/knowledge-card.tsx +109 -109
  657. package/.next/standalone/src/components/cortex/knowledge-tab.tsx +158 -158
  658. package/.next/standalone/src/components/cortex/lobe-settings.tsx +215 -215
  659. package/.next/standalone/src/components/cortex/marketplace-card.tsx +126 -126
  660. package/.next/standalone/src/components/cortex/marketplace-tab.tsx +113 -113
  661. package/.next/standalone/src/components/dashboard/activity-chart.tsx +41 -0
  662. package/.next/standalone/src/components/dashboard/model-usage-chart.tsx +61 -0
  663. package/.next/standalone/src/components/dashboard/recent-sessions.tsx +68 -0
  664. package/.next/standalone/src/components/dashboard/stats-cards.tsx +36 -0
  665. package/.next/standalone/src/components/files/file-explorer.tsx +703 -0
  666. package/.next/standalone/src/components/layout/providers.tsx +38 -0
  667. package/.next/standalone/src/components/layout/sidebar.tsx +170 -0
  668. package/.next/standalone/src/components/layout/tier-provider.tsx +53 -0
  669. package/.next/standalone/src/components/layout/update-banner.tsx +92 -0
  670. package/.next/standalone/src/components/mobile/bottom-nav.tsx +46 -0
  671. package/.next/standalone/src/components/mobile/mobile-chat-input.tsx +244 -0
  672. package/.next/standalone/src/components/mobile/mobile-header.tsx +44 -0
  673. package/.next/standalone/src/components/mobile/mobile-session-card.tsx +56 -0
  674. package/.next/standalone/src/components/mobile/mobile-terminal-input.tsx +71 -0
  675. package/.next/standalone/src/components/mobile/mobile-terminal-pane.tsx +300 -0
  676. package/.next/standalone/src/components/mobile/mobile-terminal-toolbar.tsx +53 -0
  677. package/.next/standalone/src/components/mobile/pull-to-refresh.tsx +82 -0
  678. package/.next/standalone/src/components/mobile/voice-input.tsx +53 -0
  679. package/.next/standalone/src/components/network/api-key-list.tsx +190 -0
  680. package/.next/standalone/src/components/network/connection-requests.tsx +94 -0
  681. package/.next/standalone/src/components/network/node-add-dialog.tsx +131 -0
  682. package/.next/standalone/src/components/network/node-badge.tsx +26 -0
  683. package/.next/standalone/src/components/network/node-list.tsx +207 -0
  684. package/.next/standalone/src/components/network/node-selector.tsx +49 -0
  685. package/.next/standalone/src/components/sessions/session-filters.tsx +116 -0
  686. package/.next/standalone/src/components/sessions/session-list.tsx +485 -0
  687. package/.next/standalone/src/components/terminal/terminal-pane.tsx +874 -0
  688. package/.next/standalone/src/components/viewer/chat-input.tsx +275 -0
  689. package/.next/standalone/src/components/viewer/message-renderer.tsx +551 -0
  690. package/.next/standalone/src/components/workspace/universe-cluster.tsx +131 -0
  691. package/.next/standalone/src/components/workspace/universe-orb.tsx +128 -0
  692. package/.next/standalone/src/components/workspace/universe-types.ts +22 -0
  693. package/.next/standalone/src/components/workspace/universe-utils.ts +11 -0
  694. package/.next/standalone/src/components/workspace/universe-view.tsx +397 -0
  695. package/.next/standalone/src/components/workspace/workspace-chooser.tsx +616 -0
  696. package/.next/standalone/src/hooks/use-benchmark.ts +71 -0
  697. package/.next/standalone/src/hooks/use-bus.ts +147 -0
  698. package/.next/standalone/src/hooks/use-idle-detection.ts +79 -0
  699. package/.next/standalone/src/hooks/use-network.ts +229 -0
  700. package/.next/standalone/src/hooks/use-sessions.ts +437 -0
  701. package/.next/standalone/src/hooks/use-speech-recognition.ts +113 -0
  702. package/.next/standalone/src/hooks/use-sse.ts +35 -0
  703. package/.next/standalone/src/hooks/use-tier.ts +39 -0
  704. package/.next/standalone/src/lib/agents.ts +70 -0
  705. package/.next/standalone/src/lib/aider/parser.ts +111 -0
  706. package/.next/standalone/src/lib/api.ts +19 -0
  707. package/.next/standalone/src/lib/auth.ts +47 -0
  708. package/.next/standalone/src/lib/claude/parser.ts +212 -0
  709. package/.next/standalone/src/lib/claude/stats.ts +204 -0
  710. package/.next/standalone/src/lib/codex/parser.ts +265 -0
  711. package/.next/standalone/src/lib/config.ts +115 -0
  712. package/.next/standalone/src/lib/cortex/benchmark.ts +67 -0
  713. package/.next/standalone/src/lib/cortex/config.ts +40 -40
  714. package/.next/standalone/src/lib/cortex/debug.ts +10 -10
  715. package/.next/standalone/src/lib/cortex/distillation/usage-store.ts +18 -18
  716. package/.next/standalone/src/lib/cortex/graph/resolver.ts +10 -10
  717. package/.next/standalone/src/lib/cortex/graph/types.ts +22 -22
  718. package/.next/standalone/src/lib/cortex/index.ts +56 -56
  719. package/.next/standalone/src/lib/cortex/ingestion/bootstrap.ts +14 -14
  720. package/.next/standalone/src/lib/cortex/knowledge/compat.ts +14 -14
  721. package/.next/standalone/src/lib/cortex/knowledge/contradiction.ts +10 -10
  722. package/.next/standalone/src/lib/cortex/knowledge/types.ts +67 -67
  723. package/.next/standalone/src/lib/cortex/lobes/config.ts +16 -16
  724. package/.next/standalone/src/lib/cortex/lobes/resolver.ts +8 -8
  725. package/.next/standalone/src/lib/cortex/lobes/shares.ts +14 -14
  726. package/.next/standalone/src/lib/cortex/mcp/server.ts +8 -8
  727. package/.next/standalone/src/lib/cortex/portability/exporter.ts +6 -6
  728. package/.next/standalone/src/lib/cortex/portability/importer.ts +10 -10
  729. package/.next/standalone/src/lib/cortex/retrieval/context-engine.ts +10 -10
  730. package/.next/standalone/src/lib/cortex/types.ts +39 -39
  731. package/.next/standalone/src/lib/cost-calculator.ts +48 -0
  732. package/.next/standalone/src/lib/db/init.ts +71 -0
  733. package/.next/standalone/src/lib/db/queries.ts +718 -0
  734. package/.next/standalone/src/lib/db/schema.ts +202 -0
  735. package/.next/standalone/src/lib/events/sse.ts +36 -0
  736. package/.next/standalone/src/lib/gemini/parser.ts +216 -0
  737. package/.next/standalone/src/lib/license.ts +56 -0
  738. package/.next/standalone/src/lib/pro.ts +31 -0
  739. package/.next/standalone/src/lib/sync/indexer.ts +429 -0
  740. package/.next/standalone/src/lib/sync/watcher.ts +64 -0
  741. package/.next/standalone/src/lib/teams.ts +31 -0
  742. package/.next/standalone/src/lib/telemetry.ts +75 -0
  743. package/.next/standalone/src/lib/terminal/server.ts +128 -0
  744. package/.next/standalone/src/lib/tier.ts +38 -0
  745. package/.next/standalone/src/lib/utils.ts +72 -0
  746. package/.next/standalone/src/middleware.ts +133 -0
  747. package/.next/standalone/src/types/claude.ts +208 -0
  748. package/.next/standalone/src/types/network.ts +61 -0
  749. package/.next/standalone/tests/setup.ts +8 -0
  750. package/.next/standalone/tsconfig.json +34 -34
  751. package/.next/standalone/vitest.config.ts +24 -0
  752. package/LICENSE +661 -661
  753. package/README.md +131 -131
  754. package/bin/cortex-hook.sh +62 -62
  755. package/bin/cortex-mcp.js +60 -60
  756. package/bin/fix-standalone-externals.js +79 -79
  757. package/bin/lib/auto-setup.js +110 -110
  758. package/bin/mdns-service.js +171 -171
  759. package/bin/postinstall.js +35 -35
  760. package/bin/setup-admin.js +195 -195
  761. package/bin/spaces-dev.js +247 -208
  762. package/bin/spaces-install.js +638 -599
  763. package/bin/spaces-reset-totp.js +50 -50
  764. package/bin/spaces-service.js +1020 -1020
  765. package/bin/spaces-setup.js +253 -253
  766. package/bin/spaces.js +788 -776
  767. package/bin/ssh-auth-keys.sh +68 -68
  768. package/bin/terminal-server.js +1807 -1683
  769. package/package.json +104 -104
  770. package/.next/standalone/.claude/settings.local.json +0 -55
  771. package/.next/standalone/.claude/spaces-env.json +0 -1
  772. package/.next/standalone/.next/server/chunks/ssr/[root-of-the-server]__e921fdfc._.js +0 -5
  773. package/.next/standalone/.next/server/chunks/ssr/_2230ad2d._.js +0 -3
  774. package/.next/standalone/.next/server/chunks/ssr/_5cf334fd._.js +0 -3
  775. package/.next/standalone/.next/server/chunks/ssr/_db0abd0a._.js +0 -3
  776. package/.next/standalone/.next/server/chunks/ssr/_db2fec84._.js +0 -3
  777. package/.next/standalone/.next/server/chunks/ssr/_f4e57187._.js +0 -3
  778. package/.next/standalone/.next/server/chunks/ssr/src_40fa36ce._.js +0 -7
  779. package/.next/standalone/.next/static/chunks/003e7aa1adfe577d.js +0 -1
  780. package/.next/standalone/.next/static/chunks/158b52b84e647ac1.js +0 -5
  781. package/.next/standalone/.next/static/chunks/232d8aae4fefab70.js +0 -1
  782. package/.next/standalone/.next/static/chunks/5325351ef49cb65f.js +0 -1
  783. package/.next/standalone/.next/static/chunks/559735e598ca3cbb.js +0 -1
  784. package/.next/standalone/.next/static/chunks/5d5d7b0095dd52ae.js +0 -1
  785. package/.next/standalone/.next/static/chunks/898f380eba90427a.js +0 -1
  786. package/.next/standalone/.next/static/chunks/95339e55722bb4ca.js +0 -5
  787. package/.next/standalone/.next/static/chunks/9cd594813c539df9.js +0 -1
  788. package/.next/standalone/.next/static/chunks/c1a95aebf6725f64.css +0 -3
  789. package/.next/standalone/.next/static/chunks/c515eb77d9410aa0.js +0 -5
  790. package/.next/standalone/.next/static/chunks/d9ae203a7f123546.js +0 -5
  791. package/.next/standalone/.next/static/chunks/f9f2628207848ac2.js +0 -1
  792. package/.next/standalone/.spaces/cortex-context.md +0 -70
  793. package/.next/standalone/node_modules/@img/sharp-win32-x64/lib/sharp-win32-x64.node +0 -0
  794. /package/.next/standalone/.next/static/{BEY-sql3lQLouidpurSQf → PcpzUspSK8QDdwzAJz8br}/_buildManifest.js +0 -0
  795. /package/.next/standalone/.next/static/{BEY-sql3lQLouidpurSQf → PcpzUspSK8QDdwzAJz8br}/_clientMiddlewareManifest.json +0 -0
  796. /package/.next/standalone/.next/static/{BEY-sql3lQLouidpurSQf → PcpzUspSK8QDdwzAJz8br}/_ssgManifest.js +0 -0
  797. /package/.next/standalone/node_modules/@img/{sharp-win32-x64 → sharp-libvips-linux-x64}/versions.json +0 -0
package/bin/spaces.js CHANGED
@@ -1,776 +1,788 @@
1
- #!/usr/bin/env node
2
-
3
- // Re-exec with higher heap if needed (embedding models + LanceDB leak memory over time)
4
- if (!process.execArgv.some(a => a.includes('max-old-space-size')) && !process.env.__SPACES_HEAP) {
5
- process.env.__SPACES_HEAP = '1';
6
- require('child_process').spawn(
7
- process.execPath,
8
- ['--max-old-space-size=8192', ...process.argv.slice(1)],
9
- { stdio: 'inherit', env: process.env }
10
- ).on('exit', (code) => process.exit(code ?? 1));
11
- // Stop the outer process from continuing
12
- return;
13
- }
14
-
15
- const net = require('net');
16
- const http = require('http');
17
- const https = require('https');
18
- const { execFileSync, spawn } = require('child_process');
19
- const path = require('path');
20
- const os = require('os');
21
- const fs = require('fs');
22
- // terminal-server is loaded lazily (after SPACES_TIER is set in process.env)
23
-
24
- // ─── Memory monitoring (only when SPACES_DEBUG or cortex.debug) ──
25
- let _lastHeapMB = 0;
26
- function logMemory(label) {
27
- if (!process.env.SPACES_DEBUG && !_spacesDebug) return;
28
- const mem = process.memoryUsage();
29
- const heapMB = Math.round(mem.heapUsed / 1048576);
30
- const rssMB = Math.round(mem.rss / 1048576);
31
- const extMB = Math.round((mem.external || 0) / 1048576);
32
- const abMB = Math.round((mem.arrayBuffers || 0) / 1048576);
33
- const delta = heapMB - _lastHeapMB;
34
- if (label || Math.abs(delta) > 20) {
35
- console.log(`[Memory] ${label || 'periodic'}: heap=${heapMB}MB rss=${rssMB}MB external=${extMB}MB arrayBuffers=${abMB}MB ${delta > 0 ? '+' : ''}${delta}MB`);
36
- }
37
- _lastHeapMB = heapMB;
38
- }
39
- let _spacesDebug = false;
40
- try {
41
- const cfgPath = path.join(os.homedir(), '.spaces', 'config.json');
42
- if (fs.existsSync(cfgPath)) _spacesDebug = !!JSON.parse(fs.readFileSync(cfgPath, 'utf-8')).cortex?.debug;
43
- } catch { /* ignore */ }
44
- setInterval(() => logMemory(), 60000);
45
-
46
- const SPACES_DIR = path.join(os.homedir(), '.spaces');
47
- const CONFIG_PATH = path.join(SPACES_DIR, 'server.json');
48
- const SESSION_SECRET_PATH = path.join(SPACES_DIR, 'session_secret');
49
- const NEXT_INTERNAL_PORT = 3400;
50
- const projectDir = path.join(__dirname, '..');
51
- const MANAGED_PACKAGES = path.join(SPACES_DIR, 'packages');
52
- const MANAGED_NODE_MODULES = path.join(MANAGED_PACKAGES, 'node_modules');
53
-
54
- // ─── CLI arg parsing ──────────────────────────────────────
55
- const args = process.argv.slice(2);
56
- const cliFlags = {};
57
- for (let i = 0; i < args.length; i++) {
58
- if (args[i] === '--setup') { cliFlags.setup = true; }
59
- else if (args[i] === '--port' && args[i + 1]) { cliFlags.port = parseInt(args[++i], 10); }
60
- else if (args[i] === '--tier' && args[i + 1]) { cliFlags.tier = args[++i]; }
61
- else if (args[i] === '--base-path' && args[i + 1]) { cliFlags.basePath = args[++i]; }
62
- else if (args[i] === '--help' || args[i] === '-h') { cliFlags.help = true; }
63
- }
64
-
65
- if (cliFlags.help) {
66
- console.log(`
67
- Spaces - Agent Workspace Manager
68
-
69
- Usage:
70
- spaces Start the server (auto-detects tier)
71
- spaces stop Stop running server
72
- spaces install <teams|pro> Install a tier package
73
- spaces uninstall [teams|pro] Uninstall packages (all if none specified)
74
- spaces verify Verify installed packages
75
- spaces upgrade [teams|pro] Upgrade installed packages
76
- spaces --setup Interactive first-time setup wizard
77
- spaces --port 3457 Override port
78
- spaces --tier team Override tier (community|team|federation)
79
- spaces --base-path /spaces Set base path for reverse proxy
80
- spaces service install Install as OS service (systemd/launchd/Task Scheduler)
81
- spaces service uninstall Remove OS service
82
- spaces service start Start the service
83
- spaces service stop Stop the service
84
- spaces service status Show service status
85
- spaces service logs Tail service logs
86
- spaces reset-totp <user> Reset TOTP/2FA for a user
87
- spaces --help Show this help
88
- `);
89
- process.exit(0);
90
- }
91
-
92
- // ─── Stop command ─────────────────────────────────────────
93
- const subcommand = args[0];
94
- if (subcommand === 'stop') {
95
- stopServer();
96
- process.exit(0);
97
- }
98
-
99
- // ─── Route install/verify/upgrade to spaces-install.js ────
100
- if (subcommand === 'install' || subcommand === 'uninstall' || subcommand === 'verify' || subcommand === 'upgrade') {
101
- // Re-exec with spaces-install.js, passing through all args
102
- const installScript = path.join(__dirname, 'spaces-install.js');
103
- const { status } = require('child_process').spawnSync(
104
- process.execPath, [installScript, ...args],
105
- { stdio: 'inherit', env: process.env }
106
- );
107
- process.exit(status || 0);
108
- }
109
-
110
- // ─── Route reset-totp to spaces-reset-totp.js ───────
111
- if (subcommand === 'reset-totp') {
112
- const resetScript = path.join(__dirname, 'spaces-reset-totp.js');
113
- const { status } = require('child_process').spawnSync(
114
- process.execPath, [resetScript, ...args.slice(1)],
115
- { stdio: 'inherit', env: process.env }
116
- );
117
- process.exit(status || 0);
118
- }
119
-
120
- // ─── Route service commands to spaces-service.js ─────
121
- if (subcommand === 'service') {
122
- const serviceScript = path.join(__dirname, 'spaces-service.js');
123
- const { status } = require('child_process').spawnSync(
124
- process.execPath, [serviceScript, ...args.slice(1)],
125
- { stdio: 'inherit', env: process.env }
126
- );
127
- process.exit(status || 0);
128
- }
129
-
130
- // ─── Setup wizard ─────────────────────────────────────────
131
- if (cliFlags.setup) {
132
- require('./spaces-setup');
133
- // spaces-setup handles its own process.exit
134
- } else if (subcommand && !subcommand.startsWith('-')) {
135
- console.error(` Unknown command: ${subcommand}`);
136
- console.error(` Run "spaces --help" for usage`);
137
- process.exit(1);
138
- } else {
139
- startServer();
140
- }
141
-
142
- // ─── TLS certificate ────────────────────────────────────────
143
- // Uses a real cert if configured in server.json (tlsCert, tlsKey, optional
144
- // tlsCa for the full chain), otherwise generates a self-signed cert so
145
- // mobile browsers that force HTTPS-first can still connect.
146
- function ensureTlsCert() {
147
- // Check for configured real cert in server.json
148
- try {
149
- const config = JSON.parse(fs.readFileSync(CONFIG_PATH, 'utf-8'));
150
- if (config.tlsCert && config.tlsKey) {
151
- if (fs.existsSync(config.tlsCert) && fs.existsSync(config.tlsKey)) {
152
- // Read cert + key, and optionally the CA bundle for full chain
153
- const certPem = fs.readFileSync(config.tlsCert, 'utf-8').trim();
154
- const caPem = (config.tlsCa && fs.existsSync(config.tlsCa))
155
- ? fs.readFileSync(config.tlsCa, 'utf-8').trim()
156
- : null;
157
- const creds = {
158
- cert: caPem ? certPem + '\n' + caPem : certPem,
159
- key: fs.readFileSync(config.tlsKey, 'utf-8'),
160
- };
161
- console.log(' Using TLS certificate: ' + config.tlsCert);
162
- return creds;
163
- }
164
- console.log(' Warning: tlsCert/tlsKey configured but files not found, falling back to self-signed');
165
- }
166
- } catch {}
167
-
168
- // Fall back to self-signed
169
- const certPath = path.join(SPACES_DIR, 'tls-cert.pem');
170
- const keyPath = path.join(SPACES_DIR, 'tls-key.pem');
171
-
172
- if (fs.existsSync(certPath) && fs.existsSync(keyPath)) {
173
- try {
174
- return { cert: fs.readFileSync(certPath), key: fs.readFileSync(keyPath) };
175
- } catch {}
176
- }
177
-
178
- try {
179
- fs.mkdirSync(SPACES_DIR, { recursive: true });
180
- execFileSync('openssl', [
181
- 'req', '-x509', '-newkey', 'rsa:2048',
182
- '-keyout', keyPath, '-out', certPath,
183
- '-sha256', '-days', '365', '-nodes',
184
- '-subj', '/CN=spaces-local',
185
- '-addext', 'subjectAltName=DNS:localhost,DNS:*.local,DNS:*.robindale.com,IP:127.0.0.1',
186
- ], { stdio: 'pipe', timeout: 15000 });
187
- console.log(' Generated self-signed TLS certificate');
188
- return { cert: fs.readFileSync(certPath), key: fs.readFileSync(keyPath) };
189
- } catch (e) {
190
- console.log(` Warning: Could not generate TLS cert (${e.message})`);
191
- console.log(' HTTPS will not be available. Install OpenSSL to enable it.');
192
- return null;
193
- }
194
- }
195
-
196
- function startServer() {
197
- // ─── Load saved config ────────────────────────────────────
198
- let savedConfig = {};
199
- if (fs.existsSync(CONFIG_PATH)) {
200
- try { savedConfig = JSON.parse(fs.readFileSync(CONFIG_PATH, 'utf-8')); } catch {}
201
- }
202
-
203
- // ─── Resolve settings (CLI > env > config > defaults) ─────
204
- const PORT = cliFlags.port
205
- || parseInt(process.env.SPACES_PORT || '', 10)
206
- || savedConfig.port
207
- || 3457;
208
-
209
- const basePath = cliFlags.basePath
210
- || process.env.SPACES_BASE_PATH
211
- || savedConfig.basePath
212
- || '';
213
-
214
- const allowedOrigins = process.env.SPACES_ALLOWED_ORIGINS
215
- || savedConfig.allowedOrigins
216
- || '';
217
-
218
- // ─── Kill any existing server on this port ───────────────────
219
- stopServer();
220
-
221
- // ─── Resolve optional packages once ─────────────────────────
222
- const proPath = resolveSpacesPro();
223
- const teamsPath = resolveSpacesTeams();
224
- const cortexPath = resolveSpacesCortex();
225
-
226
- // Tier resolution: CLI > env > config > auto-detect
227
- let tier = cliFlags.tier
228
- || process.env.SPACES_TIER
229
- || savedConfig.tier
230
- || '';
231
-
232
- // Auto-detect tier from package presence
233
- if (!tier) {
234
- if (proPath) tier = 'federation';
235
- else if (teamsPath) tier = 'team';
236
- else tier = 'community';
237
- }
238
-
239
- console.log('');
240
- console.log(' Spaces - Agent Workspace Manager');
241
- console.log(' =================================');
242
- console.log('');
243
- console.log(` Tier: ${tier}`);
244
-
245
- // ─── Server tier prerequisites ────────────────────────────
246
- const childEnv = { ...process.env };
247
-
248
- if (tier !== 'community') {
249
- // team/federation tiers require @spaces/teams (auth + collaboration)
250
- if ((tier === 'team' || tier === 'federation') && !teamsPath) {
251
- console.error(' Error: @spaces/teams is required for team/federation tiers.');
252
- console.error(' Install it: spaces install teams');
253
- process.exit(1);
254
- }
255
- // federation tier requires @spaces/pro (network)
256
- if (tier === 'federation' && !proPath) {
257
- console.error(' Error: @spaces/pro is required for the federation tier.');
258
- console.error(' Install it: spaces install pro');
259
- process.exit(1);
260
- }
261
-
262
- // Check for session secret
263
- let sessionSecret = process.env.SPACES_SESSION_SECRET || '';
264
- if (!sessionSecret && fs.existsSync(SESSION_SECRET_PATH)) {
265
- sessionSecret = fs.readFileSync(SESSION_SECRET_PATH, 'utf-8').trim();
266
- }
267
-
268
- // Check for admin DB auto-setup if missing
269
- const adminDbPath = path.join(SPACES_DIR, 'admin.db');
270
- if (!fs.existsSync(adminDbPath) || !sessionSecret) {
271
- const { autoSetup } = require('./lib/auto-setup');
272
- const result = autoSetup({ SPACES_DIR, SESSION_SECRET_PATH, ADMIN_DB_PATH: adminDbPath, CONFIG_PATH, tier, port: PORT, basePath });
273
- sessionSecret = result.sessionSecret;
274
- }
275
-
276
- childEnv.SPACES_SESSION_SECRET = sessionSecret;
277
- process.env.SPACES_SESSION_SECRET = sessionSecret;
278
- console.log(` Admin DB: ${adminDbPath}`);
279
- }
280
-
281
- // Set tier in both child env and own process env (terminal-server reads it)
282
- childEnv.SPACES_TIER = tier;
283
- process.env.SPACES_TIER = tier;
284
- if (basePath) {
285
- childEnv.SPACES_BASE_PATH = basePath;
286
- console.log(` Base path: ${basePath}`);
287
- }
288
- if (allowedOrigins) {
289
- childEnv.SPACES_ALLOWED_ORIGINS = allowedOrigins;
290
- process.env.SPACES_ALLOWED_ORIGINS = allowedOrigins;
291
- console.log(` Allowed origins: ${allowedOrigins}`);
292
- }
293
-
294
- // ─── Resolve NODE_PATH for @spaces/pro and @spaces/teams ──
295
- // Update both childEnv (for the Next.js child process) and process.env
296
- // (for terminal-server.js which runs in this same parent process).
297
- // Include:
298
- // - Managed packages node_modules (~/.spaces/packages/node_modules)
299
- // - Each managed package's own node_modules (for its bundled deps)
300
- // - The host app's own node_modules (fallback for peer deps)
301
- const appNodeModules = path.join(projectDir, 'node_modules');
302
- const nodePaths = [MANAGED_NODE_MODULES, appNodeModules];
303
- // Add each resolved package's own node_modules so its bundled deps are found
304
- for (const pkgPath of [proPath, teamsPath, cortexPath]) {
305
- if (pkgPath) {
306
- // pkgPath is the package root (symlink target or direct path)
307
- const realPath = fs.realpathSync(pkgPath);
308
- const pkgNodeModules = path.join(realPath, 'node_modules');
309
- if (fs.existsSync(pkgNodeModules) && !nodePaths.includes(pkgNodeModules)) {
310
- nodePaths.push(pkgNodeModules);
311
- }
312
- // Also add the parent scope for @spaces/* resolution
313
- const parentModules = path.dirname(path.dirname(pkgPath));
314
- if (!nodePaths.includes(parentModules)) {
315
- nodePaths.push(parentModules);
316
- }
317
- }
318
- }
319
- for (const dir of nodePaths) {
320
- for (const target of [childEnv, process.env]) {
321
- const existing = target.NODE_PATH || '';
322
- if (!existing.includes(dir)) {
323
- target.NODE_PATH = existing
324
- ? `${dir}${path.delimiter}${existing}`
325
- : dir;
326
- }
327
- }
328
- }
329
- // Re-init module paths so require() in this process picks up the new NODE_PATH
330
- require('module').Module._initPaths();
331
-
332
- // Verify collaboration pipeline works end-to-end at startup
333
- if (tier === 'team' || tier === 'federation') {
334
- try {
335
- const teams = require('@spaces/teams');
336
- console.log(' Collaboration: @spaces/teams loaded OK');
337
- // Smoke-test that peer deps resolve from this process
338
- require('better-sqlite3');
339
- } catch (e) {
340
- console.error(` Warning: Collaboration may not work — ${e.message}`);
341
- console.error(' Check that NODE_PATH includes the host app node_modules.');
342
- }
343
- }
344
-
345
- // ─── Detect @spaces/cortex ──────────────────────────────────
346
- if (cortexPath) {
347
- childEnv.SPACES_HAS_CORTEX = '1';
348
- process.env.SPACES_HAS_CORTEX = '1';
349
- console.log(' Cortex: @spaces/cortex detected');
350
- }
351
-
352
- console.log('');
353
-
354
- // ─── Background update check (non-blocking, repeats every 4h) ──
355
- // Checks npm registry, GitHub releases, and addon git repos.
356
- // Writes results to ~/.spaces/update-check.json for the UI banner.
357
- const updateCheckPath = path.join(SPACES_DIR, 'update-check.json');
358
- const checkForUpdates = async (logIfAvailable = true) => {
359
- try {
360
- const pkg = require(path.join(projectDir, 'package.json'));
361
- const currentVersion = pkg.version;
362
- const npmName = pkg.name || '@jlongo78/agent-spaces';
363
- const githubRepo = 'jlongo78/spaces';
364
-
365
- // Check npm registry
366
- let npmLatest = null;
367
- try {
368
- const res = await fetch(`https://registry.npmjs.org/${encodeURIComponent(npmName)}/latest`, {
369
- signal: AbortSignal.timeout(5000),
370
- });
371
- if (res.ok) {
372
- const data = await res.json();
373
- npmLatest = data.version;
374
- }
375
- } catch { /* */ }
376
-
377
- // Check GitHub for latest release or tag (may be ahead of npm)
378
- let githubLatest = null;
379
- let githubPrerelease = false;
380
- let githubUrl = null;
381
- try {
382
- // Check releases first (includes pre-releases)
383
- const res = await fetch(`https://api.github.com/repos/${githubRepo}/releases?per_page=5`, {
384
- signal: AbortSignal.timeout(5000),
385
- headers: { 'Accept': 'application/vnd.github.v3+json', 'User-Agent': 'agent-spaces' },
386
- });
387
- if (res.ok) {
388
- const releases = await res.json();
389
- if (releases.length > 0) {
390
- // Find latest stable and latest pre-release
391
- const stable = releases.find(r => !r.prerelease && !r.draft);
392
- const pre = releases.find(r => r.prerelease && !r.draft);
393
- if (stable) {
394
- githubLatest = stable.tag_name.replace(/^v/, '');
395
- githubUrl = stable.html_url;
396
- }
397
- if (pre) {
398
- const preVersion = pre.tag_name.replace(/^v/, '');
399
- // If pre-release is newer than stable, surface it
400
- if (!githubLatest || preVersion > githubLatest) {
401
- githubLatest = preVersion;
402
- githubPrerelease = true;
403
- githubUrl = pre.html_url;
404
- }
405
- }
406
- }
407
- }
408
- } catch { /* */ }
409
-
410
- // Check addon git repos
411
- const gitSafe = ['-c', 'safe.directory=*'];
412
- const addonPaths = { teams: teamsPath, pro: proPath, cortex: cortexPath };
413
- const addons = {};
414
- for (const [key, addonDir] of Object.entries(addonPaths)) {
415
- if (!addonDir || !fs.existsSync(path.join(addonDir, '.git'))) continue;
416
- const realDir = fs.realpathSync(addonDir);
417
- try {
418
- execFileSync('git', [...gitSafe, 'fetch', '--quiet'], { cwd: realDir, stdio: 'ignore', timeout: 5000 });
419
- const local = execFileSync('git', [...gitSafe, 'rev-parse', 'HEAD'], { cwd: realDir, encoding: 'utf-8' }).trim();
420
- const remote = execFileSync('git', [...gitSafe, 'rev-parse', '@{u}'], { cwd: realDir, encoding: 'utf-8' }).trim();
421
- let commitsBehind = 0;
422
- if (local !== remote) {
423
- try {
424
- const count = execFileSync('git', [...gitSafe, 'rev-list', '--count', `HEAD..@{u}`], { cwd: realDir, encoding: 'utf-8' }).trim();
425
- commitsBehind = parseInt(count, 10) || 0;
426
- } catch { /* */ }
427
- }
428
- const behind = commitsBehind > 0;
429
- addons[key] = { installed: true, behind, commitsBehind, localHead: local.slice(0, 7) };
430
- if (behind && logIfAvailable) {
431
- console.log(` Update available for @spaces/${key}: ${commitsBehind} commit(s) behind (run: spaces upgrade ${key})`);
432
- }
433
- } catch {
434
- addons[key] = { installed: true, behind: false, commitsBehind: 0 };
435
- }
436
- }
437
-
438
- const npmAvailable = npmLatest && npmLatest !== currentVersion && npmLatest > currentVersion;
439
- const githubAvailable = githubLatest && githubLatest !== currentVersion && githubLatest > currentVersion;
440
-
441
- const result = {
442
- current: currentVersion,
443
- npm: npmLatest ? { latest: npmLatest, available: !!npmAvailable } : null,
444
- github: githubLatest ? { latest: githubLatest, prerelease: githubPrerelease, url: githubUrl, available: !!githubAvailable } : null,
445
- // Backward compat: 'available' and 'latest' still work for the existing banner
446
- latest: npmLatest || githubLatest || currentVersion,
447
- available: !!(npmAvailable || githubAvailable),
448
- addons,
449
- checkedAt: Date.now(),
450
- name: npmName,
451
- };
452
-
453
- fs.writeFileSync(updateCheckPath, JSON.stringify(result, null, 2));
454
-
455
- if (npmAvailable && logIfAvailable) {
456
- console.log(` Update available: ${currentVersion} ${npmLatest} (run: npm i -g ${npmName})`);
457
- }
458
- if (githubAvailable && (!npmAvailable || githubLatest > npmLatest) && logIfAvailable) {
459
- console.log(` GitHub ${githubPrerelease ? 'pre-release' : 'release'}: ${githubLatest} (${githubUrl || `https://github.com/${githubRepo}/releases`})`);
460
- }
461
- } catch { /* network error — skip silently */ }
462
- };
463
- checkForUpdates(true);
464
- setInterval(() => checkForUpdates(false), 4 * 3600_000);
465
-
466
- // Check for ~/.claude/ directory
467
- const claudeDir = path.join(os.homedir(), '.claude');
468
- if (!fs.existsSync(claudeDir)) {
469
- console.log(' Warning: ~/.claude/ not found. Have you used Claude Code yet?');
470
- console.log('');
471
- }
472
-
473
- // ─── Detect build type ────────────────────────────────────
474
- const standaloneServer = path.join(projectDir, '.next', 'standalone', 'server.js');
475
- const fullBuildDir = path.join(projectDir, '.next', 'BUILD_ID');
476
- const isStandalone = fs.existsSync(standaloneServer);
477
- const isFullBuild = fs.existsSync(fullBuildDir);
478
-
479
- if (!isStandalone && !isFullBuild) {
480
- console.error(' Error: No build found.');
481
- console.error(' Run "npm run build" first, or install via "npm install -g @jlongo78/agent-spaces".');
482
- process.exit(1);
483
- }
484
-
485
- // ─── Spawn Next.js ────────────────────────────────────────
486
- let next;
487
- if (isStandalone) {
488
- const parentNodeModules = path.join(projectDir, 'node_modules');
489
- const existingNodePath = childEnv.NODE_PATH || '';
490
- childEnv.NODE_PATH = existingNodePath
491
- ? `${parentNodeModules}${path.delimiter}${existingNodePath}`
492
- : parentNodeModules;
493
-
494
- next = spawn(process.execPath, [standaloneServer], {
495
- cwd: path.dirname(standaloneServer),
496
- stdio: ['ignore', 'pipe', 'pipe'],
497
- env: {
498
- ...childEnv,
499
- PORT: String(NEXT_INTERNAL_PORT),
500
- HOSTNAME: '0.0.0.0',
501
- NODE_ENV: 'production',
502
- },
503
- });
504
- } else {
505
- next = spawn('npx', ['next', 'start', '--port', String(NEXT_INTERNAL_PORT)], {
506
- cwd: projectDir,
507
- stdio: ['ignore', 'pipe', 'pipe'],
508
- env: {
509
- ...childEnv,
510
- PORT: String(NEXT_INTERNAL_PORT),
511
- HOSTNAME: '0.0.0.0',
512
- NODE_ENV: 'production',
513
- },
514
- shell: true,
515
- });
516
- }
517
-
518
- let nextReady = false;
519
-
520
- next.stdout.on('data', (data) => {
521
- const msg = data.toString();
522
- process.stdout.write(msg);
523
- if (!nextReady && (msg.includes('Ready') || msg.includes('started server') || msg.includes('Listening') || msg.includes('localhost'))) {
524
- nextReady = true;
525
- console.log(`\n Ready at http://localhost:${PORT}`);
526
- if (tlsCreds) console.log(` Also at https://localhost:${PORT} (self-signed)`);
527
- console.log('');
528
-
529
- const url = `http://localhost:${PORT}`;
530
- const isService = process.env.SPACES_SERVICE === '1';
531
- if (!isService) {
532
- try {
533
- if (process.platform === 'win32') {
534
- execFileSync('cmd', ['/c', 'start', url], { stdio: 'ignore' });
535
- } else if (process.platform === 'darwin') {
536
- execFileSync('open', [url], { stdio: 'ignore' });
537
- } else {
538
- execFileSync('xdg-open', [url], { stdio: 'ignore' });
539
- }
540
- } catch {
541
- console.log(` Open ${url} in your browser`);
542
- }
543
- }
544
- }
545
- });
546
-
547
- next.stderr.on('data', (data) => {
548
- const msg = data.toString();
549
- if (!msg.includes('Warning')) {
550
- process.stderr.write(data);
551
- }
552
- });
553
-
554
- next.on('exit', (code) => {
555
- if (code !== 0 && code !== null) {
556
- console.error(` [spaces] Next.js exited with code ${code}`);
557
- }
558
- cleanup();
559
- });
560
-
561
- // ─── Proxy request handler (shared by HTTP and HTTPS) ────
562
- function proxyHandler(req, res) {
563
- // WebSocket paths are handled by the 'upgrade' event, but a plain
564
- // HTTP request to /ws (e.g. health check) should not be proxied to
565
- // Next.js which would 308-redirect it due to trailingSlash.
566
- const urlPath = (req.url || '').split('?')[0];
567
- if (urlPath === '/ws' || urlPath.endsWith('/ws') || urlPath.endsWith('/ws/')) {
568
- res.writeHead(200, { 'Content-Type': 'text/plain' });
569
- res.end('WebSocket endpoint');
570
- return;
571
- }
572
-
573
- const proxyReq = http.request(
574
- {
575
- hostname: '127.0.0.1',
576
- port: NEXT_INTERNAL_PORT,
577
- path: req.url,
578
- method: req.method,
579
- headers: req.headers,
580
- },
581
- (proxyRes) => {
582
- res.writeHead(proxyRes.statusCode, proxyRes.headers);
583
- proxyRes.pipe(res);
584
- }
585
- );
586
- proxyReq.on('error', () => {
587
- if (!res.headersSent) {
588
- res.writeHead(502);
589
- res.end('Server not ready yet. Refresh in a moment.');
590
- }
591
- });
592
- req.pipe(proxyReq);
593
- }
594
-
595
- // ─── HTTP + HTTPS on a single port ─────────────────────────
596
- // Peek at the first byte of each connection: 0x16 = TLS ClientHello,
597
- // anything else = plain HTTP. This lets mobile browsers that force
598
- // HTTPS-first connect to the same port without a separate listener.
599
- const httpServer = http.createServer(proxyHandler);
600
- const tlsCreds = ensureTlsCert();
601
- const httpsServer = tlsCreds ? https.createServer(tlsCreds, proxyHandler) : null;
602
-
603
- const { createTerminalServer } = require('./terminal-server');
604
- createTerminalServer(httpServer);
605
- if (httpsServer) createTerminalServer(httpsServer);
606
-
607
- // Wire up 'address()' on the inner servers so terminal-server can
608
- // discover the port, then emit 'listening' so its event handler fires.
609
- const patchAddress = (srv) => { srv.address = () => ({ port: PORT }); };
610
- patchAddress(httpServer);
611
- if (httpsServer) patchAddress(httpsServer);
612
-
613
- const dualServer = net.createServer((socket) => {
614
- // If client connects but sends nothing for 5s, destroy.
615
- // Once routed, clear the timeout so long-lived connections (WebSocket) aren't killed.
616
- socket.setTimeout(5000, () => socket.destroy());
617
- socket.once('data', (buf) => {
618
- socket.setTimeout(0);
619
- socket.pause();
620
- const target = (buf[0] === 0x16 && httpsServer) ? httpsServer : httpServer;
621
- target.emit('connection', socket);
622
- socket.unshift(buf);
623
- socket.resume();
624
- });
625
- });
626
-
627
- dualServer.listen(PORT, () => {
628
- // Now that the port is bound, tell the inner servers they're "listening"
629
- // so terminal-server's mDNS and message watcher start up.
630
- httpServer.emit('listening');
631
- if (httpsServer) httpsServer.emit('listening');
632
-
633
- if (httpsServer) {
634
- console.log(` Starting server on port ${PORT} (HTTP + HTTPS)`);
635
- } else {
636
- console.log(` Starting server on http://localhost:${PORT}`);
637
- }
638
- console.log('');
639
- });
640
-
641
- function cleanup() {
642
- next.kill();
643
- dualServer.close();
644
- process.exit(0);
645
- }
646
-
647
- process.on('SIGINT', cleanup);
648
- process.on('SIGTERM', cleanup);
649
- }
650
-
651
- // ─── Stop running server ─────────────────────────────────────
652
- function findPidsOnPort(port) {
653
- const pids = new Set();
654
- if (process.platform === 'win32') {
655
- try {
656
- const output = execFileSync('netstat', ['-ano'], { encoding: 'utf-8' });
657
- for (const line of output.split(String.fromCharCode(10))) {
658
- if (line.includes(':' + port + ' ') && line.includes('LISTENING')) {
659
- const parts = line.trim().split(/\s+/);
660
- const pid = parseInt(parts[parts.length - 1], 10);
661
- if (pid > 0) pids.add(pid);
662
- }
663
- }
664
- } catch {}
665
- } else {
666
- try {
667
- const output = execFileSync('lsof', ['-ti', `tcp:${port}`], { encoding: 'utf-8' }).trim();
668
- for (const p of output.split('\n')) { if (p.trim()) pids.add(parseInt(p.trim(), 10)); }
669
- } catch {}
670
- try {
671
- const output = execFileSync('fuser', [`${port}/tcp`], { encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] }).trim();
672
- for (const p of output.split(/\s+/)) { if (p.trim()) pids.add(parseInt(p.trim(), 10)); }
673
- } catch {}
674
- }
675
- return [...pids].filter(p => !isNaN(p) && p > 0);
676
- }
677
-
678
- function stopServer() {
679
- let savedConfig = {};
680
- if (fs.existsSync(CONFIG_PATH)) {
681
- try { savedConfig = JSON.parse(fs.readFileSync(CONFIG_PATH, 'utf-8')); } catch {}
682
- }
683
- const port = parseInt(process.env.SPACES_PORT || '', 10) || savedConfig.port || 3457;
684
- const ports = [port, NEXT_INTERNAL_PORT];
685
- let killed = 0;
686
-
687
- for (const p of ports) {
688
- for (const pid of findPidsOnPort(p)) {
689
- try {
690
- process.kill(pid, 'SIGTERM');
691
- killed++;
692
- } catch {}
693
- }
694
- }
695
-
696
- if (killed > 0) {
697
- console.log(` Stopped ${killed} process(es) on port ${port}`);
698
- // Wait for ports to be released, escalate to SIGKILL if needed
699
- const deadline = Date.now() + 2000;
700
- let escalated = false;
701
- while (Date.now() < deadline) {
702
- const remaining = ports.flatMap(p => findPidsOnPort(p));
703
- if (remaining.length === 0) break;
704
- if (!escalated && Date.now() > deadline - 1000) {
705
- for (const pid of remaining) {
706
- try { process.kill(pid, 'SIGKILL'); } catch {}
707
- }
708
- escalated = true;
709
- }
710
- if (process.platform === 'win32') { require('child_process').spawnSync('powershell', ['-Command', 'Start-Sleep -Milliseconds 200']); } else { require('child_process').spawnSync('sleep', ['0.2']); }
711
- }
712
- } else {
713
- console.log(` No running server found on port ${port}`);
714
- }
715
- }
716
-
717
- // ─── @spaces/pro resolution ──────────────────────────────────
718
- function resolveSpacesPro() {
719
- // 1. Managed install (~/.spaces/packages/)
720
- const managed = path.join(MANAGED_NODE_MODULES, '@spaces', 'pro');
721
- if (fs.existsSync(path.join(managed, 'dist', 'index.js'))) return managed;
722
-
723
- // 2. Local node_modules (npm link / optionalDep)
724
- try {
725
- return require.resolve('@spaces/pro');
726
- } catch {}
727
-
728
- // 3. Global npm prefix (legacy)
729
- try {
730
- const globalPrefix = execFileSync('npm', ['prefix', '-g'], { encoding: 'utf-8' }).trim();
731
- const globalProPath = path.join(globalPrefix, 'lib', 'node_modules', '@spaces', 'pro');
732
- if (fs.existsSync(globalProPath)) return globalProPath;
733
- const altPath = path.join(globalPrefix, 'node_modules', '@spaces', 'pro');
734
- if (fs.existsSync(altPath)) return altPath;
735
- } catch {}
736
-
737
- return null;
738
- }
739
-
740
- // ─── @spaces/teams resolution ────────────────────────────────
741
- function resolveSpacesTeams() {
742
- // 1. Managed install (~/.spaces/packages/)
743
- const managed = path.join(MANAGED_NODE_MODULES, '@spaces', 'teams');
744
- if (fs.existsSync(path.join(managed, 'dist', 'index.js'))) return managed;
745
-
746
- // 2. Local node_modules (npm link / optionalDep)
747
- try {
748
- return require.resolve('@spaces/teams');
749
- } catch {}
750
-
751
- // 3. Global npm prefix (legacy)
752
- try {
753
- const globalPrefix = execFileSync('npm', ['prefix', '-g'], { encoding: 'utf-8' }).trim();
754
- const globalTeamsPath = path.join(globalPrefix, 'lib', 'node_modules', '@spaces', 'teams');
755
- if (fs.existsSync(globalTeamsPath)) return globalTeamsPath;
756
- const altPath = path.join(globalPrefix, 'node_modules', '@spaces', 'teams');
757
- if (fs.existsSync(altPath)) return altPath;
758
- } catch {}
759
-
760
- return null;
761
- }
762
-
763
- // ─── @spaces/cortex resolution ───────────────────────────────
764
- function resolveSpacesCortex() {
765
- const managed = path.join(MANAGED_NODE_MODULES, '@spaces', 'cortex');
766
- if (fs.existsSync(path.join(managed, 'dist', 'index.js'))) return managed;
767
- try { return require.resolve('@spaces/cortex'); } catch {}
768
- try {
769
- const globalPrefix = execFileSync('npm', ['prefix', '-g'], { encoding: 'utf-8' }).trim();
770
- const globalPath = path.join(globalPrefix, 'lib', 'node_modules', '@spaces', 'cortex');
771
- if (fs.existsSync(globalPath)) return globalPath;
772
- const altPath = path.join(globalPrefix, 'node_modules', '@spaces', 'cortex');
773
- if (fs.existsSync(altPath)) return altPath;
774
- } catch {}
775
- return null;
776
- }
1
+ #!/usr/bin/env node
2
+
3
+ // Re-exec with higher heap if needed (embedding models + LanceDB leak memory over time)
4
+ if (!process.execArgv.some(a => a.includes('max-old-space-size')) && !process.env.__SPACES_HEAP) {
5
+ process.env.__SPACES_HEAP = '1';
6
+ require('child_process').spawn(
7
+ process.execPath,
8
+ ['--max-old-space-size=8192', ...process.argv.slice(1)],
9
+ { stdio: 'inherit', env: process.env }
10
+ ).on('exit', (code) => process.exit(code ?? 1));
11
+ // Stop the outer process from continuing
12
+ return;
13
+ }
14
+
15
+ const net = require('net');
16
+ const http = require('http');
17
+ const https = require('https');
18
+ const { execFileSync, spawn } = require('child_process');
19
+ const path = require('path');
20
+ const os = require('os');
21
+ const fs = require('fs');
22
+
23
+ // Load .env.local into process.env so Next.js API routes see the keys
24
+ const _envLocalPath = path.join(__dirname, '..', '.env.local');
25
+ try {
26
+ if (fs.existsSync(_envLocalPath)) {
27
+ for (const line of fs.readFileSync(_envLocalPath, 'utf-8').split('\n')) {
28
+ const m = line.match(/^([A-Z_][A-Z0-9_]*)=(.+)$/);
29
+ if (m && !process.env[m[1]]) process.env[m[1]] = m[2].trim();
30
+ }
31
+ }
32
+ } catch {}
33
+
34
+ // terminal-server is loaded lazily (after SPACES_TIER is set in process.env)
35
+
36
+ // ─── Memory monitoring (only when SPACES_DEBUG or cortex.debug) ──
37
+ let _lastHeapMB = 0;
38
+ function logMemory(label) {
39
+ if (!process.env.SPACES_DEBUG && !_spacesDebug) return;
40
+ const mem = process.memoryUsage();
41
+ const heapMB = Math.round(mem.heapUsed / 1048576);
42
+ const rssMB = Math.round(mem.rss / 1048576);
43
+ const extMB = Math.round((mem.external || 0) / 1048576);
44
+ const abMB = Math.round((mem.arrayBuffers || 0) / 1048576);
45
+ const delta = heapMB - _lastHeapMB;
46
+ if (label || Math.abs(delta) > 20) {
47
+ console.log(`[Memory] ${label || 'periodic'}: heap=${heapMB}MB rss=${rssMB}MB external=${extMB}MB arrayBuffers=${abMB}MB ${delta > 0 ? '+' : ''}${delta}MB`);
48
+ }
49
+ _lastHeapMB = heapMB;
50
+ }
51
+ let _spacesDebug = false;
52
+ try {
53
+ const cfgPath = path.join(os.homedir(), '.spaces', 'config.json');
54
+ if (fs.existsSync(cfgPath)) _spacesDebug = !!JSON.parse(fs.readFileSync(cfgPath, 'utf-8')).cortex?.debug;
55
+ } catch { /* ignore */ }
56
+ setInterval(() => logMemory(), 60000);
57
+
58
+ const SPACES_DIR = path.join(os.homedir(), '.spaces');
59
+ const CONFIG_PATH = path.join(SPACES_DIR, 'server.json');
60
+ const SESSION_SECRET_PATH = path.join(SPACES_DIR, 'session_secret');
61
+ const NEXT_INTERNAL_PORT = 3400;
62
+ const projectDir = path.join(__dirname, '..');
63
+ const MANAGED_PACKAGES = path.join(SPACES_DIR, 'packages');
64
+ const MANAGED_NODE_MODULES = path.join(MANAGED_PACKAGES, 'node_modules');
65
+
66
+ // ─── CLI arg parsing ──────────────────────────────────────
67
+ const args = process.argv.slice(2);
68
+ const cliFlags = {};
69
+ for (let i = 0; i < args.length; i++) {
70
+ if (args[i] === '--setup') { cliFlags.setup = true; }
71
+ else if (args[i] === '--port' && args[i + 1]) { cliFlags.port = parseInt(args[++i], 10); }
72
+ else if (args[i] === '--tier' && args[i + 1]) { cliFlags.tier = args[++i]; }
73
+ else if (args[i] === '--base-path' && args[i + 1]) { cliFlags.basePath = args[++i]; }
74
+ else if (args[i] === '--help' || args[i] === '-h') { cliFlags.help = true; }
75
+ }
76
+
77
+ if (cliFlags.help) {
78
+ console.log(`
79
+ Spaces - Agent Workspace Manager
80
+
81
+ Usage:
82
+ spaces Start the server (auto-detects tier)
83
+ spaces stop Stop running server
84
+ spaces install <teams|pro> Install a tier package
85
+ spaces uninstall [teams|pro] Uninstall packages (all if none specified)
86
+ spaces verify Verify installed packages
87
+ spaces upgrade [teams|pro] Upgrade installed packages
88
+ spaces --setup Interactive first-time setup wizard
89
+ spaces --port 3457 Override port
90
+ spaces --tier team Override tier (community|team|federation)
91
+ spaces --base-path /spaces Set base path for reverse proxy
92
+ spaces service install Install as OS service (systemd/launchd/Task Scheduler)
93
+ spaces service uninstall Remove OS service
94
+ spaces service start Start the service
95
+ spaces service stop Stop the service
96
+ spaces service status Show service status
97
+ spaces service logs Tail service logs
98
+ spaces reset-totp <user> Reset TOTP/2FA for a user
99
+ spaces --help Show this help
100
+ `);
101
+ process.exit(0);
102
+ }
103
+
104
+ // ─── Stop command ─────────────────────────────────────────
105
+ const subcommand = args[0];
106
+ if (subcommand === 'stop') {
107
+ stopServer();
108
+ process.exit(0);
109
+ }
110
+
111
+ // ─── Route install/verify/upgrade to spaces-install.js ────
112
+ if (subcommand === 'install' || subcommand === 'uninstall' || subcommand === 'verify' || subcommand === 'upgrade') {
113
+ // Re-exec with spaces-install.js, passing through all args
114
+ const installScript = path.join(__dirname, 'spaces-install.js');
115
+ const { status } = require('child_process').spawnSync(
116
+ process.execPath, [installScript, ...args],
117
+ { stdio: 'inherit', env: process.env }
118
+ );
119
+ process.exit(status || 0);
120
+ }
121
+
122
+ // ─── Route reset-totp to spaces-reset-totp.js ───────
123
+ if (subcommand === 'reset-totp') {
124
+ const resetScript = path.join(__dirname, 'spaces-reset-totp.js');
125
+ const { status } = require('child_process').spawnSync(
126
+ process.execPath, [resetScript, ...args.slice(1)],
127
+ { stdio: 'inherit', env: process.env }
128
+ );
129
+ process.exit(status || 0);
130
+ }
131
+
132
+ // ─── Route service commands to spaces-service.js ─────
133
+ if (subcommand === 'service') {
134
+ const serviceScript = path.join(__dirname, 'spaces-service.js');
135
+ const { status } = require('child_process').spawnSync(
136
+ process.execPath, [serviceScript, ...args.slice(1)],
137
+ { stdio: 'inherit', env: process.env }
138
+ );
139
+ process.exit(status || 0);
140
+ }
141
+
142
+ // ─── Setup wizard ─────────────────────────────────────────
143
+ if (cliFlags.setup) {
144
+ require('./spaces-setup');
145
+ // spaces-setup handles its own process.exit
146
+ } else if (subcommand && !subcommand.startsWith('-')) {
147
+ console.error(` Unknown command: ${subcommand}`);
148
+ console.error(` Run "spaces --help" for usage`);
149
+ process.exit(1);
150
+ } else {
151
+ startServer();
152
+ }
153
+
154
+ // ─── TLS certificate ────────────────────────────────────────
155
+ // Uses a real cert if configured in server.json (tlsCert, tlsKey, optional
156
+ // tlsCa for the full chain), otherwise generates a self-signed cert so
157
+ // mobile browsers that force HTTPS-first can still connect.
158
+ function ensureTlsCert() {
159
+ // Check for configured real cert in server.json
160
+ try {
161
+ const config = JSON.parse(fs.readFileSync(CONFIG_PATH, 'utf-8'));
162
+ if (config.tlsCert && config.tlsKey) {
163
+ if (fs.existsSync(config.tlsCert) && fs.existsSync(config.tlsKey)) {
164
+ // Read cert + key, and optionally the CA bundle for full chain
165
+ const certPem = fs.readFileSync(config.tlsCert, 'utf-8').trim();
166
+ const caPem = (config.tlsCa && fs.existsSync(config.tlsCa))
167
+ ? fs.readFileSync(config.tlsCa, 'utf-8').trim()
168
+ : null;
169
+ const creds = {
170
+ cert: caPem ? certPem + '\n' + caPem : certPem,
171
+ key: fs.readFileSync(config.tlsKey, 'utf-8'),
172
+ };
173
+ console.log(' Using TLS certificate: ' + config.tlsCert);
174
+ return creds;
175
+ }
176
+ console.log(' Warning: tlsCert/tlsKey configured but files not found, falling back to self-signed');
177
+ }
178
+ } catch {}
179
+
180
+ // Fall back to self-signed
181
+ const certPath = path.join(SPACES_DIR, 'tls-cert.pem');
182
+ const keyPath = path.join(SPACES_DIR, 'tls-key.pem');
183
+
184
+ if (fs.existsSync(certPath) && fs.existsSync(keyPath)) {
185
+ try {
186
+ return { cert: fs.readFileSync(certPath), key: fs.readFileSync(keyPath) };
187
+ } catch {}
188
+ }
189
+
190
+ try {
191
+ fs.mkdirSync(SPACES_DIR, { recursive: true });
192
+ execFileSync('openssl', [
193
+ 'req', '-x509', '-newkey', 'rsa:2048',
194
+ '-keyout', keyPath, '-out', certPath,
195
+ '-sha256', '-days', '365', '-nodes',
196
+ '-subj', '/CN=spaces-local',
197
+ '-addext', 'subjectAltName=DNS:localhost,DNS:*.local,DNS:*.robindale.com,IP:127.0.0.1',
198
+ ], { stdio: 'pipe', timeout: 15000 });
199
+ console.log(' Generated self-signed TLS certificate');
200
+ return { cert: fs.readFileSync(certPath), key: fs.readFileSync(keyPath) };
201
+ } catch (e) {
202
+ console.log(` Warning: Could not generate TLS cert (${e.message})`);
203
+ console.log(' HTTPS will not be available. Install OpenSSL to enable it.');
204
+ return null;
205
+ }
206
+ }
207
+
208
+ function startServer() {
209
+ // ─── Load saved config ────────────────────────────────────
210
+ let savedConfig = {};
211
+ if (fs.existsSync(CONFIG_PATH)) {
212
+ try { savedConfig = JSON.parse(fs.readFileSync(CONFIG_PATH, 'utf-8')); } catch {}
213
+ }
214
+
215
+ // ─── Resolve settings (CLI > env > config > defaults) ─────
216
+ const PORT = cliFlags.port
217
+ || parseInt(process.env.SPACES_PORT || '', 10)
218
+ || savedConfig.port
219
+ || 3457;
220
+
221
+ const basePath = cliFlags.basePath
222
+ || process.env.SPACES_BASE_PATH
223
+ || savedConfig.basePath
224
+ || '';
225
+
226
+ const allowedOrigins = process.env.SPACES_ALLOWED_ORIGINS
227
+ || savedConfig.allowedOrigins
228
+ || '';
229
+
230
+ // ─── Kill any existing server on this port ───────────────────
231
+ stopServer();
232
+
233
+ // ─── Resolve optional packages once ─────────────────────────
234
+ const proPath = resolveSpacesPro();
235
+ const teamsPath = resolveSpacesTeams();
236
+ const cortexPath = resolveSpacesCortex();
237
+
238
+ // Tier resolution: CLI > env > config > auto-detect
239
+ let tier = cliFlags.tier
240
+ || process.env.SPACES_TIER
241
+ || savedConfig.tier
242
+ || '';
243
+
244
+ // Auto-detect tier from package presence
245
+ if (!tier) {
246
+ if (proPath) tier = 'federation';
247
+ else if (teamsPath) tier = 'team';
248
+ else tier = 'community';
249
+ }
250
+
251
+ console.log('');
252
+ console.log(' Spaces - Agent Workspace Manager');
253
+ console.log(' =================================');
254
+ console.log('');
255
+ console.log(` Tier: ${tier}`);
256
+
257
+ // ─── Server tier prerequisites ────────────────────────────
258
+ const childEnv = { ...process.env };
259
+
260
+ if (tier !== 'community') {
261
+ // team/federation tiers require @spaces/teams (auth + collaboration)
262
+ if ((tier === 'team' || tier === 'federation') && !teamsPath) {
263
+ console.error(' Error: @spaces/teams is required for team/federation tiers.');
264
+ console.error(' Install it: spaces install teams');
265
+ process.exit(1);
266
+ }
267
+ // federation tier requires @spaces/pro (network)
268
+ if (tier === 'federation' && !proPath) {
269
+ console.error(' Error: @spaces/pro is required for the federation tier.');
270
+ console.error(' Install it: spaces install pro');
271
+ process.exit(1);
272
+ }
273
+
274
+ // Check for session secret
275
+ let sessionSecret = process.env.SPACES_SESSION_SECRET || '';
276
+ if (!sessionSecret && fs.existsSync(SESSION_SECRET_PATH)) {
277
+ sessionSecret = fs.readFileSync(SESSION_SECRET_PATH, 'utf-8').trim();
278
+ }
279
+
280
+ // Check for admin DB — auto-setup if missing
281
+ const adminDbPath = path.join(SPACES_DIR, 'admin.db');
282
+ if (!fs.existsSync(adminDbPath) || !sessionSecret) {
283
+ const { autoSetup } = require('./lib/auto-setup');
284
+ const result = autoSetup({ SPACES_DIR, SESSION_SECRET_PATH, ADMIN_DB_PATH: adminDbPath, CONFIG_PATH, tier, port: PORT, basePath });
285
+ sessionSecret = result.sessionSecret;
286
+ }
287
+
288
+ childEnv.SPACES_SESSION_SECRET = sessionSecret;
289
+ process.env.SPACES_SESSION_SECRET = sessionSecret;
290
+ console.log(` Admin DB: ${adminDbPath}`);
291
+ }
292
+
293
+ // Set tier in both child env and own process env (terminal-server reads it)
294
+ childEnv.SPACES_TIER = tier;
295
+ process.env.SPACES_TIER = tier;
296
+ if (basePath) {
297
+ childEnv.SPACES_BASE_PATH = basePath;
298
+ console.log(` Base path: ${basePath}`);
299
+ }
300
+ if (allowedOrigins) {
301
+ childEnv.SPACES_ALLOWED_ORIGINS = allowedOrigins;
302
+ process.env.SPACES_ALLOWED_ORIGINS = allowedOrigins;
303
+ console.log(` Allowed origins: ${allowedOrigins}`);
304
+ }
305
+
306
+ // ─── Resolve NODE_PATH for @spaces/pro and @spaces/teams ──
307
+ // Update both childEnv (for the Next.js child process) and process.env
308
+ // (for terminal-server.js which runs in this same parent process).
309
+ // Include:
310
+ // - Managed packages node_modules (~/.spaces/packages/node_modules)
311
+ // - Each managed package's own node_modules (for its bundled deps)
312
+ // - The host app's own node_modules (fallback for peer deps)
313
+ const appNodeModules = path.join(projectDir, 'node_modules');
314
+ const nodePaths = [MANAGED_NODE_MODULES, appNodeModules];
315
+ // Add each resolved package's own node_modules so its bundled deps are found
316
+ for (const pkgPath of [proPath, teamsPath, cortexPath]) {
317
+ if (pkgPath) {
318
+ // pkgPath is the package root (symlink target or direct path)
319
+ const realPath = fs.realpathSync(pkgPath);
320
+ const pkgNodeModules = path.join(realPath, 'node_modules');
321
+ if (fs.existsSync(pkgNodeModules) && !nodePaths.includes(pkgNodeModules)) {
322
+ nodePaths.push(pkgNodeModules);
323
+ }
324
+ // Also add the parent scope for @spaces/* resolution
325
+ const parentModules = path.dirname(path.dirname(pkgPath));
326
+ if (!nodePaths.includes(parentModules)) {
327
+ nodePaths.push(parentModules);
328
+ }
329
+ }
330
+ }
331
+ for (const dir of nodePaths) {
332
+ for (const target of [childEnv, process.env]) {
333
+ const existing = target.NODE_PATH || '';
334
+ if (!existing.includes(dir)) {
335
+ target.NODE_PATH = existing
336
+ ? `${dir}${path.delimiter}${existing}`
337
+ : dir;
338
+ }
339
+ }
340
+ }
341
+ // Re-init module paths so require() in this process picks up the new NODE_PATH
342
+ require('module').Module._initPaths();
343
+
344
+ // Verify collaboration pipeline works end-to-end at startup
345
+ if (tier === 'team' || tier === 'federation') {
346
+ try {
347
+ const teams = require('@spaces/teams');
348
+ console.log(' Collaboration: @spaces/teams loaded OK');
349
+ // Smoke-test that peer deps resolve from this process
350
+ require('better-sqlite3');
351
+ } catch (e) {
352
+ console.error(` Warning: Collaboration may not work — ${e.message}`);
353
+ console.error(' Check that NODE_PATH includes the host app node_modules.');
354
+ }
355
+ }
356
+
357
+ // ─── Detect @spaces/cortex ──────────────────────────────────
358
+ if (cortexPath) {
359
+ childEnv.SPACES_HAS_CORTEX = '1';
360
+ process.env.SPACES_HAS_CORTEX = '1';
361
+ console.log(' Cortex: @spaces/cortex detected');
362
+ }
363
+
364
+ console.log('');
365
+
366
+ // ─── Background update check (non-blocking, repeats every 4h) ──
367
+ // Checks npm registry, GitHub releases, and addon git repos.
368
+ // Writes results to ~/.spaces/update-check.json for the UI banner.
369
+ const updateCheckPath = path.join(SPACES_DIR, 'update-check.json');
370
+ const checkForUpdates = async (logIfAvailable = true) => {
371
+ try {
372
+ const pkg = require(path.join(projectDir, 'package.json'));
373
+ const currentVersion = pkg.version;
374
+ const npmName = pkg.name || '@jlongo78/agent-spaces';
375
+ const githubRepo = 'jlongo78/spaces';
376
+
377
+ // Check npm registry
378
+ let npmLatest = null;
379
+ try {
380
+ const res = await fetch(`https://registry.npmjs.org/${encodeURIComponent(npmName)}/latest`, {
381
+ signal: AbortSignal.timeout(5000),
382
+ });
383
+ if (res.ok) {
384
+ const data = await res.json();
385
+ npmLatest = data.version;
386
+ }
387
+ } catch { /* */ }
388
+
389
+ // Check GitHub for latest release or tag (may be ahead of npm)
390
+ let githubLatest = null;
391
+ let githubPrerelease = false;
392
+ let githubUrl = null;
393
+ try {
394
+ // Check releases first (includes pre-releases)
395
+ const res = await fetch(`https://api.github.com/repos/${githubRepo}/releases?per_page=5`, {
396
+ signal: AbortSignal.timeout(5000),
397
+ headers: { 'Accept': 'application/vnd.github.v3+json', 'User-Agent': 'agent-spaces' },
398
+ });
399
+ if (res.ok) {
400
+ const releases = await res.json();
401
+ if (releases.length > 0) {
402
+ // Find latest stable and latest pre-release
403
+ const stable = releases.find(r => !r.prerelease && !r.draft);
404
+ const pre = releases.find(r => r.prerelease && !r.draft);
405
+ if (stable) {
406
+ githubLatest = stable.tag_name.replace(/^v/, '');
407
+ githubUrl = stable.html_url;
408
+ }
409
+ if (pre) {
410
+ const preVersion = pre.tag_name.replace(/^v/, '');
411
+ // If pre-release is newer than stable, surface it
412
+ if (!githubLatest || preVersion > githubLatest) {
413
+ githubLatest = preVersion;
414
+ githubPrerelease = true;
415
+ githubUrl = pre.html_url;
416
+ }
417
+ }
418
+ }
419
+ }
420
+ } catch { /* */ }
421
+
422
+ // Check addon git repos
423
+ const gitSafe = ['-c', 'safe.directory=*'];
424
+ const addonPaths = { teams: teamsPath, pro: proPath, cortex: cortexPath };
425
+ const addons = {};
426
+ for (const [key, addonDir] of Object.entries(addonPaths)) {
427
+ if (!addonDir || !fs.existsSync(path.join(addonDir, '.git'))) continue;
428
+ const realDir = fs.realpathSync(addonDir);
429
+ try {
430
+ execFileSync('git', [...gitSafe, 'fetch', '--quiet'], { cwd: realDir, stdio: 'ignore', timeout: 5000 });
431
+ const local = execFileSync('git', [...gitSafe, 'rev-parse', 'HEAD'], { cwd: realDir, encoding: 'utf-8' }).trim();
432
+ const remote = execFileSync('git', [...gitSafe, 'rev-parse', '@{u}'], { cwd: realDir, encoding: 'utf-8' }).trim();
433
+ let commitsBehind = 0;
434
+ if (local !== remote) {
435
+ try {
436
+ const count = execFileSync('git', [...gitSafe, 'rev-list', '--count', `HEAD..@{u}`], { cwd: realDir, encoding: 'utf-8' }).trim();
437
+ commitsBehind = parseInt(count, 10) || 0;
438
+ } catch { /* */ }
439
+ }
440
+ const behind = commitsBehind > 0;
441
+ addons[key] = { installed: true, behind, commitsBehind, localHead: local.slice(0, 7) };
442
+ if (behind && logIfAvailable) {
443
+ console.log(` Update available for @spaces/${key}: ${commitsBehind} commit(s) behind (run: spaces upgrade ${key})`);
444
+ }
445
+ } catch {
446
+ addons[key] = { installed: true, behind: false, commitsBehind: 0 };
447
+ }
448
+ }
449
+
450
+ const npmAvailable = npmLatest && npmLatest !== currentVersion && npmLatest > currentVersion;
451
+ const githubAvailable = githubLatest && githubLatest !== currentVersion && githubLatest > currentVersion;
452
+
453
+ const result = {
454
+ current: currentVersion,
455
+ npm: npmLatest ? { latest: npmLatest, available: !!npmAvailable } : null,
456
+ github: githubLatest ? { latest: githubLatest, prerelease: githubPrerelease, url: githubUrl, available: !!githubAvailable } : null,
457
+ // Backward compat: 'available' and 'latest' still work for the existing banner
458
+ latest: npmLatest || githubLatest || currentVersion,
459
+ available: !!(npmAvailable || githubAvailable),
460
+ addons,
461
+ checkedAt: Date.now(),
462
+ name: npmName,
463
+ };
464
+
465
+ fs.writeFileSync(updateCheckPath, JSON.stringify(result, null, 2));
466
+
467
+ if (npmAvailable && logIfAvailable) {
468
+ console.log(` Update available: ${currentVersion} → ${npmLatest} (run: npm i -g ${npmName})`);
469
+ }
470
+ if (githubAvailable && (!npmAvailable || githubLatest > npmLatest) && logIfAvailable) {
471
+ console.log(` GitHub ${githubPrerelease ? 'pre-release' : 'release'}: ${githubLatest} (${githubUrl || `https://github.com/${githubRepo}/releases`})`);
472
+ }
473
+ } catch { /* network error — skip silently */ }
474
+ };
475
+ checkForUpdates(true);
476
+ setInterval(() => checkForUpdates(false), 4 * 3600_000);
477
+
478
+ // Check for ~/.claude/ directory
479
+ const claudeDir = path.join(os.homedir(), '.claude');
480
+ if (!fs.existsSync(claudeDir)) {
481
+ console.log(' Warning: ~/.claude/ not found. Have you used Claude Code yet?');
482
+ console.log('');
483
+ }
484
+
485
+ // ─── Detect build type ────────────────────────────────────
486
+ const standaloneServer = path.join(projectDir, '.next', 'standalone', 'server.js');
487
+ const fullBuildDir = path.join(projectDir, '.next', 'BUILD_ID');
488
+ const isStandalone = fs.existsSync(standaloneServer);
489
+ const isFullBuild = fs.existsSync(fullBuildDir);
490
+
491
+ if (!isStandalone && !isFullBuild) {
492
+ console.error(' Error: No build found.');
493
+ console.error(' Run "npm run build" first, or install via "npm install -g @jlongo78/agent-spaces".');
494
+ process.exit(1);
495
+ }
496
+
497
+ // ─── Spawn Next.js ────────────────────────────────────────
498
+ let next;
499
+ if (isStandalone) {
500
+ const parentNodeModules = path.join(projectDir, 'node_modules');
501
+ const existingNodePath = childEnv.NODE_PATH || '';
502
+ childEnv.NODE_PATH = existingNodePath
503
+ ? `${parentNodeModules}${path.delimiter}${existingNodePath}`
504
+ : parentNodeModules;
505
+
506
+ next = spawn(process.execPath, [standaloneServer], {
507
+ cwd: path.dirname(standaloneServer),
508
+ stdio: ['ignore', 'pipe', 'pipe'],
509
+ env: {
510
+ ...childEnv,
511
+ PORT: String(NEXT_INTERNAL_PORT),
512
+ HOSTNAME: '0.0.0.0',
513
+ NODE_ENV: 'production',
514
+ },
515
+ });
516
+ } else {
517
+ next = spawn('npx', ['next', 'start', '--port', String(NEXT_INTERNAL_PORT)], {
518
+ cwd: projectDir,
519
+ stdio: ['ignore', 'pipe', 'pipe'],
520
+ env: {
521
+ ...childEnv,
522
+ PORT: String(NEXT_INTERNAL_PORT),
523
+ HOSTNAME: '0.0.0.0',
524
+ NODE_ENV: 'production',
525
+ },
526
+ shell: true,
527
+ });
528
+ }
529
+
530
+ let nextReady = false;
531
+
532
+ next.stdout.on('data', (data) => {
533
+ const msg = data.toString();
534
+ process.stdout.write(msg);
535
+ if (!nextReady && (msg.includes('Ready') || msg.includes('started server') || msg.includes('Listening') || msg.includes('localhost'))) {
536
+ nextReady = true;
537
+ console.log(`\n Ready at http://localhost:${PORT}`);
538
+ if (tlsCreds) console.log(` Also at https://localhost:${PORT} (self-signed)`);
539
+ console.log('');
540
+
541
+ const url = `http://localhost:${PORT}`;
542
+ const isService = process.env.SPACES_SERVICE === '1';
543
+ if (!isService) {
544
+ try {
545
+ if (process.platform === 'win32') {
546
+ execFileSync('cmd', ['/c', 'start', url], { stdio: 'ignore' });
547
+ } else if (process.platform === 'darwin') {
548
+ execFileSync('open', [url], { stdio: 'ignore' });
549
+ } else {
550
+ execFileSync('xdg-open', [url], { stdio: 'ignore' });
551
+ }
552
+ } catch {
553
+ console.log(` Open ${url} in your browser`);
554
+ }
555
+ }
556
+ }
557
+ });
558
+
559
+ next.stderr.on('data', (data) => {
560
+ const msg = data.toString();
561
+ if (!msg.includes('Warning')) {
562
+ process.stderr.write(data);
563
+ }
564
+ });
565
+
566
+ next.on('exit', (code) => {
567
+ if (code !== 0 && code !== null) {
568
+ console.error(` [spaces] Next.js exited with code ${code}`);
569
+ }
570
+ cleanup();
571
+ });
572
+
573
+ // ─── Proxy request handler (shared by HTTP and HTTPS) ────
574
+ function proxyHandler(req, res) {
575
+ // WebSocket paths are handled by the 'upgrade' event, but a plain
576
+ // HTTP request to /ws (e.g. health check) should not be proxied to
577
+ // Next.js which would 308-redirect it due to trailingSlash.
578
+ const urlPath = (req.url || '').split('?')[0];
579
+ if (urlPath === '/ws' || urlPath.endsWith('/ws') || urlPath.endsWith('/ws/')) {
580
+ res.writeHead(200, { 'Content-Type': 'text/plain' });
581
+ res.end('WebSocket endpoint');
582
+ return;
583
+ }
584
+
585
+ const proxyReq = http.request(
586
+ {
587
+ hostname: '127.0.0.1',
588
+ port: NEXT_INTERNAL_PORT,
589
+ path: req.url,
590
+ method: req.method,
591
+ headers: req.headers,
592
+ },
593
+ (proxyRes) => {
594
+ res.writeHead(proxyRes.statusCode, proxyRes.headers);
595
+ proxyRes.pipe(res);
596
+ }
597
+ );
598
+ proxyReq.on('error', () => {
599
+ if (!res.headersSent) {
600
+ res.writeHead(502);
601
+ res.end('Server not ready yet. Refresh in a moment.');
602
+ }
603
+ });
604
+ req.pipe(proxyReq);
605
+ }
606
+
607
+ // ─── HTTP + HTTPS on a single port ─────────────────────────
608
+ // Peek at the first byte of each connection: 0x16 = TLS ClientHello,
609
+ // anything else = plain HTTP. This lets mobile browsers that force
610
+ // HTTPS-first connect to the same port without a separate listener.
611
+ const httpServer = http.createServer(proxyHandler);
612
+ const tlsCreds = ensureTlsCert();
613
+ const httpsServer = tlsCreds ? https.createServer(tlsCreds, proxyHandler) : null;
614
+
615
+ const { createTerminalServer } = require('./terminal-server');
616
+ createTerminalServer(httpServer);
617
+ if (httpsServer) createTerminalServer(httpsServer);
618
+
619
+ // Wire up 'address()' on the inner servers so terminal-server can
620
+ // discover the port, then emit 'listening' so its event handler fires.
621
+ const patchAddress = (srv) => { srv.address = () => ({ port: PORT }); };
622
+ patchAddress(httpServer);
623
+ if (httpsServer) patchAddress(httpsServer);
624
+
625
+ const dualServer = net.createServer((socket) => {
626
+ // If client connects but sends nothing for 5s, destroy.
627
+ // Once routed, clear the timeout so long-lived connections (WebSocket) aren't killed.
628
+ socket.setTimeout(5000, () => socket.destroy());
629
+ socket.once('data', (buf) => {
630
+ socket.setTimeout(0);
631
+ socket.pause();
632
+ const target = (buf[0] === 0x16 && httpsServer) ? httpsServer : httpServer;
633
+ target.emit('connection', socket);
634
+ socket.unshift(buf);
635
+ socket.resume();
636
+ });
637
+ });
638
+
639
+ dualServer.listen(PORT, () => {
640
+ // Now that the port is bound, tell the inner servers they're "listening"
641
+ // so terminal-server's mDNS and message watcher start up.
642
+ httpServer.emit('listening');
643
+ if (httpsServer) httpsServer.emit('listening');
644
+
645
+ if (httpsServer) {
646
+ console.log(` Starting server on port ${PORT} (HTTP + HTTPS)`);
647
+ } else {
648
+ console.log(` Starting server on http://localhost:${PORT}`);
649
+ }
650
+ console.log('');
651
+ });
652
+
653
+ function cleanup() {
654
+ next.kill();
655
+ dualServer.close();
656
+ process.exit(0);
657
+ }
658
+
659
+ process.on('SIGINT', cleanup);
660
+ process.on('SIGTERM', cleanup);
661
+ }
662
+
663
+ // ─── Stop running server ─────────────────────────────────────
664
+ function findPidsOnPort(port) {
665
+ const pids = new Set();
666
+ if (process.platform === 'win32') {
667
+ try {
668
+ const output = execFileSync('netstat', ['-ano'], { encoding: 'utf-8' });
669
+ for (const line of output.split(String.fromCharCode(10))) {
670
+ if (line.includes(':' + port + ' ') && line.includes('LISTENING')) {
671
+ const parts = line.trim().split(/\s+/);
672
+ const pid = parseInt(parts[parts.length - 1], 10);
673
+ if (pid > 0) pids.add(pid);
674
+ }
675
+ }
676
+ } catch {}
677
+ } else {
678
+ try {
679
+ const output = execFileSync('lsof', ['-ti', `tcp:${port}`], { encoding: 'utf-8' }).trim();
680
+ for (const p of output.split('\n')) { if (p.trim()) pids.add(parseInt(p.trim(), 10)); }
681
+ } catch {}
682
+ try {
683
+ const output = execFileSync('fuser', [`${port}/tcp`], { encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] }).trim();
684
+ for (const p of output.split(/\s+/)) { if (p.trim()) pids.add(parseInt(p.trim(), 10)); }
685
+ } catch {}
686
+ }
687
+ return [...pids].filter(p => !isNaN(p) && p > 0);
688
+ }
689
+
690
+ function stopServer() {
691
+ let savedConfig = {};
692
+ if (fs.existsSync(CONFIG_PATH)) {
693
+ try { savedConfig = JSON.parse(fs.readFileSync(CONFIG_PATH, 'utf-8')); } catch {}
694
+ }
695
+ const port = parseInt(process.env.SPACES_PORT || '', 10) || savedConfig.port || 3457;
696
+ const ports = [port, NEXT_INTERNAL_PORT];
697
+ let killed = 0;
698
+
699
+ for (const p of ports) {
700
+ for (const pid of findPidsOnPort(p)) {
701
+ try {
702
+ process.kill(pid, 'SIGTERM');
703
+ killed++;
704
+ } catch {}
705
+ }
706
+ }
707
+
708
+ if (killed > 0) {
709
+ console.log(` Stopped ${killed} process(es) on port ${port}`);
710
+ // Wait for ports to be released, escalate to SIGKILL if needed
711
+ const deadline = Date.now() + 2000;
712
+ let escalated = false;
713
+ while (Date.now() < deadline) {
714
+ const remaining = ports.flatMap(p => findPidsOnPort(p));
715
+ if (remaining.length === 0) break;
716
+ if (!escalated && Date.now() > deadline - 1000) {
717
+ for (const pid of remaining) {
718
+ try { process.kill(pid, 'SIGKILL'); } catch {}
719
+ }
720
+ escalated = true;
721
+ }
722
+ if (process.platform === 'win32') { require('child_process').spawnSync('powershell', ['-Command', 'Start-Sleep -Milliseconds 200']); } else { require('child_process').spawnSync('sleep', ['0.2']); }
723
+ }
724
+ } else {
725
+ console.log(` No running server found on port ${port}`);
726
+ }
727
+ }
728
+
729
+ // ─── @spaces/pro resolution ──────────────────────────────────
730
+ function resolveSpacesPro() {
731
+ // 1. Managed install (~/.spaces/packages/)
732
+ const managed = path.join(MANAGED_NODE_MODULES, '@spaces', 'pro');
733
+ if (fs.existsSync(path.join(managed, 'dist', 'index.js'))) return managed;
734
+
735
+ // 2. Local node_modules (npm link / optionalDep)
736
+ try {
737
+ return require.resolve('@spaces/pro');
738
+ } catch {}
739
+
740
+ // 3. Global npm prefix (legacy)
741
+ try {
742
+ const globalPrefix = execFileSync('npm', ['prefix', '-g'], { encoding: 'utf-8' }).trim();
743
+ const globalProPath = path.join(globalPrefix, 'lib', 'node_modules', '@spaces', 'pro');
744
+ if (fs.existsSync(globalProPath)) return globalProPath;
745
+ const altPath = path.join(globalPrefix, 'node_modules', '@spaces', 'pro');
746
+ if (fs.existsSync(altPath)) return altPath;
747
+ } catch {}
748
+
749
+ return null;
750
+ }
751
+
752
+ // ─── @spaces/teams resolution ────────────────────────────────
753
+ function resolveSpacesTeams() {
754
+ // 1. Managed install (~/.spaces/packages/)
755
+ const managed = path.join(MANAGED_NODE_MODULES, '@spaces', 'teams');
756
+ if (fs.existsSync(path.join(managed, 'dist', 'index.js'))) return managed;
757
+
758
+ // 2. Local node_modules (npm link / optionalDep)
759
+ try {
760
+ return require.resolve('@spaces/teams');
761
+ } catch {}
762
+
763
+ // 3. Global npm prefix (legacy)
764
+ try {
765
+ const globalPrefix = execFileSync('npm', ['prefix', '-g'], { encoding: 'utf-8' }).trim();
766
+ const globalTeamsPath = path.join(globalPrefix, 'lib', 'node_modules', '@spaces', 'teams');
767
+ if (fs.existsSync(globalTeamsPath)) return globalTeamsPath;
768
+ const altPath = path.join(globalPrefix, 'node_modules', '@spaces', 'teams');
769
+ if (fs.existsSync(altPath)) return altPath;
770
+ } catch {}
771
+
772
+ return null;
773
+ }
774
+
775
+ // ─── @spaces/cortex resolution ───────────────────────────────
776
+ function resolveSpacesCortex() {
777
+ const managed = path.join(MANAGED_NODE_MODULES, '@spaces', 'cortex');
778
+ if (fs.existsSync(path.join(managed, 'dist', 'index.js'))) return managed;
779
+ try { return require.resolve('@spaces/cortex'); } catch {}
780
+ try {
781
+ const globalPrefix = execFileSync('npm', ['prefix', '-g'], { encoding: 'utf-8' }).trim();
782
+ const globalPath = path.join(globalPrefix, 'lib', 'node_modules', '@spaces', 'cortex');
783
+ if (fs.existsSync(globalPath)) return globalPath;
784
+ const altPath = path.join(globalPrefix, 'node_modules', '@spaces', 'cortex');
785
+ if (fs.existsSync(altPath)) return altPath;
786
+ } catch {}
787
+ return null;
788
+ }