@mecha.im/cli 0.3.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (691) hide show
  1. package/Dockerfile +103 -0
  2. package/README.md +173 -0
  3. package/agent/activity.ts +31 -0
  4. package/agent/costs.ts +93 -0
  5. package/agent/dashboard/dist/assets/addon-fit-YJmn1quW.js +12 -0
  6. package/agent/dashboard/dist/assets/addon-web-links-BWjmmSgS.js +12 -0
  7. package/agent/dashboard/dist/assets/index-B4OKDQ7e.css +32 -0
  8. package/agent/dashboard/dist/assets/index-fZ_N6Wsi.js +193 -0
  9. package/agent/dashboard/dist/assets/xterm-BqvuqXEL.js +27 -0
  10. package/agent/dashboard/dist/index.html +19 -0
  11. package/agent/dashboard/dist/pixel-engine/characters/char_0.png +0 -0
  12. package/agent/dashboard/dist/pixel-engine/characters/char_1.png +0 -0
  13. package/agent/dashboard/dist/pixel-engine/characters/char_2.png +0 -0
  14. package/agent/dashboard/dist/pixel-engine/characters/char_3.png +0 -0
  15. package/agent/dashboard/dist/pixel-engine/characters/char_4.png +0 -0
  16. package/agent/dashboard/dist/pixel-engine/characters/char_5.png +0 -0
  17. package/agent/dashboard/dist/pixel-engine/default-layout.json +4320 -0
  18. package/agent/dashboard/dist/pixel-engine/floors/floor_0.png +0 -0
  19. package/agent/dashboard/dist/pixel-engine/floors/floor_1.png +0 -0
  20. package/agent/dashboard/dist/pixel-engine/floors/floor_2.png +0 -0
  21. package/agent/dashboard/dist/pixel-engine/floors/floor_3.png +0 -0
  22. package/agent/dashboard/dist/pixel-engine/floors/floor_4.png +0 -0
  23. package/agent/dashboard/dist/pixel-engine/floors/floor_5.png +0 -0
  24. package/agent/dashboard/dist/pixel-engine/floors/floor_6.png +0 -0
  25. package/agent/dashboard/dist/pixel-engine/floors/floor_7.png +0 -0
  26. package/agent/dashboard/dist/pixel-engine/floors/floor_8.png +0 -0
  27. package/agent/dashboard/dist/pixel-engine/fonts/FSPixelSansUnicode-Regular.ttf +0 -0
  28. package/agent/dashboard/dist/pixel-engine/furniture/BIN/BIN.png +0 -0
  29. package/agent/dashboard/dist/pixel-engine/furniture/BIN/manifest.json +13 -0
  30. package/agent/dashboard/dist/pixel-engine/furniture/BLACKBOARD/BLACKBOARD.png +0 -0
  31. package/agent/dashboard/dist/pixel-engine/furniture/BLACKBOARD/manifest.json +16 -0
  32. package/agent/dashboard/dist/pixel-engine/furniture/BOOKSHELF/BOOKSHELF.png +0 -0
  33. package/agent/dashboard/dist/pixel-engine/furniture/BOOKSHELF/manifest.json +16 -0
  34. package/agent/dashboard/dist/pixel-engine/furniture/CACTUS/CACTUS.png +0 -0
  35. package/agent/dashboard/dist/pixel-engine/furniture/CACTUS/manifest.json +13 -0
  36. package/agent/dashboard/dist/pixel-engine/furniture/CLOCK/CLOCK.png +0 -0
  37. package/agent/dashboard/dist/pixel-engine/furniture/CLOCK/manifest.json +13 -0
  38. package/agent/dashboard/dist/pixel-engine/furniture/COFFEE/COFFEE.png +0 -0
  39. package/agent/dashboard/dist/pixel-engine/furniture/COFFEE/manifest.json +13 -0
  40. package/agent/dashboard/dist/pixel-engine/furniture/COFFEE_TABLE/COFFEE_TABLE.png +0 -0
  41. package/agent/dashboard/dist/pixel-engine/furniture/COFFEE_TABLE/manifest.json +13 -0
  42. package/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_BENCH/CUSHIONED_BENCH.png +0 -0
  43. package/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_BENCH/manifest.json +13 -0
  44. package/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_CHAIR/CUSHIONED_CHAIR_BACK.png +0 -0
  45. package/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_CHAIR/CUSHIONED_CHAIR_FRONT.png +0 -0
  46. package/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_CHAIR/CUSHIONED_CHAIR_SIDE.png +0 -0
  47. package/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_CHAIR/manifest.json +44 -0
  48. package/agent/dashboard/dist/pixel-engine/furniture/DESK/DESK_FRONT.png +0 -0
  49. package/agent/dashboard/dist/pixel-engine/furniture/DESK/DESK_SIDE.png +0 -0
  50. package/agent/dashboard/dist/pixel-engine/furniture/DESK/manifest.json +33 -0
  51. package/agent/dashboard/dist/pixel-engine/furniture/DOUBLE_BOOKSHELF/DOUBLE_BOOKSHELF.png +0 -0
  52. package/agent/dashboard/dist/pixel-engine/furniture/DOUBLE_BOOKSHELF/manifest.json +16 -0
  53. package/agent/dashboard/dist/pixel-engine/furniture/FILE_BOXES_1/FILE_BOXES_1.png +0 -0
  54. package/agent/dashboard/dist/pixel-engine/furniture/FILE_BOXES_1/manifest.json +13 -0
  55. package/agent/dashboard/dist/pixel-engine/furniture/FILE_BOXES_2/FILE_BOXES_2.png +0 -0
  56. package/agent/dashboard/dist/pixel-engine/furniture/FILE_BOXES_2/manifest.json +13 -0
  57. package/agent/dashboard/dist/pixel-engine/furniture/HANGING_PLANT/HANGING_PLANT.png +0 -0
  58. package/agent/dashboard/dist/pixel-engine/furniture/HANGING_PLANT/manifest.json +13 -0
  59. package/agent/dashboard/dist/pixel-engine/furniture/LARGE_PAINTING/LARGE_PAINTING.png +0 -0
  60. package/agent/dashboard/dist/pixel-engine/furniture/LARGE_PAINTING/manifest.json +13 -0
  61. package/agent/dashboard/dist/pixel-engine/furniture/LARGE_PLANT/LARGE_PLANT.png +0 -0
  62. package/agent/dashboard/dist/pixel-engine/furniture/LARGE_PLANT/manifest.json +13 -0
  63. package/agent/dashboard/dist/pixel-engine/furniture/PC/PC_BACK.png +0 -0
  64. package/agent/dashboard/dist/pixel-engine/furniture/PC/PC_FRONT_OFF.png +0 -0
  65. package/agent/dashboard/dist/pixel-engine/furniture/PC/PC_FRONT_ON_1.png +0 -0
  66. package/agent/dashboard/dist/pixel-engine/furniture/PC/PC_FRONT_ON_2.png +0 -0
  67. package/agent/dashboard/dist/pixel-engine/furniture/PC/PC_FRONT_ON_3.png +0 -0
  68. package/agent/dashboard/dist/pixel-engine/furniture/PC/PC_SIDE.png +0 -0
  69. package/agent/dashboard/dist/pixel-engine/furniture/PC/manifest.json +88 -0
  70. package/agent/dashboard/dist/pixel-engine/furniture/PLANT/PLANT.png +0 -0
  71. package/agent/dashboard/dist/pixel-engine/furniture/PLANT/manifest.json +13 -0
  72. package/agent/dashboard/dist/pixel-engine/furniture/PLANT_2/PLANT_2.png +0 -0
  73. package/agent/dashboard/dist/pixel-engine/furniture/PLANT_2/manifest.json +13 -0
  74. package/agent/dashboard/dist/pixel-engine/furniture/POT/POT.png +0 -0
  75. package/agent/dashboard/dist/pixel-engine/furniture/POT/manifest.json +13 -0
  76. package/agent/dashboard/dist/pixel-engine/furniture/SMALL_PAINTING/SMALL_PAINTING.png +0 -0
  77. package/agent/dashboard/dist/pixel-engine/furniture/SMALL_PAINTING/manifest.json +13 -0
  78. package/agent/dashboard/dist/pixel-engine/furniture/SMALL_PAINTING_2/SMALL_PAINTING_2.png +0 -0
  79. package/agent/dashboard/dist/pixel-engine/furniture/SMALL_PAINTING_2/manifest.json +13 -0
  80. package/agent/dashboard/dist/pixel-engine/furniture/SMALL_TABLE/SMALL_TABLE_FRONT.png +0 -0
  81. package/agent/dashboard/dist/pixel-engine/furniture/SMALL_TABLE/SMALL_TABLE_SIDE.png +0 -0
  82. package/agent/dashboard/dist/pixel-engine/furniture/SMALL_TABLE/manifest.json +33 -0
  83. package/agent/dashboard/dist/pixel-engine/furniture/SOFA/SOFA_BACK.png +0 -0
  84. package/agent/dashboard/dist/pixel-engine/furniture/SOFA/SOFA_FRONT.png +0 -0
  85. package/agent/dashboard/dist/pixel-engine/furniture/SOFA/SOFA_SIDE.png +0 -0
  86. package/agent/dashboard/dist/pixel-engine/furniture/SOFA/manifest.json +44 -0
  87. package/agent/dashboard/dist/pixel-engine/furniture/TABLE_FRONT/TABLE_FRONT.png +0 -0
  88. package/agent/dashboard/dist/pixel-engine/furniture/TABLE_FRONT/manifest.json +13 -0
  89. package/agent/dashboard/dist/pixel-engine/furniture/VENDER_MACHINE/VENDER_MACHINE.png +0 -0
  90. package/agent/dashboard/dist/pixel-engine/furniture/VENDER_MACHINE/manifest.json +16 -0
  91. package/agent/dashboard/dist/pixel-engine/furniture/WATER_MACHINE/WATER_MACHINE.png +0 -0
  92. package/agent/dashboard/dist/pixel-engine/furniture/WATER_MACHINE/manifest.json +16 -0
  93. package/agent/dashboard/dist/pixel-engine/furniture/WHITEBOARD/WHITEBOARD.png +0 -0
  94. package/agent/dashboard/dist/pixel-engine/furniture/WHITEBOARD/manifest.json +16 -0
  95. package/agent/dashboard/dist/pixel-engine/furniture/WOODEN_BENCH/WOODEN_BENCH.png +0 -0
  96. package/agent/dashboard/dist/pixel-engine/furniture/WOODEN_BENCH/manifest.json +13 -0
  97. package/agent/dashboard/dist/pixel-engine/furniture/WOODEN_CHAIR/WOODEN_CHAIR_BACK.png +0 -0
  98. package/agent/dashboard/dist/pixel-engine/furniture/WOODEN_CHAIR/WOODEN_CHAIR_FRONT.png +0 -0
  99. package/agent/dashboard/dist/pixel-engine/furniture/WOODEN_CHAIR/WOODEN_CHAIR_SIDE.png +0 -0
  100. package/agent/dashboard/dist/pixel-engine/furniture/WOODEN_CHAIR/manifest.json +44 -0
  101. package/agent/dashboard/dist/pixel-engine/furniture/index.json +32 -0
  102. package/agent/dashboard/dist/pixel-engine/walls/wall_0.png +0 -0
  103. package/agent/dashboard/index.html +18 -0
  104. package/agent/dashboard/package.json +41 -0
  105. package/agent/dashboard/public/pixel-engine/CREDITS.md +17 -0
  106. package/agent/dashboard/public/pixel-engine/characters/char_0.png +0 -0
  107. package/agent/dashboard/public/pixel-engine/characters/char_1.png +0 -0
  108. package/agent/dashboard/public/pixel-engine/characters/char_2.png +0 -0
  109. package/agent/dashboard/public/pixel-engine/characters/char_3.png +0 -0
  110. package/agent/dashboard/public/pixel-engine/characters/char_4.png +0 -0
  111. package/agent/dashboard/public/pixel-engine/characters/char_5.png +0 -0
  112. package/agent/dashboard/public/pixel-engine/default-layout.json +4320 -0
  113. package/agent/dashboard/public/pixel-engine/floors/floor_0.png +0 -0
  114. package/agent/dashboard/public/pixel-engine/floors/floor_1.png +0 -0
  115. package/agent/dashboard/public/pixel-engine/floors/floor_2.png +0 -0
  116. package/agent/dashboard/public/pixel-engine/floors/floor_3.png +0 -0
  117. package/agent/dashboard/public/pixel-engine/floors/floor_4.png +0 -0
  118. package/agent/dashboard/public/pixel-engine/floors/floor_5.png +0 -0
  119. package/agent/dashboard/public/pixel-engine/floors/floor_6.png +0 -0
  120. package/agent/dashboard/public/pixel-engine/floors/floor_7.png +0 -0
  121. package/agent/dashboard/public/pixel-engine/floors/floor_8.png +0 -0
  122. package/agent/dashboard/public/pixel-engine/fonts/FSPixelSansUnicode-Regular.ttf +0 -0
  123. package/agent/dashboard/public/pixel-engine/furniture/BIN/BIN.png +0 -0
  124. package/agent/dashboard/public/pixel-engine/furniture/BIN/manifest.json +13 -0
  125. package/agent/dashboard/public/pixel-engine/furniture/BLACKBOARD/BLACKBOARD.png +0 -0
  126. package/agent/dashboard/public/pixel-engine/furniture/BLACKBOARD/manifest.json +16 -0
  127. package/agent/dashboard/public/pixel-engine/furniture/BOOKSHELF/BOOKSHELF.png +0 -0
  128. package/agent/dashboard/public/pixel-engine/furniture/BOOKSHELF/manifest.json +16 -0
  129. package/agent/dashboard/public/pixel-engine/furniture/CACTUS/CACTUS.png +0 -0
  130. package/agent/dashboard/public/pixel-engine/furniture/CACTUS/manifest.json +13 -0
  131. package/agent/dashboard/public/pixel-engine/furniture/CLOCK/CLOCK.png +0 -0
  132. package/agent/dashboard/public/pixel-engine/furniture/CLOCK/manifest.json +13 -0
  133. package/agent/dashboard/public/pixel-engine/furniture/COFFEE/COFFEE.png +0 -0
  134. package/agent/dashboard/public/pixel-engine/furniture/COFFEE/manifest.json +13 -0
  135. package/agent/dashboard/public/pixel-engine/furniture/COFFEE_TABLE/COFFEE_TABLE.png +0 -0
  136. package/agent/dashboard/public/pixel-engine/furniture/COFFEE_TABLE/manifest.json +13 -0
  137. package/agent/dashboard/public/pixel-engine/furniture/CUSHIONED_BENCH/CUSHIONED_BENCH.png +0 -0
  138. package/agent/dashboard/public/pixel-engine/furniture/CUSHIONED_BENCH/manifest.json +13 -0
  139. package/agent/dashboard/public/pixel-engine/furniture/CUSHIONED_CHAIR/CUSHIONED_CHAIR_BACK.png +0 -0
  140. package/agent/dashboard/public/pixel-engine/furniture/CUSHIONED_CHAIR/CUSHIONED_CHAIR_FRONT.png +0 -0
  141. package/agent/dashboard/public/pixel-engine/furniture/CUSHIONED_CHAIR/CUSHIONED_CHAIR_SIDE.png +0 -0
  142. package/agent/dashboard/public/pixel-engine/furniture/CUSHIONED_CHAIR/manifest.json +44 -0
  143. package/agent/dashboard/public/pixel-engine/furniture/DESK/DESK_FRONT.png +0 -0
  144. package/agent/dashboard/public/pixel-engine/furniture/DESK/DESK_SIDE.png +0 -0
  145. package/agent/dashboard/public/pixel-engine/furniture/DESK/manifest.json +33 -0
  146. package/agent/dashboard/public/pixel-engine/furniture/DOUBLE_BOOKSHELF/DOUBLE_BOOKSHELF.png +0 -0
  147. package/agent/dashboard/public/pixel-engine/furniture/DOUBLE_BOOKSHELF/manifest.json +16 -0
  148. package/agent/dashboard/public/pixel-engine/furniture/FILE_BOXES_1/FILE_BOXES_1.png +0 -0
  149. package/agent/dashboard/public/pixel-engine/furniture/FILE_BOXES_1/manifest.json +13 -0
  150. package/agent/dashboard/public/pixel-engine/furniture/FILE_BOXES_2/FILE_BOXES_2.png +0 -0
  151. package/agent/dashboard/public/pixel-engine/furniture/FILE_BOXES_2/manifest.json +13 -0
  152. package/agent/dashboard/public/pixel-engine/furniture/HANGING_PLANT/HANGING_PLANT.png +0 -0
  153. package/agent/dashboard/public/pixel-engine/furniture/HANGING_PLANT/manifest.json +13 -0
  154. package/agent/dashboard/public/pixel-engine/furniture/LARGE_PAINTING/LARGE_PAINTING.png +0 -0
  155. package/agent/dashboard/public/pixel-engine/furniture/LARGE_PAINTING/manifest.json +13 -0
  156. package/agent/dashboard/public/pixel-engine/furniture/LARGE_PLANT/LARGE_PLANT.png +0 -0
  157. package/agent/dashboard/public/pixel-engine/furniture/LARGE_PLANT/manifest.json +13 -0
  158. package/agent/dashboard/public/pixel-engine/furniture/PC/PC_BACK.png +0 -0
  159. package/agent/dashboard/public/pixel-engine/furniture/PC/PC_FRONT_OFF.png +0 -0
  160. package/agent/dashboard/public/pixel-engine/furniture/PC/PC_FRONT_ON_1.png +0 -0
  161. package/agent/dashboard/public/pixel-engine/furniture/PC/PC_FRONT_ON_2.png +0 -0
  162. package/agent/dashboard/public/pixel-engine/furniture/PC/PC_FRONT_ON_3.png +0 -0
  163. package/agent/dashboard/public/pixel-engine/furniture/PC/PC_SIDE.png +0 -0
  164. package/agent/dashboard/public/pixel-engine/furniture/PC/manifest.json +88 -0
  165. package/agent/dashboard/public/pixel-engine/furniture/PLANT/PLANT.png +0 -0
  166. package/agent/dashboard/public/pixel-engine/furniture/PLANT/manifest.json +13 -0
  167. package/agent/dashboard/public/pixel-engine/furniture/PLANT_2/PLANT_2.png +0 -0
  168. package/agent/dashboard/public/pixel-engine/furniture/PLANT_2/manifest.json +13 -0
  169. package/agent/dashboard/public/pixel-engine/furniture/POT/POT.png +0 -0
  170. package/agent/dashboard/public/pixel-engine/furniture/POT/manifest.json +13 -0
  171. package/agent/dashboard/public/pixel-engine/furniture/SMALL_PAINTING/SMALL_PAINTING.png +0 -0
  172. package/agent/dashboard/public/pixel-engine/furniture/SMALL_PAINTING/manifest.json +13 -0
  173. package/agent/dashboard/public/pixel-engine/furniture/SMALL_PAINTING_2/SMALL_PAINTING_2.png +0 -0
  174. package/agent/dashboard/public/pixel-engine/furniture/SMALL_PAINTING_2/manifest.json +13 -0
  175. package/agent/dashboard/public/pixel-engine/furniture/SMALL_TABLE/SMALL_TABLE_FRONT.png +0 -0
  176. package/agent/dashboard/public/pixel-engine/furniture/SMALL_TABLE/SMALL_TABLE_SIDE.png +0 -0
  177. package/agent/dashboard/public/pixel-engine/furniture/SMALL_TABLE/manifest.json +33 -0
  178. package/agent/dashboard/public/pixel-engine/furniture/SOFA/SOFA_BACK.png +0 -0
  179. package/agent/dashboard/public/pixel-engine/furniture/SOFA/SOFA_FRONT.png +0 -0
  180. package/agent/dashboard/public/pixel-engine/furniture/SOFA/SOFA_SIDE.png +0 -0
  181. package/agent/dashboard/public/pixel-engine/furniture/SOFA/manifest.json +44 -0
  182. package/agent/dashboard/public/pixel-engine/furniture/TABLE_FRONT/TABLE_FRONT.png +0 -0
  183. package/agent/dashboard/public/pixel-engine/furniture/TABLE_FRONT/manifest.json +13 -0
  184. package/agent/dashboard/public/pixel-engine/furniture/VENDER_MACHINE/VENDER_MACHINE.png +0 -0
  185. package/agent/dashboard/public/pixel-engine/furniture/VENDER_MACHINE/manifest.json +16 -0
  186. package/agent/dashboard/public/pixel-engine/furniture/WATER_MACHINE/WATER_MACHINE.png +0 -0
  187. package/agent/dashboard/public/pixel-engine/furniture/WATER_MACHINE/manifest.json +16 -0
  188. package/agent/dashboard/public/pixel-engine/furniture/WHITEBOARD/WHITEBOARD.png +0 -0
  189. package/agent/dashboard/public/pixel-engine/furniture/WHITEBOARD/manifest.json +16 -0
  190. package/agent/dashboard/public/pixel-engine/furniture/WOODEN_BENCH/WOODEN_BENCH.png +0 -0
  191. package/agent/dashboard/public/pixel-engine/furniture/WOODEN_BENCH/manifest.json +13 -0
  192. package/agent/dashboard/public/pixel-engine/furniture/WOODEN_CHAIR/WOODEN_CHAIR_BACK.png +0 -0
  193. package/agent/dashboard/public/pixel-engine/furniture/WOODEN_CHAIR/WOODEN_CHAIR_FRONT.png +0 -0
  194. package/agent/dashboard/public/pixel-engine/furniture/WOODEN_CHAIR/WOODEN_CHAIR_SIDE.png +0 -0
  195. package/agent/dashboard/public/pixel-engine/furniture/WOODEN_CHAIR/manifest.json +44 -0
  196. package/agent/dashboard/public/pixel-engine/furniture/index.json +32 -0
  197. package/agent/dashboard/public/pixel-engine/walls/wall_0.png +0 -0
  198. package/agent/dashboard/src/app.tsx +223 -0
  199. package/agent/dashboard/src/components/alert.tsx +24 -0
  200. package/agent/dashboard/src/components/badge.tsx +41 -0
  201. package/agent/dashboard/src/components/button.tsx +44 -0
  202. package/agent/dashboard/src/components/card.tsx +21 -0
  203. package/agent/dashboard/src/components/dialog.tsx +28 -0
  204. package/agent/dashboard/src/components/index.ts +7 -0
  205. package/agent/dashboard/src/components/input.tsx +45 -0
  206. package/agent/dashboard/src/components/status-dot.tsx +31 -0
  207. package/agent/dashboard/src/index.css +126 -0
  208. package/agent/dashboard/src/lib/api.test.ts +41 -0
  209. package/agent/dashboard/src/lib/api.ts +53 -0
  210. package/agent/dashboard/src/lib/fleet-context.tsx +92 -0
  211. package/agent/dashboard/src/lib/format.test.ts +119 -0
  212. package/agent/dashboard/src/lib/format.ts +41 -0
  213. package/agent/dashboard/src/lib/login-gate.tsx +117 -0
  214. package/agent/dashboard/src/lib/sanitize.test.ts +86 -0
  215. package/agent/dashboard/src/lib/sanitize.ts +13 -0
  216. package/agent/dashboard/src/lib/snippet.test.ts +50 -0
  217. package/agent/dashboard/src/lib/snippet.ts +15 -0
  218. package/agent/dashboard/src/lib/ws-parse.test.ts +72 -0
  219. package/agent/dashboard/src/lib/ws-parse.ts +46 -0
  220. package/agent/dashboard/src/main.tsx +25 -0
  221. package/agent/dashboard/src/pixel-engine/colorize.ts +230 -0
  222. package/agent/dashboard/src/pixel-engine/components/BottomToolbar.tsx +83 -0
  223. package/agent/dashboard/src/pixel-engine/components/DebugView.tsx +175 -0
  224. package/agent/dashboard/src/pixel-engine/components/OfficeCanvas.tsx +838 -0
  225. package/agent/dashboard/src/pixel-engine/components/PixelOffice.tsx +646 -0
  226. package/agent/dashboard/src/pixel-engine/components/SettingsModal.tsx +213 -0
  227. package/agent/dashboard/src/pixel-engine/components/ToolOverlay.tsx +203 -0
  228. package/agent/dashboard/src/pixel-engine/components/ZoomControls.tsx +177 -0
  229. package/agent/dashboard/src/pixel-engine/constants.ts +132 -0
  230. package/agent/dashboard/src/pixel-engine/editor/EditorToolbar.tsx +737 -0
  231. package/agent/dashboard/src/pixel-engine/editor/editorActions.ts +291 -0
  232. package/agent/dashboard/src/pixel-engine/editor/editorState.ts +123 -0
  233. package/agent/dashboard/src/pixel-engine/engine/characters.ts +595 -0
  234. package/agent/dashboard/src/pixel-engine/engine/gameLoop.ts +50 -0
  235. package/agent/dashboard/src/pixel-engine/engine/interactionScheduler.ts +442 -0
  236. package/agent/dashboard/src/pixel-engine/engine/matrixEffect.ts +139 -0
  237. package/agent/dashboard/src/pixel-engine/engine/officeState.ts +748 -0
  238. package/agent/dashboard/src/pixel-engine/engine/renderer.ts +693 -0
  239. package/agent/dashboard/src/pixel-engine/floorTiles.ts +74 -0
  240. package/agent/dashboard/src/pixel-engine/hooks/useAssetLoader.ts +488 -0
  241. package/agent/dashboard/src/pixel-engine/hooks/useEditorActions.ts +661 -0
  242. package/agent/dashboard/src/pixel-engine/hooks/useEditorKeyboard.ts +83 -0
  243. package/agent/dashboard/src/pixel-engine/hooks/useLayoutPersistence.ts +162 -0
  244. package/agent/dashboard/src/pixel-engine/hooks/useOfficeStream.ts +353 -0
  245. package/agent/dashboard/src/pixel-engine/layout/furnitureCatalog.ts +417 -0
  246. package/agent/dashboard/src/pixel-engine/layout/layoutSerializer.ts +416 -0
  247. package/agent/dashboard/src/pixel-engine/layout/tileMap.ts +109 -0
  248. package/agent/dashboard/src/pixel-engine/notificationSound.ts +69 -0
  249. package/agent/dashboard/src/pixel-engine/pixel-office.css +55 -0
  250. package/agent/dashboard/src/pixel-engine/sprites/bubble-permission.json +27 -0
  251. package/agent/dashboard/src/pixel-engine/sprites/bubble-waiting.json +27 -0
  252. package/agent/dashboard/src/pixel-engine/sprites/spriteCache.ts +82 -0
  253. package/agent/dashboard/src/pixel-engine/sprites/spriteData.ts +176 -0
  254. package/agent/dashboard/src/pixel-engine/toolUtils.ts +28 -0
  255. package/agent/dashboard/src/pixel-engine/types.ts +232 -0
  256. package/agent/dashboard/src/pixel-engine/wallTiles.ts +203 -0
  257. package/agent/dashboard/src/views/auth.tsx +249 -0
  258. package/agent/dashboard/src/views/fleet.tsx +263 -0
  259. package/agent/dashboard/src/views/network.tsx +106 -0
  260. package/agent/dashboard/src/views/schedule-types.ts +12 -0
  261. package/agent/dashboard/src/views/schedule-utils.ts +49 -0
  262. package/agent/dashboard/src/views/schedule.tsx +326 -0
  263. package/agent/dashboard/src/views/sessions/conversation-viewer.tsx +279 -0
  264. package/agent/dashboard/src/views/sessions/session-detail.tsx +48 -0
  265. package/agent/dashboard/src/views/sessions/session-list.tsx +265 -0
  266. package/agent/dashboard/src/views/sessions/terminal-pane.tsx +270 -0
  267. package/agent/dashboard/src/views/sessions.tsx +55 -0
  268. package/agent/dashboard/src/views/settings-config.tsx +265 -0
  269. package/agent/dashboard/src/views/settings-logs.tsx +99 -0
  270. package/agent/dashboard/src/views/settings-status.tsx +215 -0
  271. package/agent/dashboard/src/views/settings.tsx +324 -0
  272. package/agent/dashboard/src/views/webhooks.tsx +230 -0
  273. package/agent/dashboard/src/vite-env.d.ts +3 -0
  274. package/agent/dashboard/tsconfig.json +13 -0
  275. package/agent/entry.ts +168 -0
  276. package/agent/event-log.ts +103 -0
  277. package/agent/node-pty-adapter.ts +39 -0
  278. package/agent/paths.ts +12 -0
  279. package/agent/pty-manager.ts +265 -0
  280. package/agent/pty-types.ts +25 -0
  281. package/agent/routes/api.ts +149 -0
  282. package/agent/routes/config.ts +116 -0
  283. package/agent/routes/dashboard.ts +101 -0
  284. package/agent/routes/schedule.ts +84 -0
  285. package/agent/routes/sessions.ts +43 -0
  286. package/agent/routes/webhooks.ts +113 -0
  287. package/agent/scheduler.ts +373 -0
  288. package/agent/server-schema.ts +25 -0
  289. package/agent/server-utils.ts +163 -0
  290. package/agent/server.ts +218 -0
  291. package/agent/server.types.ts +39 -0
  292. package/agent/session-history-types.ts +78 -0
  293. package/agent/session-history-utils.ts +96 -0
  294. package/agent/session-history.ts +318 -0
  295. package/agent/session.ts +152 -0
  296. package/agent/tools/mecha-call.ts +106 -0
  297. package/agent/tools/mecha-fleet.ts +176 -0
  298. package/agent/tools/mecha-list.ts +56 -0
  299. package/agent/tools/mecha-server.ts +70 -0
  300. package/agent/types.ts +35 -0
  301. package/agent/webhook.ts +127 -0
  302. package/agent/ws-terminal.ts +174 -0
  303. package/dist/agent/activity.d.ts +14 -0
  304. package/dist/agent/activity.d.ts.map +1 -0
  305. package/dist/agent/activity.js +25 -0
  306. package/dist/agent/activity.js.map +1 -0
  307. package/dist/agent/character-config.test.d.ts +2 -0
  308. package/dist/agent/character-config.test.d.ts.map +1 -0
  309. package/dist/agent/character-config.test.js +51 -0
  310. package/dist/agent/character-config.test.js.map +1 -0
  311. package/dist/agent/costs.d.ts +16 -0
  312. package/dist/agent/costs.d.ts.map +1 -0
  313. package/dist/agent/costs.js +82 -0
  314. package/dist/agent/costs.js.map +1 -0
  315. package/dist/agent/dashboard/dist/assets/addon-fit-YJmn1quW.js +12 -0
  316. package/dist/agent/dashboard/dist/assets/addon-web-links-BWjmmSgS.js +12 -0
  317. package/dist/agent/dashboard/dist/assets/index-B4OKDQ7e.css +32 -0
  318. package/dist/agent/dashboard/dist/assets/index-fZ_N6Wsi.js +193 -0
  319. package/dist/agent/dashboard/dist/assets/xterm-BqvuqXEL.js +27 -0
  320. package/dist/agent/dashboard/dist/index.html +19 -0
  321. package/dist/agent/dashboard/dist/pixel-engine/characters/char_0.png +0 -0
  322. package/dist/agent/dashboard/dist/pixel-engine/characters/char_1.png +0 -0
  323. package/dist/agent/dashboard/dist/pixel-engine/characters/char_2.png +0 -0
  324. package/dist/agent/dashboard/dist/pixel-engine/characters/char_3.png +0 -0
  325. package/dist/agent/dashboard/dist/pixel-engine/characters/char_4.png +0 -0
  326. package/dist/agent/dashboard/dist/pixel-engine/characters/char_5.png +0 -0
  327. package/dist/agent/dashboard/dist/pixel-engine/default-layout.json +4320 -0
  328. package/dist/agent/dashboard/dist/pixel-engine/floors/floor_0.png +0 -0
  329. package/dist/agent/dashboard/dist/pixel-engine/floors/floor_1.png +0 -0
  330. package/dist/agent/dashboard/dist/pixel-engine/floors/floor_2.png +0 -0
  331. package/dist/agent/dashboard/dist/pixel-engine/floors/floor_3.png +0 -0
  332. package/dist/agent/dashboard/dist/pixel-engine/floors/floor_4.png +0 -0
  333. package/dist/agent/dashboard/dist/pixel-engine/floors/floor_5.png +0 -0
  334. package/dist/agent/dashboard/dist/pixel-engine/floors/floor_6.png +0 -0
  335. package/dist/agent/dashboard/dist/pixel-engine/floors/floor_7.png +0 -0
  336. package/dist/agent/dashboard/dist/pixel-engine/floors/floor_8.png +0 -0
  337. package/dist/agent/dashboard/dist/pixel-engine/fonts/FSPixelSansUnicode-Regular.ttf +0 -0
  338. package/dist/agent/dashboard/dist/pixel-engine/furniture/BIN/BIN.png +0 -0
  339. package/dist/agent/dashboard/dist/pixel-engine/furniture/BIN/manifest.json +13 -0
  340. package/dist/agent/dashboard/dist/pixel-engine/furniture/BLACKBOARD/BLACKBOARD.png +0 -0
  341. package/dist/agent/dashboard/dist/pixel-engine/furniture/BLACKBOARD/manifest.json +16 -0
  342. package/dist/agent/dashboard/dist/pixel-engine/furniture/BOOKSHELF/BOOKSHELF.png +0 -0
  343. package/dist/agent/dashboard/dist/pixel-engine/furniture/BOOKSHELF/manifest.json +16 -0
  344. package/dist/agent/dashboard/dist/pixel-engine/furniture/CACTUS/CACTUS.png +0 -0
  345. package/dist/agent/dashboard/dist/pixel-engine/furniture/CACTUS/manifest.json +13 -0
  346. package/dist/agent/dashboard/dist/pixel-engine/furniture/CLOCK/CLOCK.png +0 -0
  347. package/dist/agent/dashboard/dist/pixel-engine/furniture/CLOCK/manifest.json +13 -0
  348. package/dist/agent/dashboard/dist/pixel-engine/furniture/COFFEE/COFFEE.png +0 -0
  349. package/dist/agent/dashboard/dist/pixel-engine/furniture/COFFEE/manifest.json +13 -0
  350. package/dist/agent/dashboard/dist/pixel-engine/furniture/COFFEE_TABLE/COFFEE_TABLE.png +0 -0
  351. package/dist/agent/dashboard/dist/pixel-engine/furniture/COFFEE_TABLE/manifest.json +13 -0
  352. package/dist/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_BENCH/CUSHIONED_BENCH.png +0 -0
  353. package/dist/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_BENCH/manifest.json +13 -0
  354. package/dist/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_CHAIR/CUSHIONED_CHAIR_BACK.png +0 -0
  355. package/dist/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_CHAIR/CUSHIONED_CHAIR_FRONT.png +0 -0
  356. package/dist/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_CHAIR/CUSHIONED_CHAIR_SIDE.png +0 -0
  357. package/dist/agent/dashboard/dist/pixel-engine/furniture/CUSHIONED_CHAIR/manifest.json +44 -0
  358. package/dist/agent/dashboard/dist/pixel-engine/furniture/DESK/DESK_FRONT.png +0 -0
  359. package/dist/agent/dashboard/dist/pixel-engine/furniture/DESK/DESK_SIDE.png +0 -0
  360. package/dist/agent/dashboard/dist/pixel-engine/furniture/DESK/manifest.json +33 -0
  361. package/dist/agent/dashboard/dist/pixel-engine/furniture/DOUBLE_BOOKSHELF/DOUBLE_BOOKSHELF.png +0 -0
  362. package/dist/agent/dashboard/dist/pixel-engine/furniture/DOUBLE_BOOKSHELF/manifest.json +16 -0
  363. package/dist/agent/dashboard/dist/pixel-engine/furniture/FILE_BOXES_1/FILE_BOXES_1.png +0 -0
  364. package/dist/agent/dashboard/dist/pixel-engine/furniture/FILE_BOXES_1/manifest.json +13 -0
  365. package/dist/agent/dashboard/dist/pixel-engine/furniture/FILE_BOXES_2/FILE_BOXES_2.png +0 -0
  366. package/dist/agent/dashboard/dist/pixel-engine/furniture/FILE_BOXES_2/manifest.json +13 -0
  367. package/dist/agent/dashboard/dist/pixel-engine/furniture/HANGING_PLANT/HANGING_PLANT.png +0 -0
  368. package/dist/agent/dashboard/dist/pixel-engine/furniture/HANGING_PLANT/manifest.json +13 -0
  369. package/dist/agent/dashboard/dist/pixel-engine/furniture/LARGE_PAINTING/LARGE_PAINTING.png +0 -0
  370. package/dist/agent/dashboard/dist/pixel-engine/furniture/LARGE_PAINTING/manifest.json +13 -0
  371. package/dist/agent/dashboard/dist/pixel-engine/furniture/LARGE_PLANT/LARGE_PLANT.png +0 -0
  372. package/dist/agent/dashboard/dist/pixel-engine/furniture/LARGE_PLANT/manifest.json +13 -0
  373. package/dist/agent/dashboard/dist/pixel-engine/furniture/PC/PC_BACK.png +0 -0
  374. package/dist/agent/dashboard/dist/pixel-engine/furniture/PC/PC_FRONT_OFF.png +0 -0
  375. package/dist/agent/dashboard/dist/pixel-engine/furniture/PC/PC_FRONT_ON_1.png +0 -0
  376. package/dist/agent/dashboard/dist/pixel-engine/furniture/PC/PC_FRONT_ON_2.png +0 -0
  377. package/dist/agent/dashboard/dist/pixel-engine/furniture/PC/PC_FRONT_ON_3.png +0 -0
  378. package/dist/agent/dashboard/dist/pixel-engine/furniture/PC/PC_SIDE.png +0 -0
  379. package/dist/agent/dashboard/dist/pixel-engine/furniture/PC/manifest.json +88 -0
  380. package/dist/agent/dashboard/dist/pixel-engine/furniture/PLANT/PLANT.png +0 -0
  381. package/dist/agent/dashboard/dist/pixel-engine/furniture/PLANT/manifest.json +13 -0
  382. package/dist/agent/dashboard/dist/pixel-engine/furniture/PLANT_2/PLANT_2.png +0 -0
  383. package/dist/agent/dashboard/dist/pixel-engine/furniture/PLANT_2/manifest.json +13 -0
  384. package/dist/agent/dashboard/dist/pixel-engine/furniture/POT/POT.png +0 -0
  385. package/dist/agent/dashboard/dist/pixel-engine/furniture/POT/manifest.json +13 -0
  386. package/dist/agent/dashboard/dist/pixel-engine/furniture/SMALL_PAINTING/SMALL_PAINTING.png +0 -0
  387. package/dist/agent/dashboard/dist/pixel-engine/furniture/SMALL_PAINTING/manifest.json +13 -0
  388. package/dist/agent/dashboard/dist/pixel-engine/furniture/SMALL_PAINTING_2/SMALL_PAINTING_2.png +0 -0
  389. package/dist/agent/dashboard/dist/pixel-engine/furniture/SMALL_PAINTING_2/manifest.json +13 -0
  390. package/dist/agent/dashboard/dist/pixel-engine/furniture/SMALL_TABLE/SMALL_TABLE_FRONT.png +0 -0
  391. package/dist/agent/dashboard/dist/pixel-engine/furniture/SMALL_TABLE/SMALL_TABLE_SIDE.png +0 -0
  392. package/dist/agent/dashboard/dist/pixel-engine/furniture/SMALL_TABLE/manifest.json +33 -0
  393. package/dist/agent/dashboard/dist/pixel-engine/furniture/SOFA/SOFA_BACK.png +0 -0
  394. package/dist/agent/dashboard/dist/pixel-engine/furniture/SOFA/SOFA_FRONT.png +0 -0
  395. package/dist/agent/dashboard/dist/pixel-engine/furniture/SOFA/SOFA_SIDE.png +0 -0
  396. package/dist/agent/dashboard/dist/pixel-engine/furniture/SOFA/manifest.json +44 -0
  397. package/dist/agent/dashboard/dist/pixel-engine/furniture/TABLE_FRONT/TABLE_FRONT.png +0 -0
  398. package/dist/agent/dashboard/dist/pixel-engine/furniture/TABLE_FRONT/manifest.json +13 -0
  399. package/dist/agent/dashboard/dist/pixel-engine/furniture/VENDER_MACHINE/VENDER_MACHINE.png +0 -0
  400. package/dist/agent/dashboard/dist/pixel-engine/furniture/VENDER_MACHINE/manifest.json +16 -0
  401. package/dist/agent/dashboard/dist/pixel-engine/furniture/WATER_MACHINE/WATER_MACHINE.png +0 -0
  402. package/dist/agent/dashboard/dist/pixel-engine/furniture/WATER_MACHINE/manifest.json +16 -0
  403. package/dist/agent/dashboard/dist/pixel-engine/furniture/WHITEBOARD/WHITEBOARD.png +0 -0
  404. package/dist/agent/dashboard/dist/pixel-engine/furniture/WHITEBOARD/manifest.json +16 -0
  405. package/dist/agent/dashboard/dist/pixel-engine/furniture/WOODEN_BENCH/WOODEN_BENCH.png +0 -0
  406. package/dist/agent/dashboard/dist/pixel-engine/furniture/WOODEN_BENCH/manifest.json +13 -0
  407. package/dist/agent/dashboard/dist/pixel-engine/furniture/WOODEN_CHAIR/WOODEN_CHAIR_BACK.png +0 -0
  408. package/dist/agent/dashboard/dist/pixel-engine/furniture/WOODEN_CHAIR/WOODEN_CHAIR_FRONT.png +0 -0
  409. package/dist/agent/dashboard/dist/pixel-engine/furniture/WOODEN_CHAIR/WOODEN_CHAIR_SIDE.png +0 -0
  410. package/dist/agent/dashboard/dist/pixel-engine/furniture/WOODEN_CHAIR/manifest.json +44 -0
  411. package/dist/agent/dashboard/dist/pixel-engine/furniture/index.json +32 -0
  412. package/dist/agent/dashboard/dist/pixel-engine/walls/wall_0.png +0 -0
  413. package/dist/agent/entry.d.ts +2 -0
  414. package/dist/agent/entry.d.ts.map +1 -0
  415. package/dist/agent/entry.js +157 -0
  416. package/dist/agent/entry.js.map +1 -0
  417. package/dist/agent/event-log.d.ts +11 -0
  418. package/dist/agent/event-log.d.ts.map +1 -0
  419. package/dist/agent/event-log.js +102 -0
  420. package/dist/agent/event-log.js.map +1 -0
  421. package/dist/agent/node-pty-adapter.d.ts +4 -0
  422. package/dist/agent/node-pty-adapter.d.ts.map +1 -0
  423. package/dist/agent/node-pty-adapter.js +35 -0
  424. package/dist/agent/node-pty-adapter.js.map +1 -0
  425. package/dist/agent/paths.d.ts +11 -0
  426. package/dist/agent/paths.d.ts.map +1 -0
  427. package/dist/agent/paths.js +12 -0
  428. package/dist/agent/paths.js.map +1 -0
  429. package/dist/agent/pty-manager.d.ts +34 -0
  430. package/dist/agent/pty-manager.d.ts.map +1 -0
  431. package/dist/agent/pty-manager.js +210 -0
  432. package/dist/agent/pty-manager.js.map +1 -0
  433. package/dist/agent/pty-types.d.ts +26 -0
  434. package/dist/agent/pty-types.d.ts.map +1 -0
  435. package/dist/agent/pty-types.js +2 -0
  436. package/dist/agent/pty-types.js.map +1 -0
  437. package/dist/agent/routes/api.d.ts +17 -0
  438. package/dist/agent/routes/api.d.ts.map +1 -0
  439. package/dist/agent/routes/api.js +125 -0
  440. package/dist/agent/routes/api.js.map +1 -0
  441. package/dist/agent/routes/config.d.ts +6 -0
  442. package/dist/agent/routes/config.d.ts.map +1 -0
  443. package/dist/agent/routes/config.js +116 -0
  444. package/dist/agent/routes/config.js.map +1 -0
  445. package/dist/agent/routes/dashboard.d.ts +4 -0
  446. package/dist/agent/routes/dashboard.d.ts.map +1 -0
  447. package/dist/agent/routes/dashboard.js +97 -0
  448. package/dist/agent/routes/dashboard.js.map +1 -0
  449. package/dist/agent/routes/schedule.d.ts +4 -0
  450. package/dist/agent/routes/schedule.d.ts.map +1 -0
  451. package/dist/agent/routes/schedule.js +98 -0
  452. package/dist/agent/routes/schedule.js.map +1 -0
  453. package/dist/agent/routes/sessions.d.ts +4 -0
  454. package/dist/agent/routes/sessions.d.ts.map +1 -0
  455. package/dist/agent/routes/sessions.js +43 -0
  456. package/dist/agent/routes/sessions.js.map +1 -0
  457. package/dist/agent/routes/webhooks.d.ts +5 -0
  458. package/dist/agent/routes/webhooks.d.ts.map +1 -0
  459. package/dist/agent/routes/webhooks.js +102 -0
  460. package/dist/agent/routes/webhooks.js.map +1 -0
  461. package/dist/agent/scheduler.d.ts +50 -0
  462. package/dist/agent/scheduler.d.ts.map +1 -0
  463. package/dist/agent/scheduler.js +332 -0
  464. package/dist/agent/scheduler.js.map +1 -0
  465. package/dist/agent/server-schema.d.ts +20 -0
  466. package/dist/agent/server-schema.d.ts.map +1 -0
  467. package/dist/agent/server-schema.js +20 -0
  468. package/dist/agent/server-schema.js.map +1 -0
  469. package/dist/agent/server-utils.d.ts +18 -0
  470. package/dist/agent/server-utils.d.ts.map +1 -0
  471. package/dist/agent/server-utils.js +132 -0
  472. package/dist/agent/server-utils.js.map +1 -0
  473. package/dist/agent/server.d.ts +17 -0
  474. package/dist/agent/server.d.ts.map +1 -0
  475. package/dist/agent/server.js +196 -0
  476. package/dist/agent/server.js.map +1 -0
  477. package/dist/agent/server.types.d.ts +38 -0
  478. package/dist/agent/server.types.d.ts.map +1 -0
  479. package/dist/agent/server.types.js +2 -0
  480. package/dist/agent/server.types.js.map +1 -0
  481. package/dist/agent/session-history-types.d.ts +74 -0
  482. package/dist/agent/session-history-types.d.ts.map +1 -0
  483. package/dist/agent/session-history-types.js +2 -0
  484. package/dist/agent/session-history-types.js.map +1 -0
  485. package/dist/agent/session-history-utils.d.ts +13 -0
  486. package/dist/agent/session-history-utils.d.ts.map +1 -0
  487. package/dist/agent/session-history-utils.js +102 -0
  488. package/dist/agent/session-history-utils.js.map +1 -0
  489. package/dist/agent/session-history.d.ts +20 -0
  490. package/dist/agent/session-history.d.ts.map +1 -0
  491. package/dist/agent/session-history.js +282 -0
  492. package/dist/agent/session-history.js.map +1 -0
  493. package/dist/agent/session.d.ts +51 -0
  494. package/dist/agent/session.d.ts.map +1 -0
  495. package/dist/agent/session.js +135 -0
  496. package/dist/agent/session.js.map +1 -0
  497. package/dist/agent/tools/mecha-call.d.ts +2 -0
  498. package/dist/agent/tools/mecha-call.d.ts.map +1 -0
  499. package/dist/agent/tools/mecha-call.js +101 -0
  500. package/dist/agent/tools/mecha-call.js.map +1 -0
  501. package/dist/agent/tools/mecha-fleet.d.ts +25 -0
  502. package/dist/agent/tools/mecha-fleet.d.ts.map +1 -0
  503. package/dist/agent/tools/mecha-fleet.js +144 -0
  504. package/dist/agent/tools/mecha-fleet.js.map +1 -0
  505. package/dist/agent/tools/mecha-list.d.ts +6 -0
  506. package/dist/agent/tools/mecha-list.d.ts.map +1 -0
  507. package/dist/agent/tools/mecha-list.js +46 -0
  508. package/dist/agent/tools/mecha-list.js.map +1 -0
  509. package/dist/agent/tools/mecha-server.d.ts +4 -0
  510. package/dist/agent/tools/mecha-server.d.ts.map +1 -0
  511. package/dist/agent/tools/mecha-server.js +49 -0
  512. package/dist/agent/tools/mecha-server.js.map +1 -0
  513. package/dist/agent/types.d.ts +38 -0
  514. package/dist/agent/types.d.ts.map +1 -0
  515. package/dist/agent/types.js +32 -0
  516. package/dist/agent/types.js.map +1 -0
  517. package/dist/agent/webhook.d.ts +12 -0
  518. package/dist/agent/webhook.d.ts.map +1 -0
  519. package/dist/agent/webhook.js +105 -0
  520. package/dist/agent/webhook.js.map +1 -0
  521. package/dist/agent/ws-terminal.d.ts +4 -0
  522. package/dist/agent/ws-terminal.d.ts.map +1 -0
  523. package/dist/agent/ws-terminal.js +160 -0
  524. package/dist/agent/ws-terminal.js.map +1 -0
  525. package/dist/shared/atomic-write.d.ts +5 -0
  526. package/dist/shared/atomic-write.d.ts.map +1 -0
  527. package/dist/shared/atomic-write.js +63 -0
  528. package/dist/shared/atomic-write.js.map +1 -0
  529. package/dist/shared/errors.d.ts +71 -0
  530. package/dist/shared/errors.d.ts.map +1 -0
  531. package/dist/shared/errors.js +52 -0
  532. package/dist/shared/errors.js.map +1 -0
  533. package/dist/shared/logger.d.ts +7 -0
  534. package/dist/shared/logger.d.ts.map +1 -0
  535. package/dist/shared/logger.js +52 -0
  536. package/dist/shared/logger.js.map +1 -0
  537. package/dist/shared/mutex.d.ts +12 -0
  538. package/dist/shared/mutex.d.ts.map +1 -0
  539. package/dist/shared/mutex.js +73 -0
  540. package/dist/shared/mutex.js.map +1 -0
  541. package/dist/shared/safe-read.d.ts +12 -0
  542. package/dist/shared/safe-read.d.ts.map +1 -0
  543. package/dist/shared/safe-read.js +33 -0
  544. package/dist/shared/safe-read.js.map +1 -0
  545. package/dist/shared/totp.d.ts +7 -0
  546. package/dist/shared/totp.d.ts.map +1 -0
  547. package/dist/shared/totp.js +75 -0
  548. package/dist/shared/totp.js.map +1 -0
  549. package/dist/shared/validation.d.ts +7 -0
  550. package/dist/shared/validation.d.ts.map +1 -0
  551. package/dist/shared/validation.js +30 -0
  552. package/dist/shared/validation.js.map +1 -0
  553. package/dist/src/auth.d.ts +48 -0
  554. package/dist/src/auth.d.ts.map +1 -0
  555. package/dist/src/auth.js +160 -0
  556. package/dist/src/auth.js.map +1 -0
  557. package/dist/src/cli-output.d.ts +14 -0
  558. package/dist/src/cli-output.d.ts.map +1 -0
  559. package/dist/src/cli-output.js +47 -0
  560. package/dist/src/cli-output.js.map +1 -0
  561. package/dist/src/cli-utils.d.ts +15 -0
  562. package/dist/src/cli-utils.d.ts.map +1 -0
  563. package/dist/src/cli-utils.js +241 -0
  564. package/dist/src/cli-utils.js.map +1 -0
  565. package/dist/src/cli.d.ts +3 -0
  566. package/dist/src/cli.d.ts.map +1 -0
  567. package/dist/src/cli.js +535 -0
  568. package/dist/src/cli.js.map +1 -0
  569. package/dist/src/cli.utils.d.ts +10 -0
  570. package/dist/src/cli.utils.d.ts.map +1 -0
  571. package/dist/src/cli.utils.js +127 -0
  572. package/dist/src/cli.utils.js.map +1 -0
  573. package/dist/src/commands/auth.d.ts +3 -0
  574. package/dist/src/commands/auth.d.ts.map +1 -0
  575. package/dist/src/commands/auth.js +139 -0
  576. package/dist/src/commands/auth.js.map +1 -0
  577. package/dist/src/commands/bot-api.d.ts +15 -0
  578. package/dist/src/commands/bot-api.d.ts.map +1 -0
  579. package/dist/src/commands/bot-api.js +40 -0
  580. package/dist/src/commands/bot-api.js.map +1 -0
  581. package/dist/src/commands/completion.d.ts +3 -0
  582. package/dist/src/commands/completion.d.ts.map +1 -0
  583. package/dist/src/commands/completion.js +108 -0
  584. package/dist/src/commands/completion.js.map +1 -0
  585. package/dist/src/commands/config.d.ts +3 -0
  586. package/dist/src/commands/config.d.ts.map +1 -0
  587. package/dist/src/commands/config.js +110 -0
  588. package/dist/src/commands/config.js.map +1 -0
  589. package/dist/src/commands/costs.d.ts +3 -0
  590. package/dist/src/commands/costs.d.ts.map +1 -0
  591. package/dist/src/commands/costs.js +88 -0
  592. package/dist/src/commands/costs.js.map +1 -0
  593. package/dist/src/commands/daemon.d.ts +3 -0
  594. package/dist/src/commands/daemon.d.ts.map +1 -0
  595. package/dist/src/commands/daemon.js +86 -0
  596. package/dist/src/commands/daemon.js.map +1 -0
  597. package/dist/src/commands/push-dashboard.d.ts +3 -0
  598. package/dist/src/commands/push-dashboard.d.ts.map +1 -0
  599. package/dist/src/commands/push-dashboard.js +48 -0
  600. package/dist/src/commands/push-dashboard.js.map +1 -0
  601. package/dist/src/commands/schedule.d.ts +3 -0
  602. package/dist/src/commands/schedule.d.ts.map +1 -0
  603. package/dist/src/commands/schedule.js +70 -0
  604. package/dist/src/commands/schedule.js.map +1 -0
  605. package/dist/src/commands/sessions.d.ts +3 -0
  606. package/dist/src/commands/sessions.d.ts.map +1 -0
  607. package/dist/src/commands/sessions.js +62 -0
  608. package/dist/src/commands/sessions.js.map +1 -0
  609. package/dist/src/commands/webhooks.d.ts +3 -0
  610. package/dist/src/commands/webhooks.d.ts.map +1 -0
  611. package/dist/src/commands/webhooks.js +44 -0
  612. package/dist/src/commands/webhooks.js.map +1 -0
  613. package/dist/src/config.d.ts +9 -0
  614. package/dist/src/config.d.ts.map +1 -0
  615. package/dist/src/config.js +56 -0
  616. package/dist/src/config.js.map +1 -0
  617. package/dist/src/daemon-audit.d.ts +13 -0
  618. package/dist/src/daemon-audit.d.ts.map +1 -0
  619. package/dist/src/daemon-audit.js +33 -0
  620. package/dist/src/daemon-audit.js.map +1 -0
  621. package/dist/src/daemon.d.ts +25 -0
  622. package/dist/src/daemon.d.ts.map +1 -0
  623. package/dist/src/daemon.js +344 -0
  624. package/dist/src/daemon.js.map +1 -0
  625. package/dist/src/dashboard-server-schema.d.ts +23 -0
  626. package/dist/src/dashboard-server-schema.d.ts.map +1 -0
  627. package/dist/src/dashboard-server-schema.js +45 -0
  628. package/dist/src/dashboard-server-schema.js.map +1 -0
  629. package/dist/src/dashboard-server-utils.d.ts +18 -0
  630. package/dist/src/dashboard-server-utils.d.ts.map +1 -0
  631. package/dist/src/dashboard-server-utils.js +107 -0
  632. package/dist/src/dashboard-server-utils.js.map +1 -0
  633. package/dist/src/dashboard-server.d.ts +2 -0
  634. package/dist/src/dashboard-server.d.ts.map +1 -0
  635. package/dist/src/dashboard-server.js +777 -0
  636. package/dist/src/dashboard-server.js.map +1 -0
  637. package/dist/src/docker.constants.d.ts +5 -0
  638. package/dist/src/docker.constants.d.ts.map +1 -0
  639. package/dist/src/docker.constants.js +7 -0
  640. package/dist/src/docker.constants.js.map +1 -0
  641. package/dist/src/docker.d.ts +16 -0
  642. package/dist/src/docker.d.ts.map +1 -0
  643. package/dist/src/docker.js +437 -0
  644. package/dist/src/docker.js.map +1 -0
  645. package/dist/src/docker.types.d.ts +13 -0
  646. package/dist/src/docker.types.d.ts.map +1 -0
  647. package/dist/src/docker.types.js +2 -0
  648. package/dist/src/docker.types.js.map +1 -0
  649. package/dist/src/docker.utils.d.ts +24 -0
  650. package/dist/src/docker.utils.d.ts.map +1 -0
  651. package/dist/src/docker.utils.js +239 -0
  652. package/dist/src/docker.utils.js.map +1 -0
  653. package/dist/src/doctor.d.ts +3 -0
  654. package/dist/src/doctor.d.ts.map +1 -0
  655. package/dist/src/doctor.js +80 -0
  656. package/dist/src/doctor.js.map +1 -0
  657. package/dist/src/doctor.utils.d.ts +30 -0
  658. package/dist/src/doctor.utils.d.ts.map +1 -0
  659. package/dist/src/doctor.utils.js +192 -0
  660. package/dist/src/doctor.utils.js.map +1 -0
  661. package/dist/src/mcp-proxy.d.ts +2 -0
  662. package/dist/src/mcp-proxy.d.ts.map +1 -0
  663. package/dist/src/mcp-proxy.js +177 -0
  664. package/dist/src/mcp-proxy.js.map +1 -0
  665. package/dist/src/resolve-endpoint.d.ts +13 -0
  666. package/dist/src/resolve-endpoint.d.ts.map +1 -0
  667. package/dist/src/resolve-endpoint.js +112 -0
  668. package/dist/src/resolve-endpoint.js.map +1 -0
  669. package/dist/src/store.d.ts +41 -0
  670. package/dist/src/store.d.ts.map +1 -0
  671. package/dist/src/store.js +263 -0
  672. package/dist/src/store.js.map +1 -0
  673. package/package.json +84 -0
  674. package/s6/mecha-agent/dependencies.d/tailscale-up +0 -0
  675. package/s6/mecha-agent/finish +4 -0
  676. package/s6/mecha-agent/run +39 -0
  677. package/s6/mecha-agent/type +1 -0
  678. package/s6/tailscale-up/dependencies.d/tailscaled +0 -0
  679. package/s6/tailscale-up/type +1 -0
  680. package/s6/tailscale-up/up +39 -0
  681. package/s6/tailscaled/run +14 -0
  682. package/s6/tailscaled/type +1 -0
  683. package/shared/atomic-write.ts +58 -0
  684. package/shared/errors.ts +137 -0
  685. package/shared/logger.ts +50 -0
  686. package/shared/mutex.ts +73 -0
  687. package/shared/safe-read.ts +52 -0
  688. package/shared/totp.ts +83 -0
  689. package/shared/validation.ts +29 -0
  690. package/tsconfig.agent.json +10 -0
  691. package/tsconfig.json +22 -0
