groove-dev 0.27.93 → 0.27.96

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 (511) hide show
  1. package/CLAUDE.md +7 -0
  2. package/README.md +25 -52
  3. package/moe-training/package.json +2 -1
  4. package/node_modules/@groove-dev/cli/package.json +1 -1
  5. package/node_modules/@groove-dev/daemon/package.json +1 -1
  6. package/node_modules/@groove-dev/daemon/src/api.js +39 -23
  7. package/node_modules/@groove-dev/daemon/src/introducer.js +21 -6
  8. package/node_modules/@groove-dev/daemon/src/process.js +20 -2
  9. package/node_modules/@groove-dev/daemon/src/providers/gemini.js +27 -28
  10. package/node_modules/@groove-dev/gui/dist/assets/{index-VB4_k5Pz.js → index-BgTyFy4f.js} +331 -331
  11. package/node_modules/@groove-dev/gui/dist/assets/{index-Bo6AeNmM.css → index-QADLyUj5.css} +1 -1
  12. package/node_modules/@groove-dev/gui/dist/index.html +2 -2
  13. package/node_modules/@groove-dev/gui/package.json +1 -1
  14. package/node_modules/@groove-dev/gui/src/components/agents/agent-file-tree.jsx +68 -16
  15. package/node_modules/@groove-dev/gui/src/components/agents/agent-panel.jsx +2 -0
  16. package/node_modules/@groove-dev/gui/src/components/agents/code-review.jsx +9 -1
  17. package/node_modules/@groove-dev/gui/src/components/agents/workspace-mode.jsx +2 -2
  18. package/node_modules/@groove-dev/gui/src/components/editor/code-editor.jsx +1 -1
  19. package/node_modules/@groove-dev/gui/src/stores/groove.js +40 -26
  20. package/node_modules/@groove-dev/gui/src/views/agents.jsx +14 -65
  21. package/node_modules/base64-js/LICENSE +21 -0
  22. package/node_modules/base64-js/README.md +34 -0
  23. package/node_modules/base64-js/base64js.min.js +1 -0
  24. package/node_modules/base64-js/index.d.ts +3 -0
  25. package/node_modules/base64-js/index.js +150 -0
  26. package/node_modules/base64-js/package.json +47 -0
  27. package/node_modules/better-sqlite3/LICENSE +21 -0
  28. package/node_modules/better-sqlite3/README.md +99 -0
  29. package/node_modules/better-sqlite3/binding.gyp +38 -0
  30. package/node_modules/better-sqlite3/deps/common.gypi +68 -0
  31. package/node_modules/better-sqlite3/deps/copy.js +31 -0
  32. package/node_modules/better-sqlite3/deps/defines.gypi +41 -0
  33. package/node_modules/better-sqlite3/deps/download.sh +122 -0
  34. package/node_modules/better-sqlite3/deps/patches/1208.patch +15 -0
  35. package/node_modules/better-sqlite3/deps/sqlite3/sqlite3.c +268845 -0
  36. package/node_modules/better-sqlite3/deps/sqlite3/sqlite3.h +14335 -0
  37. package/node_modules/better-sqlite3/deps/sqlite3/sqlite3ext.h +739 -0
  38. package/node_modules/better-sqlite3/deps/sqlite3.gyp +80 -0
  39. package/node_modules/better-sqlite3/deps/test_extension.c +21 -0
  40. package/node_modules/better-sqlite3/lib/database.js +90 -0
  41. package/node_modules/better-sqlite3/lib/index.js +3 -0
  42. package/node_modules/better-sqlite3/lib/methods/aggregate.js +43 -0
  43. package/node_modules/better-sqlite3/lib/methods/backup.js +67 -0
  44. package/node_modules/better-sqlite3/lib/methods/function.js +31 -0
  45. package/node_modules/better-sqlite3/lib/methods/inspect.js +7 -0
  46. package/node_modules/better-sqlite3/lib/methods/pragma.js +12 -0
  47. package/node_modules/better-sqlite3/lib/methods/serialize.js +16 -0
  48. package/node_modules/better-sqlite3/lib/methods/table.js +189 -0
  49. package/node_modules/better-sqlite3/lib/methods/transaction.js +78 -0
  50. package/node_modules/better-sqlite3/lib/methods/wrappers.js +54 -0
  51. package/node_modules/better-sqlite3/lib/sqlite-error.js +20 -0
  52. package/node_modules/better-sqlite3/lib/util.js +12 -0
  53. package/node_modules/better-sqlite3/package.json +59 -0
  54. package/node_modules/better-sqlite3/src/addon.cpp +47 -0
  55. package/node_modules/better-sqlite3/src/better_sqlite3.cpp +74 -0
  56. package/node_modules/better-sqlite3/src/objects/backup.cpp +120 -0
  57. package/node_modules/better-sqlite3/src/objects/backup.hpp +36 -0
  58. package/node_modules/better-sqlite3/src/objects/database.cpp +417 -0
  59. package/node_modules/better-sqlite3/src/objects/database.hpp +103 -0
  60. package/node_modules/better-sqlite3/src/objects/statement-iterator.cpp +113 -0
  61. package/node_modules/better-sqlite3/src/objects/statement-iterator.hpp +50 -0
  62. package/node_modules/better-sqlite3/src/objects/statement.cpp +383 -0
  63. package/node_modules/better-sqlite3/src/objects/statement.hpp +58 -0
  64. package/node_modules/better-sqlite3/src/util/bind-map.cpp +73 -0
  65. package/node_modules/better-sqlite3/src/util/binder.cpp +193 -0
  66. package/node_modules/better-sqlite3/src/util/constants.cpp +172 -0
  67. package/node_modules/better-sqlite3/src/util/custom-aggregate.cpp +121 -0
  68. package/node_modules/better-sqlite3/src/util/custom-function.cpp +59 -0
  69. package/node_modules/better-sqlite3/src/util/custom-table.cpp +409 -0
  70. package/node_modules/better-sqlite3/src/util/data-converter.cpp +17 -0
  71. package/node_modules/better-sqlite3/src/util/data.cpp +194 -0
  72. package/node_modules/better-sqlite3/src/util/helpers.cpp +109 -0
  73. package/node_modules/better-sqlite3/src/util/macros.cpp +83 -0
  74. package/node_modules/better-sqlite3/src/util/query-macros.cpp +71 -0
  75. package/node_modules/better-sqlite3/src/util/row-builder.cpp +49 -0
  76. package/node_modules/bindings/LICENSE.md +22 -0
  77. package/node_modules/bindings/README.md +98 -0
  78. package/node_modules/bindings/bindings.js +221 -0
  79. package/node_modules/bindings/package.json +28 -0
  80. package/node_modules/bl/.travis.yml +17 -0
  81. package/node_modules/bl/BufferList.js +396 -0
  82. package/node_modules/bl/LICENSE.md +13 -0
  83. package/node_modules/bl/README.md +247 -0
  84. package/node_modules/bl/bl.js +84 -0
  85. package/node_modules/bl/package.json +37 -0
  86. package/node_modules/bl/test/convert.js +21 -0
  87. package/node_modules/bl/test/indexOf.js +492 -0
  88. package/node_modules/bl/test/isBufferList.js +32 -0
  89. package/node_modules/bl/test/test.js +869 -0
  90. package/node_modules/buffer/AUTHORS.md +70 -0
  91. package/node_modules/buffer/LICENSE +21 -0
  92. package/node_modules/buffer/README.md +410 -0
  93. package/node_modules/buffer/index.d.ts +186 -0
  94. package/node_modules/buffer/index.js +1817 -0
  95. package/node_modules/buffer/package.json +96 -0
  96. package/node_modules/decompress-response/index.d.ts +22 -0
  97. package/node_modules/decompress-response/index.js +58 -0
  98. package/node_modules/decompress-response/license +9 -0
  99. package/node_modules/decompress-response/node_modules/mimic-response/index.d.ts +17 -0
  100. package/node_modules/decompress-response/node_modules/mimic-response/index.js +77 -0
  101. package/node_modules/decompress-response/node_modules/mimic-response/license +9 -0
  102. package/node_modules/decompress-response/node_modules/mimic-response/package.json +42 -0
  103. package/node_modules/decompress-response/node_modules/mimic-response/readme.md +78 -0
  104. package/node_modules/decompress-response/package.json +56 -0
  105. package/node_modules/decompress-response/readme.md +48 -0
  106. package/node_modules/deep-extend/LICENSE +20 -0
  107. package/node_modules/deep-extend/README.md +91 -0
  108. package/node_modules/deep-extend/index.js +1 -0
  109. package/node_modules/deep-extend/lib/deep-extend.js +150 -0
  110. package/node_modules/deep-extend/package.json +62 -0
  111. package/node_modules/end-of-stream/LICENSE +21 -0
  112. package/node_modules/end-of-stream/README.md +54 -0
  113. package/node_modules/end-of-stream/index.js +96 -0
  114. package/node_modules/end-of-stream/package.json +37 -0
  115. package/node_modules/expand-template/.travis.yml +6 -0
  116. package/node_modules/expand-template/LICENSE +21 -0
  117. package/node_modules/expand-template/README.md +43 -0
  118. package/node_modules/expand-template/index.js +26 -0
  119. package/node_modules/expand-template/package.json +29 -0
  120. package/node_modules/expand-template/test.js +67 -0
  121. package/node_modules/file-uri-to-path/.npmignore +1 -0
  122. package/node_modules/file-uri-to-path/.travis.yml +30 -0
  123. package/node_modules/file-uri-to-path/History.md +21 -0
  124. package/node_modules/file-uri-to-path/LICENSE +20 -0
  125. package/node_modules/file-uri-to-path/README.md +74 -0
  126. package/node_modules/file-uri-to-path/index.d.ts +2 -0
  127. package/node_modules/file-uri-to-path/index.js +66 -0
  128. package/node_modules/file-uri-to-path/package.json +32 -0
  129. package/node_modules/file-uri-to-path/test/test.js +24 -0
  130. package/node_modules/file-uri-to-path/test/tests.json +13 -0
  131. package/node_modules/fs-constants/LICENSE +21 -0
  132. package/node_modules/fs-constants/README.md +26 -0
  133. package/node_modules/fs-constants/browser.js +1 -0
  134. package/node_modules/fs-constants/index.js +1 -0
  135. package/node_modules/fs-constants/package.json +19 -0
  136. package/node_modules/github-from-package/.travis.yml +4 -0
  137. package/node_modules/github-from-package/LICENSE +18 -0
  138. package/node_modules/github-from-package/example/package.json +8 -0
  139. package/node_modules/github-from-package/example/url.js +3 -0
  140. package/node_modules/github-from-package/index.js +17 -0
  141. package/node_modules/github-from-package/package.json +30 -0
  142. package/node_modules/github-from-package/readme.markdown +53 -0
  143. package/node_modules/github-from-package/test/a.json +8 -0
  144. package/node_modules/github-from-package/test/b.json +5 -0
  145. package/node_modules/github-from-package/test/c.json +5 -0
  146. package/node_modules/github-from-package/test/d.json +7 -0
  147. package/node_modules/github-from-package/test/e.json +5 -0
  148. package/node_modules/github-from-package/test/url.js +19 -0
  149. package/node_modules/ieee754/LICENSE +11 -0
  150. package/node_modules/ieee754/README.md +51 -0
  151. package/node_modules/ieee754/index.d.ts +10 -0
  152. package/node_modules/ieee754/index.js +85 -0
  153. package/node_modules/ieee754/package.json +52 -0
  154. package/node_modules/ini/LICENSE +15 -0
  155. package/node_modules/ini/README.md +102 -0
  156. package/node_modules/ini/ini.js +206 -0
  157. package/node_modules/ini/package.json +33 -0
  158. package/node_modules/minimist/.eslintrc +29 -0
  159. package/node_modules/minimist/.github/FUNDING.yml +12 -0
  160. package/node_modules/minimist/.nycrc +14 -0
  161. package/node_modules/minimist/CHANGELOG.md +298 -0
  162. package/node_modules/minimist/LICENSE +18 -0
  163. package/node_modules/minimist/README.md +121 -0
  164. package/node_modules/minimist/example/parse.js +4 -0
  165. package/node_modules/minimist/index.js +263 -0
  166. package/node_modules/minimist/package.json +75 -0
  167. package/node_modules/minimist/test/all_bool.js +34 -0
  168. package/node_modules/minimist/test/bool.js +177 -0
  169. package/node_modules/minimist/test/dash.js +43 -0
  170. package/node_modules/minimist/test/default_bool.js +37 -0
  171. package/node_modules/minimist/test/dotted.js +24 -0
  172. package/node_modules/minimist/test/kv_short.js +32 -0
  173. package/node_modules/minimist/test/long.js +33 -0
  174. package/node_modules/minimist/test/num.js +38 -0
  175. package/node_modules/minimist/test/parse.js +209 -0
  176. package/node_modules/minimist/test/parse_modified.js +11 -0
  177. package/node_modules/minimist/test/proto.js +64 -0
  178. package/node_modules/minimist/test/short.js +69 -0
  179. package/node_modules/minimist/test/stop_early.js +17 -0
  180. package/node_modules/minimist/test/unknown.js +104 -0
  181. package/node_modules/minimist/test/whitespace.js +10 -0
  182. package/node_modules/mkdirp-classic/LICENSE +21 -0
  183. package/node_modules/mkdirp-classic/README.md +18 -0
  184. package/node_modules/mkdirp-classic/index.js +98 -0
  185. package/node_modules/mkdirp-classic/package.json +18 -0
  186. package/node_modules/moe-training/DEPLOY_CENTRAL_COMMAND.md +413 -0
  187. package/node_modules/moe-training/client/consent.js +96 -0
  188. package/node_modules/moe-training/client/envelope-builder.js +56 -0
  189. package/node_modules/moe-training/client/index.js +10 -0
  190. package/node_modules/moe-training/client/parsers/claude-code.js +108 -0
  191. package/node_modules/moe-training/client/parsers/codex.js +80 -0
  192. package/node_modules/moe-training/client/parsers/gemini.js +80 -0
  193. package/node_modules/moe-training/client/parsers/grok.js +16 -0
  194. package/node_modules/moe-training/client/parsers/index.js +20 -0
  195. package/node_modules/moe-training/client/scrubber.js +128 -0
  196. package/node_modules/moe-training/client/session-attestation.js +115 -0
  197. package/node_modules/moe-training/client/step-classifier.js +68 -0
  198. package/node_modules/moe-training/client/trajectory-capture.js +307 -0
  199. package/node_modules/moe-training/client/transmission-queue.js +104 -0
  200. package/node_modules/moe-training/package.json +21 -0
  201. package/node_modules/moe-training/server/enrichment.js +24 -0
  202. package/node_modules/moe-training/server/index.js +119 -0
  203. package/node_modules/moe-training/server/ledger.js +110 -0
  204. package/node_modules/moe-training/server/routes/ingest.js +96 -0
  205. package/node_modules/moe-training/server/routes/sessions.js +43 -0
  206. package/node_modules/moe-training/server/routes/stats.js +31 -0
  207. package/node_modules/moe-training/server/scoring.js +63 -0
  208. package/node_modules/moe-training/server/session-registry.js +156 -0
  209. package/node_modules/moe-training/server/stats.js +129 -0
  210. package/node_modules/moe-training/server/stitcher.js +69 -0
  211. package/node_modules/moe-training/server/storage.js +147 -0
  212. package/node_modules/moe-training/server/verifier.js +102 -0
  213. package/node_modules/moe-training/shared/constants.js +30 -0
  214. package/node_modules/moe-training/shared/crypto.js +45 -0
  215. package/node_modules/moe-training/shared/envelope-schema.js +220 -0
  216. package/node_modules/moe-training/test/client/consent.test.js +121 -0
  217. package/node_modules/moe-training/test/client/envelope-builder.test.js +107 -0
  218. package/node_modules/moe-training/test/client/parsers/claude-code.test.js +119 -0
  219. package/node_modules/moe-training/test/client/parsers/codex.test.js +83 -0
  220. package/node_modules/moe-training/test/client/parsers/gemini.test.js +99 -0
  221. package/node_modules/moe-training/test/client/scrubber.test.js +143 -0
  222. package/node_modules/moe-training/test/client/session-attestation-security.test.js +95 -0
  223. package/node_modules/moe-training/test/client/step-classifier.test.js +135 -0
  224. package/node_modules/moe-training/test/client/transmission-queue.test.js +33 -0
  225. package/node_modules/moe-training/test/integration/handshake.test.js +260 -0
  226. package/node_modules/moe-training/test/server/ingest-security.test.js +166 -0
  227. package/node_modules/moe-training/test/server/ledger.test.js +131 -0
  228. package/node_modules/moe-training/test/server/scoring.test.js +242 -0
  229. package/node_modules/moe-training/test/server/session-registry.test.js +125 -0
  230. package/node_modules/moe-training/test/server/stitcher.test.js +157 -0
  231. package/node_modules/moe-training/test/server/verifier.test.js +232 -0
  232. package/node_modules/moe-training/test/shared/crypto.test.js +87 -0
  233. package/node_modules/moe-training/test/shared/envelope-schema.test.js +351 -0
  234. package/node_modules/napi-build-utils/.github/workflows/run-npm-tests.yml +31 -0
  235. package/node_modules/napi-build-utils/LICENSE +21 -0
  236. package/node_modules/napi-build-utils/README.md +52 -0
  237. package/node_modules/napi-build-utils/index.js +214 -0
  238. package/node_modules/napi-build-utils/index.md +0 -0
  239. package/node_modules/napi-build-utils/package.json +42 -0
  240. package/node_modules/prebuild-install/CHANGELOG.md +131 -0
  241. package/node_modules/prebuild-install/CONTRIBUTING.md +6 -0
  242. package/node_modules/prebuild-install/LICENSE +21 -0
  243. package/node_modules/prebuild-install/README.md +163 -0
  244. package/node_modules/prebuild-install/asset.js +44 -0
  245. package/node_modules/prebuild-install/bin.js +78 -0
  246. package/node_modules/prebuild-install/download.js +142 -0
  247. package/node_modules/prebuild-install/error.js +14 -0
  248. package/node_modules/prebuild-install/help.txt +16 -0
  249. package/node_modules/prebuild-install/index.js +1 -0
  250. package/node_modules/prebuild-install/log.js +33 -0
  251. package/node_modules/prebuild-install/node_modules/node-abi/LICENSE +21 -0
  252. package/node_modules/prebuild-install/node_modules/node-abi/README.md +54 -0
  253. package/node_modules/prebuild-install/node_modules/node-abi/abi_registry.json +439 -0
  254. package/node_modules/prebuild-install/node_modules/node-abi/index.js +179 -0
  255. package/node_modules/prebuild-install/node_modules/node-abi/package.json +45 -0
  256. package/node_modules/prebuild-install/node_modules/semver/LICENSE +15 -0
  257. package/node_modules/prebuild-install/node_modules/semver/README.md +665 -0
  258. package/node_modules/prebuild-install/node_modules/semver/bin/semver.js +191 -0
  259. package/node_modules/prebuild-install/node_modules/semver/classes/comparator.js +143 -0
  260. package/node_modules/prebuild-install/node_modules/semver/classes/index.js +7 -0
  261. package/node_modules/prebuild-install/node_modules/semver/classes/range.js +557 -0
  262. package/node_modules/prebuild-install/node_modules/semver/classes/semver.js +333 -0
  263. package/node_modules/prebuild-install/node_modules/semver/functions/clean.js +8 -0
  264. package/node_modules/prebuild-install/node_modules/semver/functions/cmp.js +54 -0
  265. package/node_modules/prebuild-install/node_modules/semver/functions/coerce.js +62 -0
  266. package/node_modules/prebuild-install/node_modules/semver/functions/compare-build.js +9 -0
  267. package/node_modules/prebuild-install/node_modules/semver/functions/compare-loose.js +5 -0
  268. package/node_modules/prebuild-install/node_modules/semver/functions/compare.js +7 -0
  269. package/node_modules/prebuild-install/node_modules/semver/functions/diff.js +60 -0
  270. package/node_modules/prebuild-install/node_modules/semver/functions/eq.js +5 -0
  271. package/node_modules/prebuild-install/node_modules/semver/functions/gt.js +5 -0
  272. package/node_modules/prebuild-install/node_modules/semver/functions/gte.js +5 -0
  273. package/node_modules/prebuild-install/node_modules/semver/functions/inc.js +21 -0
  274. package/node_modules/prebuild-install/node_modules/semver/functions/lt.js +5 -0
  275. package/node_modules/prebuild-install/node_modules/semver/functions/lte.js +5 -0
  276. package/node_modules/prebuild-install/node_modules/semver/functions/major.js +5 -0
  277. package/node_modules/prebuild-install/node_modules/semver/functions/minor.js +5 -0
  278. package/node_modules/prebuild-install/node_modules/semver/functions/neq.js +5 -0
  279. package/node_modules/prebuild-install/node_modules/semver/functions/parse.js +18 -0
  280. package/node_modules/prebuild-install/node_modules/semver/functions/patch.js +5 -0
  281. package/node_modules/prebuild-install/node_modules/semver/functions/prerelease.js +8 -0
  282. package/node_modules/prebuild-install/node_modules/semver/functions/rcompare.js +5 -0
  283. package/node_modules/prebuild-install/node_modules/semver/functions/rsort.js +5 -0
  284. package/node_modules/prebuild-install/node_modules/semver/functions/satisfies.js +12 -0
  285. package/node_modules/prebuild-install/node_modules/semver/functions/sort.js +5 -0
  286. package/node_modules/prebuild-install/node_modules/semver/functions/valid.js +8 -0
  287. package/node_modules/prebuild-install/node_modules/semver/index.js +91 -0
  288. package/node_modules/prebuild-install/node_modules/semver/internal/constants.js +37 -0
  289. package/node_modules/prebuild-install/node_modules/semver/internal/debug.js +11 -0
  290. package/node_modules/prebuild-install/node_modules/semver/internal/identifiers.js +29 -0
  291. package/node_modules/prebuild-install/node_modules/semver/internal/lrucache.js +42 -0
  292. package/node_modules/prebuild-install/node_modules/semver/internal/parse-options.js +17 -0
  293. package/node_modules/prebuild-install/node_modules/semver/internal/re.js +223 -0
  294. package/node_modules/prebuild-install/node_modules/semver/package.json +78 -0
  295. package/node_modules/prebuild-install/node_modules/semver/preload.js +4 -0
  296. package/node_modules/prebuild-install/node_modules/semver/range.bnf +16 -0
  297. package/node_modules/prebuild-install/node_modules/semver/ranges/gtr.js +6 -0
  298. package/node_modules/prebuild-install/node_modules/semver/ranges/intersects.js +9 -0
  299. package/node_modules/prebuild-install/node_modules/semver/ranges/ltr.js +6 -0
  300. package/node_modules/prebuild-install/node_modules/semver/ranges/max-satisfying.js +27 -0
  301. package/node_modules/prebuild-install/node_modules/semver/ranges/min-satisfying.js +26 -0
  302. package/node_modules/prebuild-install/node_modules/semver/ranges/min-version.js +63 -0
  303. package/node_modules/prebuild-install/node_modules/semver/ranges/outside.js +82 -0
  304. package/node_modules/prebuild-install/node_modules/semver/ranges/simplify.js +49 -0
  305. package/node_modules/prebuild-install/node_modules/semver/ranges/subset.js +249 -0
  306. package/node_modules/prebuild-install/node_modules/semver/ranges/to-comparators.js +10 -0
  307. package/node_modules/prebuild-install/node_modules/semver/ranges/valid.js +13 -0
  308. package/node_modules/prebuild-install/package.json +67 -0
  309. package/node_modules/prebuild-install/proxy.js +35 -0
  310. package/node_modules/prebuild-install/rc.js +64 -0
  311. package/node_modules/prebuild-install/util.js +143 -0
  312. package/node_modules/pump/.github/FUNDING.yml +2 -0
  313. package/node_modules/pump/.travis.yml +5 -0
  314. package/node_modules/pump/LICENSE +21 -0
  315. package/node_modules/pump/README.md +74 -0
  316. package/node_modules/pump/SECURITY.md +5 -0
  317. package/node_modules/pump/empty.js +1 -0
  318. package/node_modules/pump/index.js +86 -0
  319. package/node_modules/pump/package.json +30 -0
  320. package/node_modules/pump/test-browser.js +66 -0
  321. package/node_modules/pump/test-node.js +53 -0
  322. package/node_modules/rc/LICENSE.APACHE2 +15 -0
  323. package/node_modules/rc/LICENSE.BSD +26 -0
  324. package/node_modules/rc/LICENSE.MIT +24 -0
  325. package/node_modules/rc/README.md +227 -0
  326. package/node_modules/rc/browser.js +7 -0
  327. package/node_modules/rc/cli.js +4 -0
  328. package/node_modules/rc/index.js +53 -0
  329. package/node_modules/rc/lib/utils.js +104 -0
  330. package/node_modules/rc/package.json +29 -0
  331. package/node_modules/rc/test/ini.js +16 -0
  332. package/node_modules/rc/test/nested-env-vars.js +50 -0
  333. package/node_modules/rc/test/test.js +59 -0
  334. package/node_modules/readable-stream/CONTRIBUTING.md +38 -0
  335. package/node_modules/readable-stream/GOVERNANCE.md +136 -0
  336. package/node_modules/readable-stream/LICENSE +47 -0
  337. package/node_modules/readable-stream/README.md +106 -0
  338. package/node_modules/readable-stream/errors-browser.js +127 -0
  339. package/node_modules/readable-stream/errors.js +116 -0
  340. package/node_modules/readable-stream/experimentalWarning.js +17 -0
  341. package/node_modules/readable-stream/lib/_stream_duplex.js +126 -0
  342. package/node_modules/readable-stream/lib/_stream_passthrough.js +37 -0
  343. package/node_modules/readable-stream/lib/_stream_readable.js +1027 -0
  344. package/node_modules/readable-stream/lib/_stream_transform.js +190 -0
  345. package/node_modules/readable-stream/lib/_stream_writable.js +641 -0
  346. package/node_modules/readable-stream/lib/internal/streams/async_iterator.js +180 -0
  347. package/node_modules/readable-stream/lib/internal/streams/buffer_list.js +183 -0
  348. package/node_modules/readable-stream/lib/internal/streams/destroy.js +96 -0
  349. package/node_modules/readable-stream/lib/internal/streams/end-of-stream.js +86 -0
  350. package/node_modules/readable-stream/lib/internal/streams/from-browser.js +3 -0
  351. package/node_modules/readable-stream/lib/internal/streams/from.js +52 -0
  352. package/node_modules/readable-stream/lib/internal/streams/pipeline.js +86 -0
  353. package/node_modules/readable-stream/lib/internal/streams/state.js +22 -0
  354. package/node_modules/readable-stream/lib/internal/streams/stream-browser.js +1 -0
  355. package/node_modules/readable-stream/lib/internal/streams/stream.js +1 -0
  356. package/node_modules/readable-stream/package.json +68 -0
  357. package/node_modules/readable-stream/readable-browser.js +9 -0
  358. package/node_modules/readable-stream/readable.js +16 -0
  359. package/node_modules/simple-concat/.travis.yml +3 -0
  360. package/node_modules/simple-concat/LICENSE +20 -0
  361. package/node_modules/simple-concat/README.md +44 -0
  362. package/node_modules/simple-concat/index.js +15 -0
  363. package/node_modules/simple-concat/package.json +47 -0
  364. package/node_modules/simple-concat/test/basic.js +41 -0
  365. package/node_modules/simple-get/.github/dependabot.yml +15 -0
  366. package/node_modules/simple-get/.github/workflows/ci.yml +23 -0
  367. package/node_modules/simple-get/LICENSE +20 -0
  368. package/node_modules/simple-get/README.md +333 -0
  369. package/node_modules/simple-get/index.js +108 -0
  370. package/node_modules/simple-get/package.json +67 -0
  371. package/node_modules/string_decoder/LICENSE +48 -0
  372. package/node_modules/string_decoder/README.md +47 -0
  373. package/node_modules/string_decoder/lib/string_decoder.js +296 -0
  374. package/node_modules/string_decoder/package.json +34 -0
  375. package/node_modules/strip-json-comments/index.js +70 -0
  376. package/node_modules/strip-json-comments/license +21 -0
  377. package/node_modules/strip-json-comments/package.json +42 -0
  378. package/node_modules/strip-json-comments/readme.md +64 -0
  379. package/node_modules/tar-fs/.travis.yml +6 -0
  380. package/node_modules/tar-fs/LICENSE +21 -0
  381. package/node_modules/tar-fs/README.md +165 -0
  382. package/node_modules/tar-fs/index.js +363 -0
  383. package/node_modules/tar-fs/node_modules/chownr/LICENSE +15 -0
  384. package/node_modules/tar-fs/node_modules/chownr/README.md +3 -0
  385. package/node_modules/tar-fs/node_modules/chownr/chownr.js +167 -0
  386. package/node_modules/tar-fs/node_modules/chownr/package.json +29 -0
  387. package/node_modules/tar-fs/package.json +41 -0
  388. package/node_modules/tar-fs/test/fixtures/a/hello.txt +1 -0
  389. package/node_modules/tar-fs/test/fixtures/b/a/test.txt +1 -0
  390. package/node_modules/tar-fs/test/fixtures/d/file1 +0 -0
  391. package/node_modules/tar-fs/test/fixtures/d/file2 +0 -0
  392. package/node_modules/tar-fs/test/fixtures/d/sub-dir/file5 +0 -0
  393. package/node_modules/tar-fs/test/fixtures/d/sub-files/file3 +0 -0
  394. package/node_modules/tar-fs/test/fixtures/d/sub-files/file4 +0 -0
  395. package/node_modules/tar-fs/test/fixtures/e/directory/.ignore +0 -0
  396. package/node_modules/tar-fs/test/fixtures/e/file +0 -0
  397. package/node_modules/tar-fs/test/fixtures/invalid.tar +0 -0
  398. package/node_modules/tar-fs/test/index.js +346 -0
  399. package/node_modules/tar-stream/LICENSE +21 -0
  400. package/node_modules/tar-stream/README.md +168 -0
  401. package/node_modules/tar-stream/extract.js +257 -0
  402. package/node_modules/tar-stream/headers.js +295 -0
  403. package/node_modules/tar-stream/index.js +2 -0
  404. package/node_modules/tar-stream/pack.js +255 -0
  405. package/node_modules/tar-stream/package.json +58 -0
  406. package/node_modules/tar-stream/sandbox.js +11 -0
  407. package/node_modules/tunnel-agent/LICENSE +55 -0
  408. package/node_modules/tunnel-agent/README.md +4 -0
  409. package/node_modules/tunnel-agent/index.js +244 -0
  410. package/node_modules/tunnel-agent/package.json +22 -0
  411. package/node_modules/util-deprecate/History.md +16 -0
  412. package/node_modules/util-deprecate/LICENSE +24 -0
  413. package/node_modules/util-deprecate/README.md +53 -0
  414. package/node_modules/util-deprecate/browser.js +67 -0
  415. package/node_modules/util-deprecate/node.js +6 -0
  416. package/node_modules/util-deprecate/package.json +27 -0
  417. package/node_modules/uuid/CHANGELOG.md +274 -0
  418. package/node_modules/uuid/CONTRIBUTING.md +18 -0
  419. package/node_modules/uuid/LICENSE.md +9 -0
  420. package/node_modules/uuid/README.md +466 -0
  421. package/node_modules/uuid/dist/bin/uuid +2 -0
  422. package/node_modules/uuid/dist/commonjs-browser/index.js +79 -0
  423. package/node_modules/uuid/dist/commonjs-browser/md5.js +223 -0
  424. package/node_modules/uuid/dist/commonjs-browser/native.js +11 -0
  425. package/node_modules/uuid/dist/commonjs-browser/nil.js +8 -0
  426. package/node_modules/uuid/dist/commonjs-browser/parse.js +45 -0
  427. package/node_modules/uuid/dist/commonjs-browser/regex.js +8 -0
  428. package/node_modules/uuid/dist/commonjs-browser/rng.js +25 -0
  429. package/node_modules/uuid/dist/commonjs-browser/sha1.js +104 -0
  430. package/node_modules/uuid/dist/commonjs-browser/stringify.js +44 -0
  431. package/node_modules/uuid/dist/commonjs-browser/v1.js +107 -0
  432. package/node_modules/uuid/dist/commonjs-browser/v3.js +16 -0
  433. package/node_modules/uuid/dist/commonjs-browser/v35.js +80 -0
  434. package/node_modules/uuid/dist/commonjs-browser/v4.js +43 -0
  435. package/node_modules/uuid/dist/commonjs-browser/v5.js +16 -0
  436. package/node_modules/uuid/dist/commonjs-browser/validate.js +17 -0
  437. package/node_modules/uuid/dist/commonjs-browser/version.js +21 -0
  438. package/node_modules/uuid/dist/esm-browser/index.js +9 -0
  439. package/node_modules/uuid/dist/esm-browser/md5.js +215 -0
  440. package/node_modules/uuid/dist/esm-browser/native.js +4 -0
  441. package/node_modules/uuid/dist/esm-browser/nil.js +1 -0
  442. package/node_modules/uuid/dist/esm-browser/parse.js +35 -0
  443. package/node_modules/uuid/dist/esm-browser/regex.js +1 -0
  444. package/node_modules/uuid/dist/esm-browser/rng.js +18 -0
  445. package/node_modules/uuid/dist/esm-browser/sha1.js +96 -0
  446. package/node_modules/uuid/dist/esm-browser/stringify.js +33 -0
  447. package/node_modules/uuid/dist/esm-browser/v1.js +95 -0
  448. package/node_modules/uuid/dist/esm-browser/v3.js +4 -0
  449. package/node_modules/uuid/dist/esm-browser/v35.js +66 -0
  450. package/node_modules/uuid/dist/esm-browser/v4.js +29 -0
  451. package/node_modules/uuid/dist/esm-browser/v5.js +4 -0
  452. package/node_modules/uuid/dist/esm-browser/validate.js +7 -0
  453. package/node_modules/uuid/dist/esm-browser/version.js +11 -0
  454. package/node_modules/uuid/dist/esm-node/index.js +9 -0
  455. package/node_modules/uuid/dist/esm-node/md5.js +13 -0
  456. package/node_modules/uuid/dist/esm-node/native.js +4 -0
  457. package/node_modules/uuid/dist/esm-node/nil.js +1 -0
  458. package/node_modules/uuid/dist/esm-node/parse.js +35 -0
  459. package/node_modules/uuid/dist/esm-node/regex.js +1 -0
  460. package/node_modules/uuid/dist/esm-node/rng.js +12 -0
  461. package/node_modules/uuid/dist/esm-node/sha1.js +13 -0
  462. package/node_modules/uuid/dist/esm-node/stringify.js +33 -0
  463. package/node_modules/uuid/dist/esm-node/v1.js +95 -0
  464. package/node_modules/uuid/dist/esm-node/v3.js +4 -0
  465. package/node_modules/uuid/dist/esm-node/v35.js +66 -0
  466. package/node_modules/uuid/dist/esm-node/v4.js +29 -0
  467. package/node_modules/uuid/dist/esm-node/v5.js +4 -0
  468. package/node_modules/uuid/dist/esm-node/validate.js +7 -0
  469. package/node_modules/uuid/dist/esm-node/version.js +11 -0
  470. package/node_modules/uuid/dist/index.js +79 -0
  471. package/node_modules/uuid/dist/md5-browser.js +223 -0
  472. package/node_modules/uuid/dist/md5.js +23 -0
  473. package/node_modules/uuid/dist/native-browser.js +11 -0
  474. package/node_modules/uuid/dist/native.js +15 -0
  475. package/node_modules/uuid/dist/nil.js +8 -0
  476. package/node_modules/uuid/dist/parse.js +45 -0
  477. package/node_modules/uuid/dist/regex.js +8 -0
  478. package/node_modules/uuid/dist/rng-browser.js +25 -0
  479. package/node_modules/uuid/dist/rng.js +24 -0
  480. package/node_modules/uuid/dist/sha1-browser.js +104 -0
  481. package/node_modules/uuid/dist/sha1.js +23 -0
  482. package/node_modules/uuid/dist/stringify.js +44 -0
  483. package/node_modules/uuid/dist/uuid-bin.js +85 -0
  484. package/node_modules/uuid/dist/v1.js +107 -0
  485. package/node_modules/uuid/dist/v3.js +16 -0
  486. package/node_modules/uuid/dist/v35.js +80 -0
  487. package/node_modules/uuid/dist/v4.js +43 -0
  488. package/node_modules/uuid/dist/v5.js +16 -0
  489. package/node_modules/uuid/dist/validate.js +17 -0
  490. package/node_modules/uuid/dist/version.js +21 -0
  491. package/node_modules/uuid/package.json +135 -0
  492. package/node_modules/uuid/wrapper.mjs +10 -0
  493. package/package.json +5 -3
  494. package/packages/cli/package.json +1 -1
  495. package/packages/daemon/package.json +1 -1
  496. package/packages/daemon/src/api.js +39 -23
  497. package/packages/daemon/src/introducer.js +21 -6
  498. package/packages/daemon/src/process.js +20 -2
  499. package/packages/daemon/src/providers/gemini.js +27 -28
  500. package/packages/gui/dist/assets/{index-VB4_k5Pz.js → index-BgTyFy4f.js} +331 -331
  501. package/packages/gui/dist/assets/{index-Bo6AeNmM.css → index-QADLyUj5.css} +1 -1
  502. package/packages/gui/dist/index.html +2 -2
  503. package/packages/gui/package.json +1 -1
  504. package/packages/gui/src/components/agents/agent-file-tree.jsx +68 -16
  505. package/packages/gui/src/components/agents/agent-panel.jsx +2 -0
  506. package/packages/gui/src/components/agents/code-review.jsx +9 -1
  507. package/packages/gui/src/components/agents/workspace-mode.jsx +2 -2
  508. package/packages/gui/src/components/editor/code-editor.jsx +1 -1
  509. package/packages/gui/src/stores/groove.js +40 -26
  510. package/packages/gui/src/views/agents.jsx +14 -65
  511. package/workspace.png +0 -0
