@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,176 @@
1
+ /**
2
+ * Fleet management MCP tools for orchestrator bots.
3
+ * Only registered when config.permissions.fleet_control is true.
4
+ * All operations proxy to the daemon's /api/fleet/* endpoints.
5
+ */
6
+
7
+ import { tool } from "@anthropic-ai/claude-agent-sdk";
8
+ import { z } from "zod";
9
+
10
+ const FLEET_URL = process.env.MECHA_FLEET_URL;
11
+ const FLEET_SECRET = process.env.MECHA_FLEET_INTERNAL_SECRET;
12
+ const BOT_NAME = process.env.MECHA_BOT_NAME;
13
+
14
+ // Rate limiting (in-memory, best-effort)
15
+ let spawnCount = 0;
16
+ let opsCount = 0;
17
+ let lastReset = Date.now();
18
+ const MAX_SPAWNS_PER_HOUR = 5;
19
+ const MAX_OPS_PER_HOUR = 20;
20
+
21
+ function checkRateLimit(isSpawn: boolean): string | null {
22
+ const now = Date.now();
23
+ if (now - lastReset > 3600_000) { spawnCount = 0; opsCount = 0; lastReset = now; }
24
+ opsCount++;
25
+ if (isSpawn) spawnCount++;
26
+ if (spawnCount > MAX_SPAWNS_PER_HOUR) return `Spawn rate limit exceeded (max ${MAX_SPAWNS_PER_HOUR}/hour)`;
27
+ if (opsCount > MAX_OPS_PER_HOUR) return `Fleet operation rate limit exceeded (max ${MAX_OPS_PER_HOUR}/hour)`;
28
+ return null;
29
+ }
30
+
31
+ async function fleetApi(path: string, method = "GET", body?: unknown): Promise<unknown> {
32
+ if (!FLEET_URL || !FLEET_SECRET) throw new Error("Fleet API not configured (MECHA_FLEET_URL not set)");
33
+ const headers: Record<string, string> = { "Authorization": `Bearer ${FLEET_SECRET}` };
34
+ if (body) headers["Content-Type"] = "application/json";
35
+ const resp = await fetch(`${FLEET_URL}/api/fleet${path}`, {
36
+ method, headers,
37
+ body: body ? JSON.stringify(body) : undefined,
38
+ signal: AbortSignal.timeout(30_000),
39
+ });
40
+ const text = await resp.text();
41
+ if (!resp.ok) throw new Error(`Fleet API error ${resp.status}: ${text}`);
42
+ try { return JSON.parse(text); } catch { return text; }
43
+ }
44
+
45
+ function textResult(data: unknown): { content: Array<{ type: "text"; text: string }> } {
46
+ return { content: [{ type: "text" as const, text: typeof data === "string" ? data : JSON.stringify(data, null, 2) }] };
47
+ }
48
+
49
+ export function createFleetTools() {
50
+ return [
51
+ tool(
52
+ "mecha_fleet_ls",
53
+ "List all bots in the fleet with status, model, uptime, and cost.",
54
+ {},
55
+ async () => {
56
+ const limit = checkRateLimit(false);
57
+ if (limit) return textResult({ error: limit });
58
+ const data = await fleetApi("/bots");
59
+ return textResult(data);
60
+ },
61
+ ),
62
+
63
+ tool(
64
+ "mecha_fleet_spawn",
65
+ "Spawn a new bot in the fleet.",
66
+ {
67
+ name: z.string().min(1).max(32).describe("Bot name"),
68
+ system: z.string().min(1).describe("System prompt"),
69
+ model: z.string().optional().describe("Model (default: sonnet)"),
70
+ auth: z.string().optional().describe("Auth profile name"),
71
+ },
72
+ async (args) => {
73
+ const limit = checkRateLimit(true);
74
+ if (limit) return textResult({ error: limit });
75
+ const data = await fleetApi("/bots", "POST", args);
76
+ return textResult(data);
77
+ },
78
+ ),
79
+
80
+ tool(
81
+ "mecha_fleet_stop",
82
+ "Stop a running bot.",
83
+ { name: z.string().min(1).max(32).describe("Bot name to stop") },
84
+ async (args) => {
85
+ if (args.name === BOT_NAME) return textResult({ error: "Cannot stop self" });
86
+ const limit = checkRateLimit(false);
87
+ if (limit) return textResult({ error: limit });
88
+ const data = await fleetApi(`/bots/${encodeURIComponent(args.name)}/stop`, "POST");
89
+ return textResult(data);
90
+ },
91
+ ),
92
+
93
+ tool(
94
+ "mecha_fleet_start",
95
+ "Start a stopped bot.",
96
+ { name: z.string().min(1).max(32).describe("Bot name to start") },
97
+ async (args) => {
98
+ const limit = checkRateLimit(false);
99
+ if (limit) return textResult({ error: limit });
100
+ const data = await fleetApi(`/bots/${encodeURIComponent(args.name)}/start`, "POST");
101
+ return textResult(data);
102
+ },
103
+ ),
104
+
105
+ tool(
106
+ "mecha_fleet_restart",
107
+ "Restart a running bot.",
108
+ { name: z.string().min(1).max(32).describe("Bot name to restart") },
109
+ async (args) => {
110
+ if (args.name === BOT_NAME) return textResult({ error: "Cannot restart self" });
111
+ const limit = checkRateLimit(false);
112
+ if (limit) return textResult({ error: limit });
113
+ const data = await fleetApi(`/bots/${encodeURIComponent(args.name)}/restart`, "POST");
114
+ return textResult(data);
115
+ },
116
+ ),
117
+
118
+ tool(
119
+ "mecha_fleet_rm",
120
+ "Remove a bot from the fleet (stops and deletes).",
121
+ { name: z.string().min(1).max(32).describe("Bot name to remove") },
122
+ async (args) => {
123
+ if (args.name === BOT_NAME) return textResult({ error: "Cannot remove self" });
124
+ const limit = checkRateLimit(false);
125
+ if (limit) return textResult({ error: limit });
126
+ const data = await fleetApi(`/bots/${encodeURIComponent(args.name)}`, "DELETE");
127
+ return textResult(data);
128
+ },
129
+ ),
130
+
131
+ tool(
132
+ "mecha_fleet_costs",
133
+ "Get cost breakdown for the fleet or a specific bot.",
134
+ {
135
+ name: z.string().optional().describe("Bot name (omit for fleet total)"),
136
+ period: z.enum(["today", "week", "month"]).optional().describe("Time period"),
137
+ },
138
+ async (args) => {
139
+ const limit = checkRateLimit(false);
140
+ if (limit) return textResult({ error: limit });
141
+ const path = args.name ? `/costs/${encodeURIComponent(args.name)}` : "/costs";
142
+ const query = args.period ? `?period=${args.period}` : "";
143
+ const data = await fleetApi(`${path}${query}`);
144
+ return textResult(data);
145
+ },
146
+ ),
147
+
148
+ tool(
149
+ "mecha_fleet_config",
150
+ "View a bot's configuration (read-only).",
151
+ {
152
+ name: z.string().min(1).max(32).describe("Bot name"),
153
+ field: z.string().optional().describe("Specific field to retrieve"),
154
+ },
155
+ async (args) => {
156
+ const limit = checkRateLimit(false);
157
+ if (limit) return textResult({ error: limit });
158
+ const query = args.field ? `?field=${encodeURIComponent(args.field)}` : "";
159
+ const data = await fleetApi(`/bots/${encodeURIComponent(args.name)}/config${query}`);
160
+ return textResult(data);
161
+ },
162
+ ),
163
+
164
+ tool(
165
+ "mecha_fleet_status",
166
+ "Get fleet health summary (running bots, costs, version).",
167
+ {},
168
+ async () => {
169
+ const limit = checkRateLimit(false);
170
+ if (limit) return textResult({ error: limit });
171
+ const data = await fleetApi("/health");
172
+ return textResult(data);
173
+ },
174
+ ),
175
+ ];
176
+ }
@@ -0,0 +1,56 @@
1
+ import { z } from "zod";
2
+ import { isValidUrl } from "../../shared/validation.js";
3
+
4
+ const headscaleResponseSchema = z.object({
5
+ machines: z.array(z.object({
6
+ name: z.string(),
7
+ ipAddresses: z.array(z.string()).default([]),
8
+ online: z.boolean().default(false),
9
+ forcedTags: z.array(z.string()).optional(),
10
+ })),
11
+ });
12
+
13
+ export async function listBots(): Promise<Array<{
14
+ name: string;
15
+ ip: string;
16
+ status: string;
17
+ }>> {
18
+ const headscaleUrl = process.env.MECHA_HEADSCALE_URL;
19
+ const headscaleApiKey = process.env.MECHA_HEADSCALE_API_KEY;
20
+
21
+ if (!headscaleUrl || !headscaleApiKey) {
22
+ return [{ name: "(unknown)", ip: "(no headscale)", status: "error: MECHA_HEADSCALE_URL and MECHA_HEADSCALE_API_KEY required" }];
23
+ }
24
+
25
+ if (!isValidUrl(headscaleUrl)) {
26
+ return [{ name: "(error)", ip: "", status: "error: MECHA_HEADSCALE_URL is not a valid http(s) URL" }];
27
+ }
28
+
29
+ try {
30
+ const resp = await fetch(`${headscaleUrl}/api/v1/machine`, {
31
+ headers: { Authorization: `Bearer ${headscaleApiKey}` },
32
+ signal: AbortSignal.timeout(10_000),
33
+ });
34
+
35
+ if (!resp.ok) {
36
+ return [{ name: "(error)", ip: "", status: `Headscale API returned ${resp.status}` }];
37
+ }
38
+
39
+ const raw = await resp.json();
40
+ const parsed = headscaleResponseSchema.safeParse(raw);
41
+ if (!parsed.success) {
42
+ return [{ name: "(error)", ip: "", status: "Headscale API returned unexpected response shape" }];
43
+ }
44
+
45
+ return parsed.data.machines
46
+ .filter((m) => m.forcedTags?.includes("tag:mecha-bot") || m.name.startsWith("mecha-"))
47
+ .map((m) => ({
48
+ name: m.name.replace(/^mecha-/, ""),
49
+ ip: m.ipAddresses[0] ?? "",
50
+ status: m.online ? "online" : "offline",
51
+ }));
52
+ } catch (err) {
53
+ const msg = err instanceof Error ? err.message : String(err);
54
+ return [{ name: "(error)", ip: "", status: `Headscale unavailable: ${msg}` }];
55
+ }
56
+ }
@@ -0,0 +1,70 @@
1
+ import { createSdkMcpServer, tool } from "@anthropic-ai/claude-agent-sdk";
2
+ import { z } from "zod";
3
+ import type { SessionManager } from "../session.js";
4
+ import type { BotConfig } from "../types.js";
5
+ import { callBot } from "./mecha-call.js";
6
+ import { listBots } from "./mecha-list.js";
7
+ import { createFleetTools } from "./mecha-fleet.js";
8
+
9
+ export function createMechaToolServer(sessionManager: SessionManager, config?: BotConfig) {
10
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
11
+ const tools: any[] = [
12
+ tool(
13
+ "mecha_new_session",
14
+ "Start a new task/session. Call this when you want to begin a fresh conversation or when the current task is complete.",
15
+ { summary: z.string().optional().describe("Summary of the completed task") },
16
+ async (args) => {
17
+ const { newTask, previousTask } = sessionManager.newSession(args.summary);
18
+ return {
19
+ content: [{
20
+ type: "text" as const,
21
+ text: JSON.stringify({
22
+ new_task_id: newTask.id,
23
+ previous_task: previousTask
24
+ ? { id: previousTask.id, summary: previousTask.summary, status: "completed" }
25
+ : null,
26
+ }),
27
+ }],
28
+ };
29
+ },
30
+ ),
31
+
32
+ tool(
33
+ "mecha_call",
34
+ "Call another mecha bot on the network and get its response. Use this to delegate tasks or ask questions to other specialized bots.",
35
+ {
36
+ bot: z.string().min(1).max(32).regex(/^[a-z0-9]([a-z0-9-]*[a-z0-9])?$/).describe("Target bot name"),
37
+ message: z.string().min(1).max(50_000).describe("Message to send to the bot"),
38
+ },
39
+ async (args) => {
40
+ const response = await callBot(args.bot, args.message);
41
+ return {
42
+ content: [{ type: "text" as const, text: response }],
43
+ };
44
+ },
45
+ ),
46
+
47
+ tool(
48
+ "mecha_list",
49
+ "List available mecha bots on the network. Shows bot names, IPs, and online status.",
50
+ {},
51
+ async () => {
52
+ const bots = await listBots();
53
+ return {
54
+ content: [{ type: "text" as const, text: JSON.stringify(bots, null, 2) }],
55
+ };
56
+ },
57
+ ),
58
+ ];
59
+
60
+ // Register fleet tools only for orchestrator bots with fleet_control permission
61
+ if (config?.permissions?.fleet_control) {
62
+ tools.push(...createFleetTools());
63
+ }
64
+
65
+ return createSdkMcpServer({
66
+ name: "mecha-tools",
67
+ version: "1.0.0",
68
+ tools,
69
+ });
70
+ }
package/agent/types.ts ADDED
@@ -0,0 +1,35 @@
1
+ import { z } from "zod";
2
+
3
+ const VALID_CRON = /^(\S+\s+){4}\S+$/;
4
+
5
+ export const botConfigSchema = z.object({
6
+ name: z.string().min(1).max(32),
7
+ system: z.string().min(1),
8
+ model: z.string().default("sonnet"),
9
+ auth: z.string().optional(),
10
+ max_turns: z.number().int().min(1).max(100).default(25),
11
+ max_budget_usd: z.number().positive().optional(),
12
+ permission_mode: z.enum(["default", "acceptEdits", "bypassPermissions", "plan", "dontAsk"]).default("default"),
13
+ schedule: z.array(z.object({
14
+ cron: z.string().refine((s) => VALID_CRON.test(s), { message: "Invalid cron expression (expected 5 fields)" }),
15
+ prompt: z.string().min(1).max(10_000),
16
+ })).optional(),
17
+ webhooks: z.object({
18
+ accept: z.array(z.string()),
19
+ secret: z.string().optional(),
20
+ }).optional(),
21
+ permissions: z.object({
22
+ fleet_control: z.boolean().default(false),
23
+ }).optional(),
24
+ workspace: z.string().optional(),
25
+ workspace_writable: z.boolean().default(false),
26
+ expose: z.number().int().min(1).max(65535).optional(),
27
+ tailscale: z.object({
28
+ auth_key_profile: z.string().optional(),
29
+ auth_key: z.string().optional(),
30
+ login_server: z.string().optional(),
31
+ tags: z.array(z.string()).default(["tag:mecha-bot"]),
32
+ }).optional(),
33
+ });
34
+
35
+ export type BotConfig = z.infer<typeof botConfigSchema>;
@@ -0,0 +1,127 @@
1
+ import { Hono } from "hono";
2
+ import type { Context } from "hono";
3
+ import { createHmac, timingSafeEqual } from "node:crypto";
4
+ import { log } from "../shared/logger.js";
5
+ import type { BotConfig } from "./types.js";
6
+
7
+ export type WebhookHandler = (prompt: string) => Promise<boolean>;
8
+
9
+ const MAX_PAYLOAD_BYTES = 100_000;
10
+
11
+ function safeExtractPayload(body: Record<string, unknown>): Record<string, unknown> {
12
+ const { installation, sender, repository, ...rest } = body;
13
+ const safe: Record<string, unknown> = { ...rest };
14
+ if (repository && typeof repository === "object") {
15
+ const repo = repository as Record<string, unknown>;
16
+ safe.repository = { full_name: repo.full_name, html_url: repo.html_url };
17
+ }
18
+ if (sender && typeof sender === "object") {
19
+ const s = sender as Record<string, unknown>;
20
+ safe.sender = { login: s.login };
21
+ }
22
+ return safe;
23
+ }
24
+
25
+ export interface WebhookState {
26
+ accept: string[];
27
+ secret: string | undefined;
28
+ }
29
+
30
+ export function createWebhookRoutes(
31
+ config: BotConfig,
32
+ handler: WebhookHandler,
33
+ isBusy: () => boolean,
34
+ ): { app: Hono; state: WebhookState } {
35
+ const app = new Hono();
36
+ const state: WebhookState = {
37
+ accept: [...(config.webhooks?.accept ?? [])],
38
+ secret: config.webhooks?.secret,
39
+ };
40
+
41
+ if (!state.secret) {
42
+ log.warn("Webhook secret not configured — payloads will be accepted without signature verification. Set webhooks.secret for production use.");
43
+ }
44
+
45
+ app.post("/webhook", async (c) => {
46
+ // Payload size limit (check header as hint, enforce on body)
47
+ const contentLength = parseInt(c.req.header("content-length") ?? "0", 10);
48
+ if (contentLength > MAX_PAYLOAD_BYTES) {
49
+ return c.json({ error: "Payload too large" }, 413);
50
+ }
51
+
52
+ // Read secret from state each request (supports runtime updates)
53
+ const currentSecret = state.secret;
54
+
55
+ // Webhook signature verification (GitHub HMAC)
56
+ if (currentSecret) {
57
+ const signature = c.req.header("x-hub-signature-256");
58
+ if (!signature) {
59
+ return c.json({ error: "Missing signature" }, 401);
60
+ }
61
+ const body = await c.req.text();
62
+ if (Buffer.byteLength(body, "utf8") > MAX_PAYLOAD_BYTES) {
63
+ return c.json({ error: "Payload too large" }, 413);
64
+ }
65
+ const expected = "sha256=" + createHmac("sha256", currentSecret).update(body).digest("hex");
66
+ const sigBuf = Buffer.from(signature);
67
+ const expBuf = Buffer.from(expected);
68
+ if (sigBuf.length !== expBuf.length || !timingSafeEqual(sigBuf, expBuf)) {
69
+ return c.json({ error: "Invalid signature" }, 401);
70
+ }
71
+ let parsed: Record<string, unknown>;
72
+ try {
73
+ parsed = JSON.parse(body);
74
+ } catch {
75
+ return c.json({ error: "Invalid JSON" }, 400);
76
+ }
77
+ return handlePayload(c, parsed);
78
+ }
79
+
80
+ // Enforce size limit even without HMAC secret
81
+ const rawBody = await c.req.text();
82
+ if (Buffer.byteLength(rawBody, "utf8") > MAX_PAYLOAD_BYTES) {
83
+ return c.json({ error: "Payload too large" }, 413);
84
+ }
85
+ let body: Record<string, unknown>;
86
+ try {
87
+ body = JSON.parse(rawBody);
88
+ } catch {
89
+ return c.json({ error: "Invalid JSON" }, 400);
90
+ }
91
+ return handlePayload(c, body);
92
+ });
93
+
94
+ async function handlePayload(c: Context, body: Record<string, unknown>) {
95
+ // Extract event type
96
+ let eventType: string;
97
+ const githubEvent = c.req.header("x-github-event");
98
+ if (githubEvent) {
99
+ const action = (body.action as string) ?? "";
100
+ eventType = action ? `${githubEvent}.${action}` : githubEvent;
101
+ } else {
102
+ eventType = (body.type as string) ?? "unknown";
103
+ }
104
+
105
+ // Check allowlist
106
+ if (!state.accept.includes(eventType)) {
107
+ return c.body(null, 204); // silently drop
108
+ }
109
+
110
+ // Busy check
111
+ if (isBusy()) {
112
+ c.header("Retry-After", "60");
113
+ return c.json({ error: "Bot is busy" }, 429);
114
+ }
115
+
116
+ // Forward to handler with redacted payload
117
+ const safePayload = safeExtractPayload(body);
118
+ const prompt = `Webhook event: ${eventType}\n\nPayload:\n${JSON.stringify(safePayload, null, 2)}`;
119
+ const accepted = await handler(prompt);
120
+ if (accepted) {
121
+ return c.json({ status: "accepted", event: eventType });
122
+ }
123
+ return c.json({ error: "Handler rejected" }, 500);
124
+ }
125
+
126
+ return { app, state };
127
+ }
@@ -0,0 +1,174 @@
1
+ import { WebSocketServer, WebSocket } from "ws";
2
+ import type { IncomingMessage } from "node:http";
3
+ import type { Server as HttpServer } from "node:http";
4
+ import { createHmac, timingSafeEqual } from "node:crypto";
5
+ import type { PtyManager } from "./pty-manager.js";
6
+ import { log } from "../shared/logger.js";
7
+
8
+ const BACKPRESSURE_LIMIT = 1_048_576; // 1 MB
9
+ const PING_INTERVAL_MS = 30_000;
10
+ const SESSION_ID_RE = /^[a-zA-Z0-9_-]+$/;
11
+
12
+ function makeSessionToken(botToken: string): string {
13
+ return createHmac("sha256", botToken).update("mecha-dashboard-session").digest("hex");
14
+ }
15
+
16
+ /** Verify session cookie from WebSocket upgrade request. */
17
+ function verifyAuth(req: IncomingMessage, botToken: string): boolean {
18
+ const cookie = req.headers.cookie;
19
+ if (!cookie) return false;
20
+ const match = cookie.match(/mecha_session=([^;]+)/);
21
+ if (!match) return false;
22
+ const expected = makeSessionToken(botToken);
23
+ const received = Buffer.from(match[1]);
24
+ const expectedBuf = Buffer.from(expected);
25
+ return received.length === expectedBuf.length && timingSafeEqual(received, expectedBuf);
26
+ }
27
+
28
+ /** Attach WebSocket terminal server to an HTTP server. */
29
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
30
+ export function attachTerminalWs(
31
+ httpServer: any,
32
+ ptyManager: PtyManager,
33
+ botToken: string,
34
+ ): void {
35
+ const wss = new WebSocketServer({ noServer: true });
36
+
37
+ httpServer.on("upgrade", (req: IncomingMessage, socket: import("node:stream").Duplex, head: Buffer) => {
38
+ const url = new URL(req.url ?? "/", "http://localhost");
39
+
40
+ // Only handle /ws/terminal path
41
+ if (!url.pathname.startsWith("/ws/terminal")) {
42
+ socket.destroy();
43
+ return;
44
+ }
45
+
46
+ // Auth check
47
+ if (!verifyAuth(req, botToken)) {
48
+ socket.write("HTTP/1.1 401 Unauthorized\r\n\r\n");
49
+ socket.destroy();
50
+ return;
51
+ }
52
+
53
+ wss.handleUpgrade(req, socket, head, (ws) => {
54
+ wss.emit("connection", ws, req);
55
+ });
56
+ });
57
+
58
+ wss.on("connection", (ws: WebSocket, req: IncomingMessage) => {
59
+ const url = new URL(req.url ?? "/", "http://localhost");
60
+ let sessionId = url.searchParams.get("session") ?? undefined;
61
+ const initCols = Number(url.searchParams.get("cols")) || 80;
62
+ const initRows = Number(url.searchParams.get("rows")) || 24;
63
+
64
+ // Validate session ID format
65
+ if (sessionId && !SESSION_ID_RE.test(sessionId)) {
66
+ ws.send(JSON.stringify({ __mecha: true, type: "error", message: `Invalid session ID: ${sessionId}` }));
67
+ ws.close(4400, "Invalid session ID");
68
+ return;
69
+ }
70
+
71
+ // Look up existing PTY session
72
+ let session = sessionId ? ptyManager.getSession(sessionId) : null;
73
+
74
+ try {
75
+ if (session) {
76
+ const attached = ptyManager.attach(session.id, ws);
77
+ if (!attached) {
78
+ session = ptyManager.spawn(sessionId, initCols, initRows);
79
+ ptyManager.attach(session.id, ws);
80
+ }
81
+ } else {
82
+ session = ptyManager.spawn(sessionId, initCols, initRows);
83
+ ptyManager.attach(session.id, ws);
84
+ }
85
+ } catch (err) {
86
+ ws.send(JSON.stringify({
87
+ __mecha: true,
88
+ type: "error",
89
+ message: err instanceof Error ? err.message : String(err),
90
+ }));
91
+ ws.close(4500, "Spawn failed");
92
+ return;
93
+ }
94
+
95
+ const sessionKey = session.id;
96
+ log.info(`Terminal WS connected: session=${sessionKey}`);
97
+
98
+ function safeSend(data: string): void {
99
+ if (ws.readyState !== WebSocket.OPEN) return;
100
+ try { ws.send(data); } catch { /* socket closing */ }
101
+ }
102
+
103
+ // Send session ID so client can use it for resume
104
+ safeSend(JSON.stringify({ __mecha: true, type: "session", id: session.claudeSessionId }));
105
+
106
+ // Replay scrollback
107
+ if (session.scrollback.length > 0) {
108
+ for (const chunk of session.scrollback) {
109
+ safeSend(chunk);
110
+ }
111
+ safeSend("\r\n\x1b[2m--- reconnected ---\x1b[0m\r\n");
112
+ }
113
+
114
+ // PTY output → WS (text frames to avoid UTF-8 fragmentation issues)
115
+ const dataDisposable = session.pty.onData((data) => {
116
+ if (ws.readyState === WebSocket.OPEN) {
117
+ if (ws.bufferedAmount > BACKPRESSURE_LIMIT) return;
118
+ safeSend(data);
119
+ }
120
+ });
121
+
122
+ // PTY exit → WS
123
+ const exitDisposable = session.pty.onExit(({ exitCode }) => {
124
+ if (ws.readyState !== WebSocket.OPEN) return;
125
+ safeSend(JSON.stringify({ __mecha: true, type: "exit", code: exitCode }));
126
+ ws.close(1000, "PTY exited");
127
+ });
128
+
129
+ // WS → PTY
130
+ ws.on("message", (data: Buffer, isBinary: boolean) => {
131
+ const s = ptyManager.getSession(sessionKey);
132
+ if (!s) return;
133
+
134
+ const str = data.toString();
135
+
136
+ if (!isBinary) {
137
+ try {
138
+ const msg = JSON.parse(str) as { type: string; cols?: number; rows?: number };
139
+ if (msg.type === "resize" && typeof msg.cols === "number" && typeof msg.rows === "number") {
140
+ ptyManager.resize(sessionKey, msg.cols, msg.rows);
141
+ return;
142
+ }
143
+ } catch {
144
+ // Not JSON — treat as PTY input
145
+ }
146
+ }
147
+ s.pty.write(str);
148
+ });
149
+
150
+ // Heartbeat
151
+ let isAlive = true;
152
+ ws.on("pong", () => { isAlive = true; });
153
+ const pingTimer = setInterval(() => {
154
+ if (!isAlive) { ws.terminate(); return; }
155
+ isAlive = false;
156
+ if (ws.readyState === WebSocket.OPEN) ws.ping();
157
+ }, PING_INTERVAL_MS);
158
+
159
+ ws.on("error", () => {
160
+ clearInterval(pingTimer);
161
+ ptyManager.detach(sessionKey, ws);
162
+ dataDisposable.dispose();
163
+ exitDisposable.dispose();
164
+ });
165
+
166
+ ws.on("close", () => {
167
+ log.info(`Terminal WS disconnected: session=${sessionKey}`);
168
+ clearInterval(pingTimer);
169
+ ptyManager.detach(sessionKey, ws);
170
+ dataDisposable.dispose();
171
+ exitDisposable.dispose();
172
+ });
173
+ });
174
+ }
@@ -0,0 +1,14 @@
1
+ import { EventEmitter } from "node:events";
2
+ export type ActivityState = "idle" | "thinking" | "calling" | "scheduled" | "webhook" | "error";
3
+ export declare class ActivityTracker extends EventEmitter {
4
+ private state;
5
+ private talkingTo;
6
+ private lastActive;
7
+ getState(): ActivityState;
8
+ getTalkingTo(): string | null;
9
+ getLastActive(): string | null;
10
+ transition(newState: ActivityState, meta?: {
11
+ talkingTo?: string;
12
+ }): void;
13
+ }
14
+ //# sourceMappingURL=activity.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"activity.d.ts","sourceRoot":"","sources":["../../agent/activity.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAE3C,MAAM,MAAM,aAAa,GAAG,MAAM,GAAG,UAAU,GAAG,SAAS,GAAG,WAAW,GAAG,SAAS,GAAG,OAAO,CAAC;AAEhG,qBAAa,eAAgB,SAAQ,YAAY;IAC/C,OAAO,CAAC,KAAK,CAAyB;IACtC,OAAO,CAAC,SAAS,CAAuB;IACxC,OAAO,CAAC,UAAU,CAAuB;IAEzC,QAAQ,IAAI,aAAa;IAIzB,YAAY,IAAI,MAAM,GAAG,IAAI;IAI7B,aAAa,IAAI,MAAM,GAAG,IAAI;IAI9B,UAAU,CAAC,QAAQ,EAAE,aAAa,EAAE,IAAI,CAAC,EAAE;QAAE,SAAS,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI;CASzE"}
@@ -0,0 +1,25 @@
1
+ import { EventEmitter } from "node:events";
2
+ export class ActivityTracker extends EventEmitter {
3
+ state = "idle";
4
+ talkingTo = null;
5
+ lastActive = null;
6
+ getState() {
7
+ return this.state;
8
+ }
9
+ getTalkingTo() {
10
+ return this.talkingTo;
11
+ }
12
+ getLastActive() {
13
+ return this.lastActive;
14
+ }
15
+ transition(newState, meta) {
16
+ const prev = this.state;
17
+ this.state = newState;
18
+ this.talkingTo = meta?.talkingTo ?? null;
19
+ if (newState !== "idle") {
20
+ this.lastActive = new Date().toISOString();
21
+ }
22
+ this.emit("change", { prev, state: newState, talkingTo: this.talkingTo });
23
+ }
24
+ }
25
+ //# sourceMappingURL=activity.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"activity.js","sourceRoot":"","sources":["../../agent/activity.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAI3C,MAAM,OAAO,eAAgB,SAAQ,YAAY;IACvC,KAAK,GAAkB,MAAM,CAAC;IAC9B,SAAS,GAAkB,IAAI,CAAC;IAChC,UAAU,GAAkB,IAAI,CAAC;IAEzC,QAAQ;QACN,OAAO,IAAI,CAAC,KAAK,CAAC;IACpB,CAAC;IAED,YAAY;QACV,OAAO,IAAI,CAAC,SAAS,CAAC;IACxB,CAAC;IAED,aAAa;QACX,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAED,UAAU,CAAC,QAAuB,EAAE,IAA6B;QAC/D,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC;QACxB,IAAI,CAAC,KAAK,GAAG,QAAQ,CAAC;QACtB,IAAI,CAAC,SAAS,GAAG,IAAI,EAAE,SAAS,IAAI,IAAI,CAAC;QACzC,IAAI,QAAQ,KAAK,MAAM,EAAE,CAAC;YACxB,IAAI,CAAC,UAAU,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAC7C,CAAC;QACD,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE,QAAQ,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;IAC5E,CAAC;CACF"}