@@ -0,0 +1,777 @@
1
+ import { Hono } from "hono";
2
+ import { cors } from "hono/cors";
3
+ import { streamSSE } from "hono/streaming";
4
+ import { serve } from "@hono/node-server";
5
+ import { serveStatic } from "@hono/node-server/serve-static";
6
+ import { join, dirname, resolve } from "node:path";
7
+ import { fileURLToPath } from "node:url";
8
+ import { createHash } from "node:crypto";
9
+ import * as docker from "./docker.js";
10
+ import { withBotLock } from "./docker.utils.js";
11
+ import { listBots as listRegistered } from "./store.js";
12
+ import { loadBotConfig, buildInlineConfig } from "./config.js";
13
+ import { addCredential, detectCredentialType, listCredentials, loadCredentials } from "./auth.js";
14
+ import { existsSync, readFileSync as fsReadFileSync, realpathSync, statSync } from "node:fs";
15
+ import { isValidName } from "../shared/validation.js";
16
+ import { resolveHostBotBaseUrl } from "./resolve-endpoint.js";
17
+ import { DASHBOARD_TOKEN, HOP_BY_HOP, spawnBodySchema, authBodySchema, totpVerifySchema, revokeAllSessions } from "./dashboard-server-schema.js";
18
+ import { safeError, dashboardSessionCookie, hasDashboardAccess, shouldBootstrapDashboardSession, guardBusy } from "./dashboard-server-utils.js";
19
+ import { getTotpSecret, setTotpSecret, clearTotpSecret, getMechaDir, getOrCreateFleetInternalSecret } from "./store.js";
20
+ import { timingSafeEqual } from "node:crypto";
21
+ import { generateSecret, verifyTOTP, totpUri } from "../shared/totp.js";
22
+ import { atomicWriteJsonAsync } from "../shared/atomic-write.js";
23
+ const __dirname = dirname(fileURLToPath(import.meta.url));
24
+ export function startDashboardServer(port, host) {
25
+ const app = new Hono();
26
+ // CORS — only allow same-origin
27
+ app.use("/*", cors({ origin: `http://localhost:${port}` }));
28
+ app.use("/*", async (c, next) => {
29
+ await next();
30
+ if (shouldBootstrapDashboardSession(c)) {
31
+ c.header("Set-Cookie", dashboardSessionCookie());
32
+ }
33
+ });
34
+ // --- TOTP (unauthenticated) ---
35
+ // Health endpoint (unauthenticated)
36
+ const daemonStartedAt = Date.now();
37
+ // Read version once at startup from package.json
38
+ let _version = "unknown";
39
+ try {
40
+ _version = JSON.parse(fsReadFileSync(join(dirname(fileURLToPath(import.meta.url)), "..", "..", "package.json"), "utf-8")).version ?? "unknown";
41
+ }
42
+ catch { /* best-effort */ }
43
+ app.get("/api/health", async (c) => {
44
+ let running = 0, stopped = 0;
45
+ try {
46
+ const bots = await docker.list();
47
+ running = bots.filter(b => b.status === "running").length;
48
+ stopped = bots.length - running;
49
+ }
50
+ catch { /* Docker may be unavailable */ }
51
+ return c.json({
52
+ status: "ok",
53
+ version: _version,
54
+ uptime: Math.floor((Date.now() - daemonStartedAt) / 1000),
55
+ bots: { running, stopped },
56
+ pid: process.pid,
57
+ });
58
+ });
59
+ app.get("/api/totp/status", (c) => {
60
+ return c.json({ enabled: !!getTotpSecret() });
61
+ });
62
+ // TOTP rate limiting: max 5 attempts per 60 seconds per IP
63
+ const totpAttempts = new Map();
64
+ const TOTP_MAX_ATTEMPTS = 5;
65
+ const TOTP_WINDOW_MS = 60_000;
66
+ app.post("/api/totp/verify", async (c) => {
67
+ const secret = getTotpSecret();
68
+ if (!secret)
69
+ return c.json({ error: "TOTP not enabled" }, 400);
70
+ // Rate limit by IP
71
+ const ip = c.req.header("x-forwarded-for")?.split(",")[0]?.trim() ?? "unknown";
72
+ const now = Date.now();
73
+ let entry = totpAttempts.get(ip);
74
+ if (!entry || now > entry.resetAt) {
75
+ entry = { count: 0, resetAt: now + TOTP_WINDOW_MS };
76
+ totpAttempts.set(ip, entry);
77
+ }
78
+ entry.count++;
79
+ if (entry.count > TOTP_MAX_ATTEMPTS) {
80
+ const retryAfter = Math.ceil((entry.resetAt - now) / 1000);
81
+ c.header("Retry-After", String(retryAfter));
82
+ return c.json({ error: "Too many attempts. Try again later." }, 429);
83
+ }
84
+ const body = await c.req.json().catch(() => null);
85
+ const parsed = totpVerifySchema.safeParse(body);
86
+ if (!parsed.success)
87
+ return c.json({ error: "Invalid code format" }, 400);
88
+ if (!verifyTOTP(secret, parsed.data.code)) {
89
+ return c.json({ error: "Invalid code" }, 401);
90
+ }
91
+ // Success — reset rate limit and set session cookie
92
+ totpAttempts.delete(ip);
93
+ c.header("Set-Cookie", dashboardSessionCookie());
94
+ return c.json({ authenticated: true });
95
+ });
96
+ // Auth middleware for all API and proxy routes (token always required)
97
+ app.use("/api/*", async (c, next) => {
98
+ // Allow TOTP status/verify through without auth
99
+ if (c.req.path === "/api/health" || c.req.path === "/api/totp/status" || c.req.path === "/api/totp/verify" || c.req.path.startsWith("/api/fleet/")) {
100
+ return next();
101
+ }
102
+ // Allow public read-only access to office layout, stream, and avatars
103
+ if ((c.req.path === "/api/office/layout" && c.req.method === "GET") ||
104
+ (c.req.path === "/api/office/stream" && c.req.method === "GET") ||
105
+ (c.req.path === "/api/office/avatars" && c.req.method === "GET")) {
106
+ return next();
107
+ }
108
+ if (!hasDashboardAccess(c)) {
109
+ return c.json({ error: "Unauthorized" }, 401);
110
+ }
111
+ await next();
112
+ });
113
+ app.use("/bot/*", async (c, next) => {
114
+ if (!hasDashboardAccess(c)) {
115
+ return c.json({ error: "Unauthorized" }, 401);
116
+ }
117
+ await next();
118
+ });
119
+ // --- Fleet Internal API (authenticated with MECHA_FLEET_INTERNAL_SECRET) ---
120
+ app.use("/api/fleet/*", async (c, next) => {
121
+ const auth = c.req.header("authorization");
122
+ if (!auth?.startsWith("Bearer "))
123
+ return c.json({ error: "Unauthorized" }, 401);
124
+ const token = auth.slice(7);
125
+ const secret = getOrCreateFleetInternalSecret();
126
+ const bufA = Buffer.from(token);
127
+ const bufB = Buffer.from(secret);
128
+ if (bufA.length !== bufB.length || !timingSafeEqual(bufA, bufB)) {
129
+ return c.json({ error: "Unauthorized" }, 401);
130
+ }
131
+ await next();
132
+ });
133
+ app.get("/api/fleet/bots", async (c) => {
134
+ const bots = await docker.list();
135
+ return c.json(bots);
136
+ });
137
+ app.post("/api/fleet/bots", async (c) => {
138
+ const body = await c.req.json().catch(() => null);
139
+ if (!body?.name || !body?.system)
140
+ return c.json({ error: "name and system required" }, 400);
141
+ const config = buildInlineConfig({ name: body.name, system: body.system, model: body.model });
142
+ if (body.auth)
143
+ config.auth = body.auth;
144
+ const containerId = await docker.spawn(config);
145
+ return c.json({ status: "spawned", name: config.name, containerId: containerId.slice(0, 12) });
146
+ });
147
+ app.post("/api/fleet/bots/:name/start", async (c) => {
148
+ const name = c.req.param("name");
149
+ if (!isValidName(name))
150
+ return c.json({ error: "Invalid bot name" }, 400);
151
+ await docker.start(name);
152
+ return c.json({ status: "started", name });
153
+ });
154
+ app.post("/api/fleet/bots/:name/stop", async (c) => {
155
+ const name = c.req.param("name");
156
+ if (!isValidName(name))
157
+ return c.json({ error: "Invalid bot name" }, 400);
158
+ await docker.stop(name);
159
+ return c.json({ status: "stopped", name });
160
+ });
161
+ app.post("/api/fleet/bots/:name/restart", async (c) => {
162
+ const name = c.req.param("name");
163
+ if (!isValidName(name))
164
+ return c.json({ error: "Invalid bot name" }, 400);
165
+ const containerId = await docker.restart(name);
166
+ return c.json({ status: "restarted", name, containerId: containerId.slice(0, 12) });
167
+ });
168
+ app.delete("/api/fleet/bots/:name", async (c) => {
169
+ const name = c.req.param("name");
170
+ if (!isValidName(name))
171
+ return c.json({ error: "Invalid bot name" }, 400);
172
+ await docker.remove(name);
173
+ return c.json({ status: "removed", name });
174
+ });
175
+ app.get("/api/fleet/bots/:name/config", async (c) => {
176
+ const name = c.req.param("name");
177
+ if (!isValidName(name))
178
+ return c.json({ error: "Invalid bot name" }, 400);
179
+ const entry = listRegistered()[name];
180
+ if (!entry?.config)
181
+ return c.json({ error: "Bot not found" }, 404);
182
+ try {
183
+ const { readFileSync } = await import("node:fs");
184
+ const { parse: parseYaml } = await import("yaml");
185
+ const raw = readFileSync(entry.config, "utf-8");
186
+ const config = parseYaml(raw);
187
+ const field = new URL(c.req.url).searchParams.get("field");
188
+ if (field)
189
+ return c.json({ [field]: config[field] });
190
+ return c.json(config);
191
+ }
192
+ catch {
193
+ return c.json({ error: "Failed to read config" }, 500);
194
+ }
195
+ });
196
+ app.get("/api/fleet/costs", async (c) => {
197
+ const bots = listRegistered();
198
+ const result = {};
199
+ for (const [name, entry] of Object.entries(bots)) {
200
+ if (!entry.path)
201
+ continue;
202
+ try {
203
+ const { readFileSync } = await import("node:fs");
204
+ const { join } = await import("node:path");
205
+ result[name] = JSON.parse(readFileSync(join(entry.path, "costs.json"), "utf-8"));
206
+ }
207
+ catch {
208
+ result[name] = {};
209
+ }
210
+ }
211
+ return c.json(result);
212
+ });
213
+ app.get("/api/fleet/costs/:name", async (c) => {
214
+ const name = c.req.param("name");
215
+ const entry = listRegistered()[name];
216
+ if (!entry?.path)
217
+ return c.json({ error: "Bot not found" }, 404);
218
+ try {
219
+ const { readFileSync } = await import("node:fs");
220
+ const { join } = await import("node:path");
221
+ return c.json(JSON.parse(readFileSync(join(entry.path, "costs.json"), "utf-8")));
222
+ }
223
+ catch {
224
+ return c.json({});
225
+ }
226
+ });
227
+ app.get("/api/fleet/health", async (c) => {
228
+ const bots = await docker.list();
229
+ const running = bots.filter(b => b.status === "running").length;
230
+ return c.json({
231
+ status: "ok",
232
+ version: _version,
233
+ uptime: Math.floor((Date.now() - daemonStartedAt) / 1000),
234
+ bots: { running, stopped: bots.length - running },
235
+ pid: process.pid,
236
+ });
237
+ });
238
+ // --- Dashboard Fleet API (dashboard auth) ---
239
+ app.get("/api/bots", async (c) => {
240
+ const bots = await docker.list();
241
+ return c.json(bots);
242
+ });
243
+ app.get("/api/session", (c) => c.json({ authenticated: true }));
244
+ app.post("/api/bots", async (c) => {
245
+ const body = await c.req.json().catch(() => null);
246
+ if (!body)
247
+ return c.json({ error: "Invalid JSON" }, 400);
248
+ const parsed = spawnBodySchema.safeParse(body);
249
+ if (!parsed.success) {
250
+ return c.json({ error: "Invalid request body" }, 400);
251
+ }
252
+ try {
253
+ let config;
254
+ if (parsed.data.config_path) {
255
+ // Restrict config_path to home directory (follow symlinks to prevent bypass)
256
+ const home = process.env.HOME;
257
+ if (!home) {
258
+ return c.json({ error: "HOME not set" }, 500);
259
+ }
260
+ let absPath;
261
+ try {
262
+ absPath = realpathSync(resolve(parsed.data.config_path));
263
+ }
264
+ catch {
265
+ return c.json({ error: "config_path not found" }, 400);
266
+ }
267
+ if (absPath !== home && !absPath.startsWith(home + "/")) {
268
+ return c.json({ error: "config_path must be under your home directory" }, 400);
269
+ }
270
+ config = loadBotConfig(absPath);
271
+ }
272
+ else if (parsed.data.name && parsed.data.system) {
273
+ config = buildInlineConfig({
274
+ name: parsed.data.name,
275
+ system: parsed.data.system,
276
+ model: parsed.data.model,
277
+ });
278
+ }
279
+ else {
280
+ return c.json({ error: "Provide config_path or name+system" }, 400);
281
+ }
282
+ const dir = parsed.data.dir ? resolve(parsed.data.dir) : undefined;
283
+ const containerId = await docker.spawn(config, dir);
284
+ return c.json({ status: "spawned", name: config.name, containerId: containerId.slice(0, 12) });
285
+ }
286
+ catch (err) {
287
+ return safeError(c, err);
288
+ }
289
+ });
290
+ app.delete("/api/bots/:name", async (c) => {
291
+ const name = c.req.param("name");
292
+ if (!isValidName(name))
293
+ return c.json({ error: "Invalid bot name" }, 400);
294
+ try {
295
+ await docker.remove(name);
296
+ return c.json({ status: "removed", name });
297
+ }
298
+ catch (err) {
299
+ return safeError(c, err);
300
+ }
301
+ });
302
+ app.post("/api/bots/:name/stop", async (c) => {
303
+ const name = c.req.param("name");
304
+ if (!isValidName(name))
305
+ return c.json({ error: "Invalid bot name" }, 400);
306
+ const blocked = await guardBusy(c, name);
307
+ if (blocked)
308
+ return blocked;
309
+ try {
310
+ await docker.stop(name);
311
+ return c.json({ status: "stopped", name });
312
+ }
313
+ catch (err) {
314
+ return safeError(c, err);
315
+ }
316
+ });
317
+ app.post("/api/bots/:name/restart", async (c) => {
318
+ const name = c.req.param("name");
319
+ if (!isValidName(name))
320
+ return c.json({ error: "Invalid bot name" }, 400);
321
+ const blocked = await guardBusy(c, name);
322
+ if (blocked)
323
+ return blocked;
324
+ try {
325
+ const containerId = await docker.restart(name);
326
+ return c.json({ status: "restarted", name, containerId: containerId.slice(0, 12) });
327
+ }
328
+ catch (err) {
329
+ return safeError(c, err);
330
+ }
331
+ });
332
+ // --- Per-bot auth API ---
333
+ app.get("/api/bots/:name/auth", async (c) => {
334
+ const name = c.req.param("name");
335
+ if (!isValidName(name))
336
+ return c.json({ error: "Invalid bot name" }, 400);
337
+ const entry = listRegistered()[name];
338
+ if (!entry?.config)
339
+ return c.json({ error: "Bot not found" }, 404);
340
+ // Read the bot's config to find its auth profile
341
+ let botAuth;
342
+ try {
343
+ const config = loadBotConfig(entry.config);
344
+ botAuth = config.auth;
345
+ }
346
+ catch { /* ignore parse errors */ }
347
+ // List available Claude auth profiles
348
+ const creds = loadCredentials();
349
+ const claudeProfiles = creds
350
+ .filter((cr) => cr.type === "api_key" || cr.type === "oauth_token")
351
+ .map((cr) => ({ name: cr.name, type: cr.type }));
352
+ return c.json({
353
+ current_profile: botAuth ?? null,
354
+ profiles: claudeProfiles,
355
+ });
356
+ });
357
+ app.put("/api/bots/:name/auth", async (c) => {
358
+ const name = c.req.param("name");
359
+ if (!isValidName(name))
360
+ return c.json({ error: "Invalid bot name" }, 400);
361
+ const body = await c.req.json().catch(() => null);
362
+ if (!body || typeof body.profile !== "string") {
363
+ return c.json({ error: "profile is required" }, 400);
364
+ }
365
+ const profile = body.profile;
366
+ // Verify the profile exists and is a Claude auth type
367
+ const creds = loadCredentials();
368
+ const cred = creds.find((cr) => cr.name === profile);
369
+ if (!cred)
370
+ return c.json({ error: `Profile "${profile}" not found` }, 404);
371
+ if (cred.type !== "api_key" && cred.type !== "oauth_token") {
372
+ return c.json({ error: `Profile "${profile}" is not a Claude auth credential` }, 400);
373
+ }
374
+ const entry = listRegistered()[name];
375
+ if (!entry?.config)
376
+ return c.json({ error: "Bot not found" }, 404);
377
+ // Check busy state BEFORE mutating config
378
+ const blocked = await guardBusy(c, name, { profile });
379
+ if (blocked)
380
+ return blocked;
381
+ // Update config and restart under bot lock to prevent concurrent mutations
382
+ try {
383
+ const result = await withBotLock(name, async () => {
384
+ const { readFileSync } = await import("node:fs");
385
+ const { parse: parseYaml, stringify: stringifyYaml } = await import("yaml");
386
+ const configPath = entry.config;
387
+ const raw = readFileSync(configPath, "utf-8");
388
+ const parsed = parseYaml(raw);
389
+ parsed.auth = profile;
390
+ await atomicWriteJsonAsync(configPath, stringifyYaml(parsed));
391
+ const containerId = await docker.restart(name);
392
+ return { status: "switched", profile, containerId: containerId.slice(0, 12) };
393
+ });
394
+ return c.json(result);
395
+ }
396
+ catch (err) {
397
+ return safeError(c, err);
398
+ }
399
+ });
400
+ // --- Auth API ---
401
+ app.get("/api/auth", (c) => c.json(listCredentials().map((c) => c.name)));
402
+ app.post("/api/auth", async (c) => {
403
+ const body = await c.req.json().catch(() => null);
404
+ const parsed = authBodySchema.safeParse(body);
405
+ if (!parsed.success) {
406
+ return c.json({ error: "profile and key required" }, 400);
407
+ }
408
+ if (!isValidName(parsed.data.profile)) {
409
+ return c.json({ error: "Invalid profile name" }, 400);
410
+ }
411
+ const detected = detectCredentialType(parsed.data.key);
412
+ addCredential({ name: parsed.data.profile, ...detected, key: parsed.data.key });
413
+ return c.json({ status: "added", profile: parsed.data.profile });
414
+ });
415
+ // --- TOTP management (authenticated) ---
416
+ app.post("/api/totp/enable", (c) => {
417
+ if (getTotpSecret())
418
+ return c.json({ error: "TOTP already enabled" }, 400);
419
+ const secret = generateSecret();
420
+ setTotpSecret(secret);
421
+ revokeAllSessions(); // invalidate all existing sessions when auth settings change
422
+ const uri = totpUri(secret, "Mecha", "dashboard");
423
+ return c.json({ secret, uri });
424
+ });
425
+ app.delete("/api/totp", async (c) => {
426
+ const secret = getTotpSecret();
427
+ if (!secret)
428
+ return c.json({ error: "TOTP not enabled" }, 400);
429
+ // Require a valid code to disable
430
+ const body = await c.req.json().catch(() => null);
431
+ const parsed = totpVerifySchema.safeParse(body);
432
+ if (!parsed.success)
433
+ return c.json({ error: "Code required to disable TOTP" }, 400);
434
+ if (!verifyTOTP(secret, parsed.data.code)) {
435
+ return c.json({ error: "Invalid code" }, 401);
436
+ }
437
+ clearTotpSecret();
438
+ revokeAllSessions(); // invalidate all sessions when TOTP is disabled
439
+ return c.json({ disabled: true });
440
+ });
441
+ // --- Network: aggregate logs from all bots (parallel) ---
442
+ app.get("/api/network", async (c) => {
443
+ const bots = await docker.list();
444
+ const runningBots = bots.filter((b) => b.status === "running");
445
+ const results = await Promise.allSettled(runningBots.map(async (bot) => {
446
+ const resolved = await resolveHostBotBaseUrl(bot.name, { allowRemote: false });
447
+ if (!resolved)
448
+ return [];
449
+ const botEntry = listRegistered()[bot.name];
450
+ const fetchHeaders = {};
451
+ if (botEntry?.botToken)
452
+ fetchHeaders["Authorization"] = `Bearer ${botEntry.botToken}`;
453
+ const resp = await fetch(`${resolved.baseUrl}/api/logs?limit=50`, {
454
+ headers: fetchHeaders,
455
+ signal: AbortSignal.timeout(3000),
456
+ });
457
+ if (!resp.ok)
458
+ return [];
459
+ const logs = await resp.json();
460
+ return logs.map((e) => ({ ...e, source_bot: bot.name }));
461
+ }));
462
+ const events = [];
463
+ for (const r of results) {
464
+ if (r.status === "fulfilled")
465
+ events.push(...r.value);
466
+ }
467
+ return c.json(events);
468
+ });
469
+ // --- Proxy to individual bot dashboard ---
470
+ app.all("/bot/:name/*", async (c) => {
471
+ const name = c.req.param("name");
472
+ if (!isValidName(name))
473
+ return c.json({ error: "Invalid bot name" }, 400);
474
+ const resolved = await resolveHostBotBaseUrl(name, { allowRemote: false });
475
+ if (!resolved)
476
+ return c.json({ error: `Bot "${name}" not reachable` }, 502);
477
+ const rawPath = c.req.path.replace(`/bot/${name}`, "") || "/";
478
+ // Decode and normalize to catch encoded traversal variants (%2e%2e, etc.)
479
+ let path;
480
+ try {
481
+ path = decodeURIComponent(rawPath);
482
+ }
483
+ catch {
484
+ return c.json({ error: "Invalid path encoding" }, 400);
485
+ }
486
+ if (path.includes("..") || path.includes("\0"))
487
+ return c.json({ error: "Invalid path" }, 400);
488
+ const url = new URL(c.req.url);
489
+ const targetUrl = `${resolved.baseUrl}${path}${url.search}`;
490
+ // Filter headers: remove hop-by-hop, auth, and cookie headers to prevent token leakage
491
+ const STRIP_REQUEST = new Set([...HOP_BY_HOP, "authorization", "cookie"]);
492
+ const forwardHeaders = {};
493
+ for (const [key, value] of c.req.raw.headers.entries()) {
494
+ if (!STRIP_REQUEST.has(key.toLowerCase())) {
495
+ forwardHeaders[key] = value;
496
+ }
497
+ }
498
+ const botEntry = listRegistered()[name];
499
+ if (botEntry?.botToken) {
500
+ forwardHeaders["Authorization"] = `Bearer ${botEntry.botToken}`;
501
+ }
502
+ try {
503
+ const resp = await fetch(targetUrl, {
504
+ method: c.req.method,
505
+ headers: forwardHeaders,
506
+ body: c.req.method !== "GET" && c.req.method !== "HEAD" ? c.req.raw.body : undefined,
507
+ redirect: "manual",
508
+ signal: AbortSignal.timeout(30_000),
509
+ });
510
+ // Filter response headers: strip hop-by-hop and set-cookie to prevent bot planting cookies on fleet origin
511
+ const STRIP_RESPONSE = new Set([...HOP_BY_HOP, "set-cookie"]);
512
+ const responseHeaders = {};
513
+ for (const [key, value] of resp.headers.entries()) {
514
+ if (!STRIP_RESPONSE.has(key.toLowerCase())) {
515
+ responseHeaders[key] = value;
516
+ }
517
+ }
518
+ return new Response(resp.body, {
519
+ status: resp.status,
520
+ headers: responseHeaders,
521
+ });
522
+ }
523
+ catch (err) {
524
+ return c.json({ error: "Proxy error" }, 502);
525
+ }
526
+ });
527
+ // --- Office Layout API ---
528
+ app.get("/api/office/layout", (c) => {
529
+ const layoutPath = join(getMechaDir(), "office-layout.json");
530
+ if (!existsSync(layoutPath)) {
531
+ return c.json({ error: "No layout" }, 404);
532
+ }
533
+ try {
534
+ const content = fsReadFileSync(layoutPath, "utf-8");
535
+ const etag = createHash("md5").update(content).digest("hex");
536
+ c.header("ETag", `"${etag}"`);
537
+ return c.json(JSON.parse(content));
538
+ }
539
+ catch {
540
+ return c.json({ error: "Failed to read or parse layout file" }, 500);
541
+ }
542
+ });
543
+ app.post("/api/office/layout", async (c) => {
544
+ const body = await c.req.json().catch(() => null);
545
+ if (!body)
546
+ return c.json({ error: "Invalid JSON" }, 400);
547
+ if (body.version !== 1 ||
548
+ !Array.isArray(body.tiles) ||
549
+ typeof body.cols !== "number" ||
550
+ typeof body.rows !== "number" ||
551
+ body.cols <= 0 ||
552
+ body.rows <= 0 ||
553
+ !Number.isInteger(body.cols) ||
554
+ !Number.isInteger(body.rows)) {
555
+ return c.json({ error: "Invalid layout: must have version=1, tiles array, cols>0, rows>0" }, 400);
556
+ }
557
+ // Enforce reasonable dimension limits and structural consistency
558
+ if (body.cols > 64 || body.rows > 64) {
559
+ return c.json({ error: "Layout dimensions too large (max 64x64)" }, 400);
560
+ }
561
+ if (body.tiles.length !== body.cols * body.rows) {
562
+ return c.json({ error: `tiles.length (${body.tiles.length}) must equal cols*rows (${body.cols * body.rows})` }, 400);
563
+ }
564
+ if (body.tileColors && Array.isArray(body.tileColors) && body.tileColors.length !== body.tiles.length) {
565
+ return c.json({ error: "tileColors.length must match tiles.length" }, 400);
566
+ }
567
+ if (body.furniture && !Array.isArray(body.furniture)) {
568
+ return c.json({ error: "furniture must be an array" }, 400);
569
+ }
570
+ const layoutPath = join(getMechaDir(), "office-layout.json");
571
+ const ifMatch = c.req.header("If-Match");
572
+ if (ifMatch) {
573
+ if (!existsSync(layoutPath)) {
574
+ return c.json({ error: "Conflict: layout does not exist" }, 409);
575
+ }
576
+ const current = fsReadFileSync(layoutPath, "utf-8");
577
+ const currentEtag = `"${createHash("md5").update(current).digest("hex")}"`;
578
+ if (ifMatch !== currentEtag) {
579
+ return c.json({ error: "Conflict: ETag mismatch" }, 409);
580
+ }
581
+ }
582
+ await atomicWriteJsonAsync(layoutPath, body);
583
+ const written = fsReadFileSync(layoutPath, "utf-8");
584
+ const newEtag = `"${createHash("md5").update(written).digest("hex")}"`;
585
+ c.header("ETag", newEtag);
586
+ return c.json({ status: "saved" });
587
+ });
588
+ // --- Office Avatars API ---
589
+ const avatarsFile = () => join(getMechaDir(), "office-avatars.json");
590
+ function readAvatars() {
591
+ const p = avatarsFile();
592
+ if (!existsSync(p))
593
+ return {};
594
+ try {
595
+ const raw = JSON.parse(fsReadFileSync(p, "utf-8"));
596
+ const result = {};
597
+ for (const [key, val] of Object.entries(raw)) {
598
+ if (val && typeof val === "object") {
599
+ const v = val;
600
+ if (typeof v.palette === "number" && typeof v.hueShift === "number" && typeof v.displayName === "string") {
601
+ result[key] = { palette: v.palette, hueShift: v.hueShift, displayName: v.displayName };
602
+ }
603
+ }
604
+ }
605
+ return result;
606
+ }
607
+ catch {
608
+ return {};
609
+ }
610
+ }
611
+ app.get("/api/office/avatars", (c) => {
612
+ return c.json(readAvatars());
613
+ });
614
+ app.post("/api/office/avatars", async (c) => {
615
+ const body = await c.req.json().catch(() => null);
616
+ if (!body)
617
+ return c.json({ error: "Invalid JSON" }, 400);
618
+ const { name, palette, hueShift, displayName } = body;
619
+ // Validate name
620
+ if (typeof name !== "string" || !isValidName(name)) {
621
+ return c.json({ error: "Invalid or missing name" }, 400);
622
+ }
623
+ // Validate palette: integer 0-5
624
+ if (typeof palette !== "number" || !Number.isInteger(palette) || palette < 0 || palette > 5) {
625
+ return c.json({ error: "palette must be an integer 0-5" }, 400);
626
+ }
627
+ // Validate hueShift: integer 0-359
628
+ if (typeof hueShift !== "number" || !Number.isInteger(hueShift) || hueShift < 0 || hueShift > 359) {
629
+ return c.json({ error: "hueShift must be an integer 0-359" }, 400);
630
+ }
631
+ // Validate displayName: string 1-32 chars, strip control chars
632
+ if (typeof displayName !== "string") {
633
+ return c.json({ error: "displayName must be a string" }, 400);
634
+ }
635
+ // eslint-disable-next-line no-control-regex
636
+ const cleaned = displayName.replace(/[\x00-\x1f\x7f]/g, "").trim();
637
+ if (cleaned.length < 1 || cleaned.length > 32) {
638
+ return c.json({ error: "displayName must be 1-32 characters after stripping control chars" }, 400);
639
+ }
640
+ // Read-merge-write atomically
641
+ const avatars = readAvatars();
642
+ avatars[name] = { palette, hueShift, displayName: cleaned };
643
+ try {
644
+ await atomicWriteJsonAsync(avatarsFile(), avatars);
645
+ }
646
+ catch {
647
+ return c.json({ error: "Failed to write avatar config" }, 500);
648
+ }
649
+ return c.json({ status: "saved" });
650
+ });
651
+ // --- Fleet Office SSE Stream ---
652
+ app.get("/api/office/stream", async (c) => {
653
+ return streamSSE(c, async (stream) => {
654
+ let seq = 0;
655
+ const knownBots = new Map(); // containerId → name
656
+ let closed = false;
657
+ let lastAvatarMtime = 0; // track avatar file changes
658
+ async function sendSnapshot() {
659
+ const bots = await docker.list();
660
+ const running = bots.filter(b => b.status === "running");
661
+ const avatars = readAvatars();
662
+ const snapshot = running.map(b => {
663
+ // Use truncated container ID (first 12 chars) — avoids exposing full ID on public endpoint
664
+ const entry = { bot_id: b.containerId.slice(0, 12), name: b.name, status: "idle" };
665
+ const av = avatars[b.name];
666
+ if (av) {
667
+ entry.palette = av.palette;
668
+ entry.hueShift = av.hueShift;
669
+ entry.displayName = av.displayName;
670
+ }
671
+ return entry;
672
+ });
673
+ await stream.writeSSE({ event: "snapshot", data: JSON.stringify({ seq: seq++, bots: snapshot }) });
674
+ knownBots.clear();
675
+ for (const b of running)
676
+ knownBots.set(b.containerId, b.name);
677
+ // Track avatar file mtime
678
+ try {
679
+ lastAvatarMtime = statSync(avatarsFile()).mtimeMs;
680
+ }
681
+ catch {
682
+ lastAvatarMtime = 0;
683
+ }
684
+ }
685
+ await sendSnapshot();
686
+ const pollInterval = setInterval(async () => {
687
+ if (closed)
688
+ return;
689
+ try {
690
+ const bots = await docker.list();
691
+ const running = new Map(bots.filter(b => b.status === "running").map(b => [b.containerId, b.name]));
692
+ for (const [id, name] of running) {
693
+ if (!knownBots.has(id)) {
694
+ knownBots.set(id, name);
695
+ const avatars = readAvatars();
696
+ const av = avatars[name];
697
+ const joinData = { seq: seq++, type: "bot_join", bot_id: id, name };
698
+ if (av) {
699
+ joinData.palette = av.palette;
700
+ joinData.hueShift = av.hueShift;
701
+ joinData.displayName = av.displayName;
702
+ }
703
+ await stream.writeSSE({ event: "state", data: JSON.stringify(joinData) });
704
+ }
705
+ }
706
+ for (const [id] of knownBots) {
707
+ if (!running.has(id)) {
708
+ knownBots.delete(id);
709
+ await stream.writeSSE({ event: "state", data: JSON.stringify({ seq: seq++, type: "bot_leave", bot_id: id }) });
710
+ }
711
+ }
712
+ // Detect avatar file changes and resend snapshot
713
+ let currentMtime = 0;
714
+ try {
715
+ currentMtime = statSync(avatarsFile()).mtimeMs;
716
+ }
717
+ catch { /* file may not exist */ }
718
+ if (currentMtime > 0 && currentMtime !== lastAvatarMtime) {
719
+ lastAvatarMtime = currentMtime;
720
+ await sendSnapshot();
721
+ }
722
+ }
723
+ catch { /* ignore polling errors */ }
724
+ }, 5000);
725
+ const heartbeatInterval = setInterval(async () => {
726
+ if (closed)
727
+ return;
728
+ try {
729
+ await stream.writeSSE({ event: "heartbeat", data: JSON.stringify({ seq: seq++ }) });
730
+ }
731
+ catch { /* connection closed */ }
732
+ }, 15000);
733
+ // Keep stream alive until client disconnects
734
+ await new Promise((resolve) => {
735
+ stream.onAbort(() => {
736
+ closed = true;
737
+ clearInterval(pollInterval);
738
+ clearInterval(heartbeatInterval);
739
+ resolve();
740
+ });
741
+ });
742
+ });
743
+ });
744
+ // --- Serve unified dashboard (from agent/dashboard/dist) ---
745
+ const staticRoot = join(__dirname, "..", "agent", "dashboard", "dist");
746
+ if (existsSync(staticRoot)) {
747
+ app.use("/*", serveStatic({ root: staticRoot }));
748
+ // SPA fallback: serve index.html for non-file, non-API routes
749
+ const indexPath = join(staticRoot, "index.html");
750
+ const cachedIndexHtml = existsSync(indexPath) ? fsReadFileSync(indexPath, "utf-8") : null;
751
+ app.get("*", (c) => {
752
+ const path = c.req.path;
753
+ if (path.startsWith("/api/") || path.startsWith("/bot/")) {
754
+ return c.json({ error: "Not found" }, 404);
755
+ }
756
+ if (cachedIndexHtml) {
757
+ return c.html(cachedIndexHtml);
758
+ }
759
+ return c.json({ error: "Dashboard not built" }, 404);
760
+ });
761
+ }
762
+ else {
763
+ app.get("/", (c) => c.json({
764
+ message: "Mecha Fleet Dashboard API",
765
+ routes: ["/api/bots", "/api/auth", "/api/network", "/bot/:name/*"],
766
+ }));
767
+ }
768
+ const hostname = host ?? process.env.MECHA_DASHBOARD_HOST ?? "127.0.0.1";
769
+ const server = serve({ fetch: app.fetch, port, hostname }, () => {
770
+ console.log(`Mecha dashboard running at http://${hostname}:${port}`);
771
+ if (!process.env.MECHA_DASHBOARD_TOKEN) {
772
+ console.log(`Dashboard token (auto-generated): ${DASHBOARD_TOKEN.slice(0, 16)}...`);
773
+ }
774
+ });
775
+ return server;
776
+ }
777
+ //# sourceMappingURL=dashboard-server.js.map