@@ -0,0 +1,135 @@
1
+ // FSL-1.1-Apache-2.0 — see LICENSE
2
+
3
+ import { describe, it } from 'node:test';
4
+ import assert from 'node:assert/strict';
5
+ import { StepClassifier } from '../../client/step-classifier.js';
6
+
7
+ describe('StepClassifier', () => {
8
+ it('user message before any action is not a correction', () => {
9
+ const classifier = new StepClassifier();
10
+ const result = classifier.classifyUserMessage('fix the bug');
11
+ assert.equal(result, null);
12
+ });
13
+
14
+ it('user message after action is a correction', () => {
15
+ const classifier = new StepClassifier();
16
+ classifier.onStep({ type: 'action' });
17
+ const result = classifier.classifyUserMessage('no, use exponential backoff');
18
+ assert.equal(result.type, 'correction');
19
+ assert.equal(result.content, 'no, use exponential backoff');
20
+ assert.equal(result.source, 'user');
21
+ });
22
+
23
+ it('classifies coordination event', () => {
24
+ const classifier = new StepClassifier();
25
+ const result = classifier.classifyCoordinationEvent({
26
+ coordination_id: 'coord-1',
27
+ direction: 'outbound',
28
+ target_agent: 'backend-1',
29
+ protocol: 'knock',
30
+ content: 'Requesting lock on src/api.js',
31
+ });
32
+ assert.equal(result.type, 'coordination');
33
+ assert.equal(result.coordination_id, 'coord-1');
34
+ assert.equal(result.direction, 'outbound');
35
+ assert.equal(result.target_agent, 'backend-1');
36
+ assert.equal(result.protocol, 'knock');
37
+ });
38
+
39
+ it('detects error recovery', () => {
40
+ const steps = [
41
+ { type: 'thought', step: 1 },
42
+ { type: 'action', step: 2 },
43
+ { type: 'error', step: 3 },
44
+ { type: 'thought', step: 4 },
45
+ { type: 'action', step: 5 },
46
+ { type: 'resolution', step: 6 },
47
+ ];
48
+ assert.equal(StepClassifier.detectErrorRecovery(steps), true);
49
+ });
50
+
51
+ it('no error recovery when no resolution after error', () => {
52
+ const steps = [
53
+ { type: 'thought', step: 1 },
54
+ { type: 'error', step: 2 },
55
+ { type: 'error', step: 3 },
56
+ ];
57
+ assert.equal(StepClassifier.detectErrorRecovery(steps), false);
58
+ });
59
+
60
+ it('no error recovery when no errors', () => {
61
+ const steps = [
62
+ { type: 'thought', step: 1 },
63
+ { type: 'action', step: 2 },
64
+ { type: 'resolution', step: 3 },
65
+ ];
66
+ assert.equal(StepClassifier.detectErrorRecovery(steps), false);
67
+ });
68
+
69
+ it('counts user interventions', () => {
70
+ const steps = [
71
+ { type: 'thought' },
72
+ { type: 'correction' },
73
+ { type: 'action' },
74
+ { type: 'correction' },
75
+ { type: 'resolution' },
76
+ ];
77
+ assert.equal(StepClassifier.countUserInterventions(steps), 2);
78
+ });
79
+
80
+ it('counts zero interventions when none present', () => {
81
+ const steps = [
82
+ { type: 'thought' },
83
+ { type: 'action' },
84
+ { type: 'resolution' },
85
+ ];
86
+ assert.equal(StepClassifier.countUserInterventions(steps), 0);
87
+ });
88
+
89
+ it('reclassifies action with error content to error', () => {
90
+ const classifier = new StepClassifier();
91
+ const step = { type: 'action', content: 'Command failed with exit code 1' };
92
+ const result = classifier.onStep(step);
93
+ assert.equal(result.type, 'error');
94
+ });
95
+
96
+ it('reclassifies observation with error content to error', () => {
97
+ const classifier = new StepClassifier();
98
+ const step = { type: 'observation', content: 'TypeError: cannot read properties of undefined' };
99
+ const result = classifier.onStep(step);
100
+ assert.equal(result.type, 'error');
101
+ });
102
+
103
+ it('does not reclassify thought with error content', () => {
104
+ const classifier = new StepClassifier();
105
+ const step = { type: 'thought', content: 'I see the Error and will fix it' };
106
+ const result = classifier.onStep(step);
107
+ assert.equal(result.type, 'thought');
108
+ });
109
+
110
+ it('marks thought after correction as correction_context', () => {
111
+ const classifier = new StepClassifier();
112
+ classifier.onStep({ type: 'action' });
113
+ classifier.onStep({ type: 'correction', content: 'no, fix the bug' });
114
+ const step = { type: 'thought', content: 'I see the issue, let me fix it' };
115
+ const result = classifier.onStep(step);
116
+ assert.equal(result.type, 'thought');
117
+ assert.equal(result.correction_context, true);
118
+ });
119
+
120
+ it('does not mark thought as correction_context without prior correction', () => {
121
+ const classifier = new StepClassifier();
122
+ classifier.onStep({ type: 'action', content: 'running test' });
123
+ const step = { type: 'thought', content: 'let me fix this' };
124
+ const result = classifier.onStep(step);
125
+ assert.equal(result.correction_context, undefined);
126
+ });
127
+
128
+ it('returns the step from onStep', () => {
129
+ const classifier = new StepClassifier();
130
+ const step = { type: 'action', content: 'hello' };
131
+ const result = classifier.onStep(step);
132
+ assert.ok(result);
133
+ assert.equal(result.type, 'action');
134
+ });
135
+ });
@@ -0,0 +1,33 @@
1
+ // FSL-1.1-Apache-2.0 — see LICENSE
2
+
3
+ import { describe, it } from 'node:test';
4
+ import assert from 'node:assert/strict';
5
+ import { TransmissionQueue } from '../../client/transmission-queue.js';
6
+
7
+ describe('TransmissionQueue', () => {
8
+ it('waitForDrain resolves immediately when queue is empty', async () => {
9
+ const queue = new TransmissionQueue('http://localhost:9999');
10
+ await queue.waitForDrain();
11
+ assert.ok(true);
12
+ });
13
+
14
+ it('waitForDrain waits for active drain to complete', async () => {
15
+ const queue = new TransmissionQueue('http://localhost:9999');
16
+ queue._queue.push({ session_id: 'test' });
17
+ const drain = Promise.resolve().then(() => {
18
+ queue._queue.length = 0;
19
+ });
20
+ queue._drainPromise = drain.finally(() => {
21
+ queue._drainPromise = null;
22
+ });
23
+ await queue.waitForDrain();
24
+ assert.equal(queue._queue.length, 0);
25
+ });
26
+
27
+ it('offlineQueueSize tracks offline envelopes', () => {
28
+ const queue = new TransmissionQueue('http://localhost:9999');
29
+ assert.equal(queue.offlineQueueSize, 0);
30
+ queue.enqueue({ session_id: 'test', attestation: { session_hmac: 'OFFLINE' } });
31
+ assert.equal(queue.offlineQueueSize, 1);
32
+ });
33
+ });
@@ -0,0 +1,260 @@
1
+ // FSL-1.1-Apache-2.0 — see LICENSE
2
+
3
+ import { describe, it, beforeEach, afterEach } from 'node:test';
4
+ import assert from 'node:assert/strict';
5
+ import { mkdtempSync, rmSync } from 'node:fs';
6
+ import { join } from 'node:path';
7
+ import { tmpdir } from 'node:os';
8
+ import { generateECDHKeypair, deriveSharedSecret, signEnvelope, verifyEnvelope } from '../../shared/crypto.js';
9
+ import { SessionRegistry } from '../../server/session-registry.js';
10
+ import { EnvelopeVerifier } from '../../server/verifier.js';
11
+ import { SessionAttestation } from '../../client/session-attestation.js';
12
+ import { EnvelopeBuilder } from '../../client/envelope-builder.js';
13
+
14
+ describe('ECDH Handshake End-to-End', () => {
15
+ let tmpDir, registry, verifier;
16
+
17
+ beforeEach(() => {
18
+ tmpDir = mkdtempSync(join(tmpdir(), 'handshake-test-'));
19
+ registry = new SessionRegistry(join(tmpDir, 'sessions.db'));
20
+ verifier = new EnvelopeVerifier(registry);
21
+ });
22
+
23
+ afterEach(() => {
24
+ registry.close();
25
+ rmSync(tmpDir, { recursive: true, force: true });
26
+ });
27
+
28
+ it('client and server derive the same shared secret', () => {
29
+ const clientKeypair = generateECDHKeypair();
30
+ const result = registry.openSession(
31
+ 'sess_hs_001', clientKeypair.publicKey, 'claude-code', 'claude-opus-4-6',
32
+ 'fp_test', 'hash_test', '0.27.77'
33
+ );
34
+
35
+ const serverSession = registry.getSession('sess_hs_001');
36
+ const clientSecret = deriveSharedSecret(clientKeypair.privateKey, result.serverPublicKey);
37
+ assert.equal(clientSecret, serverSession.shared_secret);
38
+ });
39
+
40
+ it('client-signed envelope passes server verification', () => {
41
+ const clientKeypair = generateECDHKeypair();
42
+ const result = registry.openSession(
43
+ 'sess_hs_002', clientKeypair.publicKey, 'claude-code', 'claude-opus-4-6',
44
+ 'fp_test', 'hash_test', '0.27.77'
45
+ );
46
+
47
+ const serverSession = registry.getSession('sess_hs_002');
48
+ const sharedSecret = deriveSharedSecret(clientKeypair.privateKey, result.serverPublicKey);
49
+ assert.equal(sharedSecret, serverSession.shared_secret);
50
+
51
+ const envelope = {
52
+ envelope_id: 'env_hs_001',
53
+ session_id: 'sess_hs_002',
54
+ chunk_sequence: 0,
55
+ contributor_id: 'cccccccccccccccccccccccccccccccc',
56
+ metadata: {
57
+ model_engine: 'claude-opus-4-6',
58
+ provider: 'claude-code',
59
+ agent_role: 'frontend',
60
+ agent_id: 'frontend-1',
61
+ },
62
+ trajectory_log: [
63
+ { step: 1, type: 'thought', timestamp: Date.now() / 1000, content: 'planning', token_count: 10 },
64
+ ],
65
+ attestation: { session_hmac: '', sequence: 0, app_version_hash: '' },
66
+ };
67
+
68
+ const envelopeForHmac = { ...envelope };
69
+ delete envelopeForHmac.attestation;
70
+ const envelopeBytes = JSON.stringify(envelopeForHmac);
71
+ const hmac = signEnvelope(sharedSecret, envelopeBytes, 0);
72
+ envelope.attestation = { session_hmac: hmac, sequence: 0, app_version_hash: 'bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb' };
73
+
74
+ const verifyResult = verifier.verify(envelope);
75
+ assert.equal(verifyResult.valid, true);
76
+ });
77
+
78
+ it('tampered envelope is rejected by server', () => {
79
+ const clientKeypair = generateECDHKeypair();
80
+ const result = registry.openSession(
81
+ 'sess_hs_003', clientKeypair.publicKey, 'claude-code', 'claude-opus-4-6',
82
+ 'fp_test', 'hash_test', '0.27.77'
83
+ );
84
+
85
+ const sharedSecret = deriveSharedSecret(clientKeypair.privateKey, result.serverPublicKey);
86
+
87
+ const envelope = {
88
+ envelope_id: 'env_hs_002',
89
+ session_id: 'sess_hs_003',
90
+ chunk_sequence: 0,
91
+ contributor_id: 'cccccccccccccccccccccccccccccccc',
92
+ metadata: {
93
+ model_engine: 'claude-opus-4-6',
94
+ provider: 'claude-code',
95
+ agent_role: 'frontend',
96
+ agent_id: 'frontend-1',
97
+ },
98
+ trajectory_log: [
99
+ { step: 1, type: 'thought', timestamp: Date.now() / 1000, content: 'original', token_count: 10 },
100
+ ],
101
+ attestation: { session_hmac: '', sequence: 0, app_version_hash: '' },
102
+ };
103
+
104
+ const envelopeForHmac = { ...envelope };
105
+ delete envelopeForHmac.attestation;
106
+ const envelopeBytes = JSON.stringify(envelopeForHmac);
107
+ const hmac = signEnvelope(sharedSecret, envelopeBytes, 0);
108
+ envelope.attestation = { session_hmac: hmac, sequence: 0, app_version_hash: 'bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb' };
109
+
110
+ envelope.trajectory_log.push({ step: 2, type: 'action', timestamp: Date.now() / 1000, content: 'injected' });
111
+
112
+ const verifyResult = verifier.verify(envelope);
113
+ assert.equal(verifyResult.valid, false);
114
+ assert.ok(verifyResult.reason.includes('HMAC'));
115
+ });
116
+
117
+ it('sequence numbers enforce ordering', () => {
118
+ const clientKeypair = generateECDHKeypair();
119
+ const result = registry.openSession(
120
+ 'sess_hs_004', clientKeypair.publicKey, 'claude-code', 'claude-opus-4-6',
121
+ 'fp_test', 'hash_test', '0.27.77'
122
+ );
123
+ const sharedSecret = deriveSharedSecret(clientKeypair.privateKey, result.serverPublicKey);
124
+
125
+ function makeEnv(seq) {
126
+ const env = {
127
+ envelope_id: `env_hs_seq_${seq}`,
128
+ session_id: 'sess_hs_004',
129
+ chunk_sequence: seq,
130
+ contributor_id: 'cccccccccccccccccccccccccccccccc',
131
+ metadata: { model_engine: 'claude-opus-4-6', provider: 'claude-code', agent_role: 'frontend', agent_id: 'frontend-1' },
132
+ trajectory_log: [{ step: seq + 1, type: 'thought', timestamp: Date.now() / 1000, content: `step ${seq}`, token_count: 5 }],
133
+ };
134
+ const bytes = JSON.stringify(env);
135
+ const hmac = signEnvelope(sharedSecret, bytes, seq);
136
+ env.attestation = { session_hmac: hmac, sequence: seq, app_version_hash: 'bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb' };
137
+ return env;
138
+ }
139
+
140
+ const r0 = verifier.verify(makeEnv(0));
141
+ assert.equal(r0.valid, true);
142
+
143
+ const r1 = verifier.verify(makeEnv(1));
144
+ assert.equal(r1.valid, true);
145
+
146
+ const rSkip = verifier.verify(makeEnv(5));
147
+ assert.equal(rSkip.valid, false);
148
+ assert.ok(rSkip.reason.includes('sequence'));
149
+
150
+ const r2 = verifier.verify(makeEnv(2));
151
+ assert.equal(r2.valid, true);
152
+ });
153
+
154
+ it('different client keypairs produce different shared secrets', () => {
155
+ const client1 = generateECDHKeypair();
156
+ const client2 = generateECDHKeypair();
157
+
158
+ registry.openSession('sess_hs_005a', client1.publicKey, 'claude-code', 'claude-opus-4-6', 'fp1', 'h1', '0.27.77');
159
+ registry.openSession('sess_hs_005b', client2.publicKey, 'claude-code', 'claude-opus-4-6', 'fp2', 'h2', '0.27.77');
160
+
161
+ const s1 = registry.getSession('sess_hs_005a');
162
+ const s2 = registry.getSession('sess_hs_005b');
163
+
164
+ assert.notEqual(s1.shared_secret, s2.shared_secret);
165
+ });
166
+
167
+ it('EnvelopeBuilder output is verifiable after client signing', () => {
168
+ const clientKeypair = generateECDHKeypair();
169
+ const result = registry.openSession(
170
+ 'sess_hs_006', clientKeypair.publicKey, 'claude-code', 'claude-opus-4-6',
171
+ 'fp_test', 'hash_test', '0.27.77'
172
+ );
173
+ const sharedSecret = deriveSharedSecret(clientKeypair.privateKey, result.serverPublicKey);
174
+
175
+ const builder = new EnvelopeBuilder('sess_hs_006', 'cccccccccccccccccccccccccccccccc', {
176
+ model_engine: 'claude-opus-4-6',
177
+ provider: 'claude-code',
178
+ agent_role: 'frontend',
179
+ agent_id: 'frontend-1',
180
+ task_complexity: 'medium',
181
+ team_size: 1,
182
+ session_quality: 80,
183
+ groove_version: '0.27.77',
184
+ });
185
+
186
+ builder.addStep({ step: 1, type: 'thought', timestamp: Date.now() / 1000, content: 'test', token_count: 5 });
187
+ const envelope = builder.flush();
188
+ assert.ok(envelope);
189
+
190
+ const envelopeForHmac = { ...envelope };
191
+ delete envelopeForHmac.attestation;
192
+ const envelopeBytes = JSON.stringify(envelopeForHmac);
193
+ const hmac = signEnvelope(sharedSecret, envelopeBytes, 0);
194
+ envelope.attestation = { session_hmac: hmac, sequence: 0, app_version_hash: 'bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb' };
195
+
196
+ const verifyResult = verifier.verify(envelope);
197
+ assert.equal(verifyResult.valid, true);
198
+ });
199
+
200
+ it('SESSION_CLOSE from builder is verifiable', () => {
201
+ const clientKeypair = generateECDHKeypair();
202
+ const result = registry.openSession(
203
+ 'sess_hs_007', clientKeypair.publicKey, 'claude-code', 'claude-opus-4-6',
204
+ 'fp_test', 'hash_test', '0.27.77'
205
+ );
206
+ const sharedSecret = deriveSharedSecret(clientKeypair.privateKey, result.serverPublicKey);
207
+
208
+ const builder = new EnvelopeBuilder('sess_hs_007', 'cccccccccccccccccccccccccccccccc', {
209
+ model_engine: 'claude-opus-4-6',
210
+ provider: 'claude-code',
211
+ agent_role: 'frontend',
212
+ agent_id: 'frontend-1',
213
+ });
214
+
215
+ const closeEnvelope = builder.buildSessionClose({
216
+ status: 'SUCCESS',
217
+ user_interventions: 0,
218
+ total_steps: 10,
219
+ total_chunks: 1,
220
+ total_tokens: 500,
221
+ duration_seconds: 60,
222
+ files_modified: 2,
223
+ errors_encountered: 0,
224
+ errors_recovered: 0,
225
+ coordination_events: 0,
226
+ });
227
+
228
+ const forHmac = { ...closeEnvelope };
229
+ delete forHmac.attestation;
230
+ const bytes = JSON.stringify(forHmac);
231
+ const hmac = signEnvelope(sharedSecret, bytes, 0);
232
+ closeEnvelope.attestation = { session_hmac: hmac, sequence: 0, app_version_hash: 'bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb' };
233
+
234
+ const verifyResult = verifier.verifyClose(closeEnvelope);
235
+ assert.equal(verifyResult.valid, true);
236
+
237
+ const session = registry.getSession('sess_hs_007');
238
+ assert.equal(session.status, 'closed');
239
+ });
240
+
241
+ it('HMAC verification is constant-time (no short-circuit)', () => {
242
+ const clientKeypair = generateECDHKeypair();
243
+ registry.openSession(
244
+ 'sess_hs_008', clientKeypair.publicKey, 'claude-code', 'claude-opus-4-6',
245
+ 'fp_test', 'hash_test', '0.27.77'
246
+ );
247
+ const session = registry.getSession('sess_hs_008');
248
+ const sharedSecret = session.shared_secret;
249
+
250
+ const payload = JSON.stringify({ test: 'data' });
251
+ const validHmac = signEnvelope(sharedSecret, payload, 0);
252
+
253
+ const wrongFirstChar = String.fromCharCode(validHmac.charCodeAt(0) ^ 1) + validHmac.slice(1);
254
+ const wrongLastChar = validHmac.slice(0, -1) + String.fromCharCode(validHmac.charCodeAt(validHmac.length - 1) ^ 1);
255
+
256
+ assert.equal(verifyEnvelope(sharedSecret, payload, 0, wrongFirstChar), false);
257
+ assert.equal(verifyEnvelope(sharedSecret, payload, 0, wrongLastChar), false);
258
+ assert.equal(verifyEnvelope(sharedSecret, payload, 0, validHmac), true);
259
+ });
260
+ });
@@ -0,0 +1,166 @@
1
+ // FSL-1.1-Apache-2.0 — see LICENSE
2
+
3
+ import { describe, it, beforeEach, afterEach } from 'node:test';
4
+ import assert from 'node:assert/strict';
5
+ import { mkdtempSync, rmSync } from 'node:fs';
6
+ import { join } from 'node:path';
7
+ import { tmpdir } from 'node:os';
8
+ import { generateECDHKeypair, signEnvelope } from '../../shared/crypto.js';
9
+ import { SessionRegistry } from '../../server/session-registry.js';
10
+ import { EnvelopeVerifier } from '../../server/verifier.js';
11
+ import { EnvelopeStorage } from '../../server/storage.js';
12
+
13
+ const VALID_CONTRIBUTOR = 'c'.repeat(32);
14
+ const VALID_APP_HASH = 'b'.repeat(64);
15
+
16
+ function makeSignedEnvelope(sessionId, sequence, sharedSecret, overrides = {}) {
17
+ const envelope = {
18
+ envelope_id: `env_test_${sequence}`,
19
+ session_id: sessionId,
20
+ chunk_sequence: sequence,
21
+ contributor_id: VALID_CONTRIBUTOR,
22
+ metadata: { model_engine: 'claude-opus-4-6', provider: 'claude-code', agent_role: 'backend', agent_id: 'backend-1' },
23
+ trajectory_log: [{ step: 1, type: 'thought', timestamp: Date.now() / 1000, content: 'test', token_count: 10 }],
24
+ ...overrides,
25
+ };
26
+
27
+ const forHmac = { ...envelope };
28
+ const envelopeBytes = JSON.stringify(forHmac);
29
+ const hmac = signEnvelope(sharedSecret, envelopeBytes, sequence);
30
+
31
+ envelope.attestation = {
32
+ session_hmac: hmac,
33
+ sequence,
34
+ app_version_hash: VALID_APP_HASH,
35
+ };
36
+
37
+ return envelope;
38
+ }
39
+
40
+ describe('Ingest Security', () => {
41
+ let registry;
42
+ let verifier;
43
+ let storage;
44
+ let tmpDir;
45
+ let sharedSecret;
46
+ const sessionId = 'sess_ingest_sec_001';
47
+
48
+ beforeEach(() => {
49
+ tmpDir = mkdtempSync(join(tmpdir(), 'ingest-sec-'));
50
+ registry = new SessionRegistry(join(tmpDir, 'sessions.db'));
51
+ storage = new EnvelopeStorage(join(tmpDir, 'envelopes'));
52
+ verifier = new EnvelopeVerifier(registry);
53
+
54
+ const clientKeypair = generateECDHKeypair();
55
+ registry.openSession(
56
+ sessionId, clientKeypair.publicKey, 'claude-code', 'claude-opus-4-6',
57
+ 'fp_ingest', 'hash_ingest', '0.27.77'
58
+ );
59
+
60
+ const session = registry.getSession(sessionId);
61
+ sharedSecret = session.shared_secret;
62
+ });
63
+
64
+ afterEach(() => {
65
+ registry.close();
66
+ rmSync(tmpDir, { recursive: true, force: true });
67
+ });
68
+
69
+ it('rejects envelope with > 500 steps', () => {
70
+ const steps = Array.from({ length: 501 }, (_, i) => ({
71
+ step: i, type: 'thought', timestamp: Date.now() / 1000,
72
+ }));
73
+ const envelope = makeSignedEnvelope(sessionId, 0, sharedSecret, { trajectory_log: steps });
74
+ const result = verifier.verify(envelope);
75
+ assert.equal(result.valid, false);
76
+ assert.ok(result.reason.includes('schema'));
77
+ });
78
+
79
+ it('rejects envelope when session has > 200 envelopes', () => {
80
+ // Simulate 200 envelopes already received
81
+ for (let i = 0; i < 200; i++) {
82
+ registry.incrementEnvelopeCount(sessionId);
83
+ }
84
+
85
+ const withinLimit = registry.checkEnvelopeCount(sessionId, 200);
86
+ assert.equal(withinLimit, false);
87
+ });
88
+
89
+ it('server generates envelope_id (client value ignored)', () => {
90
+ const envelope = makeSignedEnvelope(sessionId, 0, sharedSecret);
91
+ const originalId = envelope.envelope_id;
92
+
93
+ // Verify passes
94
+ const result = verifier.verify(envelope);
95
+ assert.equal(result.valid, true);
96
+
97
+ // In the real ingest flow, server overwrites envelope_id
98
+ // Verify the dedup infrastructure works
99
+ const generatedId = 'env_server_generated';
100
+ registry.recordProcessedEnvelope(generatedId, sessionId);
101
+ assert.equal(registry.isEnvelopeProcessed(generatedId), true);
102
+ assert.equal(registry.isEnvelopeProcessed(originalId), false);
103
+ });
104
+
105
+ it('rejects invalid model_engine via schema validation', () => {
106
+ const envelope = makeSignedEnvelope(sessionId, 0, sharedSecret, {
107
+ metadata: { model_engine: 'gpt-5-turbo', provider: 'claude-code', agent_role: 'backend', agent_id: 'backend-1' },
108
+ });
109
+ const result = verifier.verify(envelope);
110
+ assert.equal(result.valid, false);
111
+ assert.ok(result.reason.includes('schema'));
112
+ });
113
+
114
+ it('rejects invalid contributor_id format', () => {
115
+ const envelope = makeSignedEnvelope(sessionId, 0, sharedSecret, {
116
+ contributor_id: 'not-a-valid-hex-id',
117
+ });
118
+ const result = verifier.verify(envelope);
119
+ assert.equal(result.valid, false);
120
+ assert.ok(result.reason.includes('schema'));
121
+ });
122
+
123
+ it('envelope dedup prevents double-processing', () => {
124
+ const envelopeId = 'env_dedup_test';
125
+ assert.equal(registry.isEnvelopeProcessed(envelopeId), false);
126
+
127
+ registry.recordProcessedEnvelope(envelopeId, sessionId);
128
+ assert.equal(registry.isEnvelopeProcessed(envelopeId), true);
129
+
130
+ // Recording again should not throw (INSERT OR IGNORE)
131
+ registry.recordProcessedEnvelope(envelopeId, sessionId);
132
+ assert.equal(registry.isEnvelopeProcessed(envelopeId), true);
133
+ });
134
+
135
+ it('per-session envelope count tracks correctly', () => {
136
+ assert.equal(registry.checkEnvelopeCount(sessionId, 200), true);
137
+
138
+ registry.incrementEnvelopeCount(sessionId);
139
+ const session = registry.getSession(sessionId);
140
+ assert.equal(session.envelope_count, 1);
141
+
142
+ registry.incrementEnvelopeCount(sessionId);
143
+ const session2 = registry.getSession(sessionId);
144
+ assert.equal(session2.envelope_count, 2);
145
+ });
146
+
147
+ it('atomic sequence check prevents race condition', () => {
148
+ // First call should succeed
149
+ const r1 = registry.checkAndIncrementSequence(sessionId, 0);
150
+ assert.equal(r1.valid, true);
151
+
152
+ // Same sequence again should fail
153
+ const r2 = registry.checkAndIncrementSequence(sessionId, 0);
154
+ assert.equal(r2.valid, false);
155
+ assert.ok(r2.reason.includes('sequence'));
156
+
157
+ // Next sequence should succeed
158
+ const r3 = registry.checkAndIncrementSequence(sessionId, 1);
159
+ assert.equal(r3.valid, true);
160
+ });
161
+
162
+ it('storage quota check works', () => {
163
+ const ok = storage.checkQuota();
164
+ assert.equal(ok, true);
165
+ });
166
+ });