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.
- package/CLAUDE.md +7 -0
- package/README.md +25 -52
- package/moe-training/package.json +2 -1
- package/node_modules/@groove-dev/cli/package.json +1 -1
- package/node_modules/@groove-dev/daemon/package.json +1 -1
- package/node_modules/@groove-dev/daemon/src/api.js +39 -23
- package/node_modules/@groove-dev/daemon/src/introducer.js +21 -6
- package/node_modules/@groove-dev/daemon/src/process.js +20 -2
- package/node_modules/@groove-dev/daemon/src/providers/gemini.js +27 -28
- package/node_modules/@groove-dev/gui/dist/assets/{index-VB4_k5Pz.js → index-BgTyFy4f.js} +331 -331
- package/node_modules/@groove-dev/gui/dist/assets/{index-Bo6AeNmM.css → index-QADLyUj5.css} +1 -1
- package/node_modules/@groove-dev/gui/dist/index.html +2 -2
- package/node_modules/@groove-dev/gui/package.json +1 -1
- package/node_modules/@groove-dev/gui/src/components/agents/agent-file-tree.jsx +68 -16
- package/node_modules/@groove-dev/gui/src/components/agents/agent-panel.jsx +2 -0
- package/node_modules/@groove-dev/gui/src/components/agents/code-review.jsx +9 -1
- package/node_modules/@groove-dev/gui/src/components/agents/workspace-mode.jsx +2 -2
- package/node_modules/@groove-dev/gui/src/components/editor/code-editor.jsx +1 -1
- package/node_modules/@groove-dev/gui/src/stores/groove.js +40 -26
- package/node_modules/@groove-dev/gui/src/views/agents.jsx +14 -65
- package/node_modules/base64-js/LICENSE +21 -0
- package/node_modules/base64-js/README.md +34 -0
- package/node_modules/base64-js/base64js.min.js +1 -0
- package/node_modules/base64-js/index.d.ts +3 -0
- package/node_modules/base64-js/index.js +150 -0
- package/node_modules/base64-js/package.json +47 -0
- package/node_modules/better-sqlite3/LICENSE +21 -0
- package/node_modules/better-sqlite3/README.md +99 -0
- package/node_modules/better-sqlite3/binding.gyp +38 -0
- package/node_modules/better-sqlite3/deps/common.gypi +68 -0
- package/node_modules/better-sqlite3/deps/copy.js +31 -0
- package/node_modules/better-sqlite3/deps/defines.gypi +41 -0
- package/node_modules/better-sqlite3/deps/download.sh +122 -0
- package/node_modules/better-sqlite3/deps/patches/1208.patch +15 -0
- package/node_modules/better-sqlite3/deps/sqlite3/sqlite3.c +268845 -0
- package/node_modules/better-sqlite3/deps/sqlite3/sqlite3.h +14335 -0
- package/node_modules/better-sqlite3/deps/sqlite3/sqlite3ext.h +739 -0
- package/node_modules/better-sqlite3/deps/sqlite3.gyp +80 -0
- package/node_modules/better-sqlite3/deps/test_extension.c +21 -0
- package/node_modules/better-sqlite3/lib/database.js +90 -0
- package/node_modules/better-sqlite3/lib/index.js +3 -0
- package/node_modules/better-sqlite3/lib/methods/aggregate.js +43 -0
- package/node_modules/better-sqlite3/lib/methods/backup.js +67 -0
- package/node_modules/better-sqlite3/lib/methods/function.js +31 -0
- package/node_modules/better-sqlite3/lib/methods/inspect.js +7 -0
- package/node_modules/better-sqlite3/lib/methods/pragma.js +12 -0
- package/node_modules/better-sqlite3/lib/methods/serialize.js +16 -0
- package/node_modules/better-sqlite3/lib/methods/table.js +189 -0
- package/node_modules/better-sqlite3/lib/methods/transaction.js +78 -0
- package/node_modules/better-sqlite3/lib/methods/wrappers.js +54 -0
- package/node_modules/better-sqlite3/lib/sqlite-error.js +20 -0
- package/node_modules/better-sqlite3/lib/util.js +12 -0
- package/node_modules/better-sqlite3/package.json +59 -0
- package/node_modules/better-sqlite3/src/addon.cpp +47 -0
- package/node_modules/better-sqlite3/src/better_sqlite3.cpp +74 -0
- package/node_modules/better-sqlite3/src/objects/backup.cpp +120 -0
- package/node_modules/better-sqlite3/src/objects/backup.hpp +36 -0
- package/node_modules/better-sqlite3/src/objects/database.cpp +417 -0
- package/node_modules/better-sqlite3/src/objects/database.hpp +103 -0
- package/node_modules/better-sqlite3/src/objects/statement-iterator.cpp +113 -0
- package/node_modules/better-sqlite3/src/objects/statement-iterator.hpp +50 -0
- package/node_modules/better-sqlite3/src/objects/statement.cpp +383 -0
- package/node_modules/better-sqlite3/src/objects/statement.hpp +58 -0
- package/node_modules/better-sqlite3/src/util/bind-map.cpp +73 -0
- package/node_modules/better-sqlite3/src/util/binder.cpp +193 -0
- package/node_modules/better-sqlite3/src/util/constants.cpp +172 -0
- package/node_modules/better-sqlite3/src/util/custom-aggregate.cpp +121 -0
- package/node_modules/better-sqlite3/src/util/custom-function.cpp +59 -0
- package/node_modules/better-sqlite3/src/util/custom-table.cpp +409 -0
- package/node_modules/better-sqlite3/src/util/data-converter.cpp +17 -0
- package/node_modules/better-sqlite3/src/util/data.cpp +194 -0
- package/node_modules/better-sqlite3/src/util/helpers.cpp +109 -0
- package/node_modules/better-sqlite3/src/util/macros.cpp +83 -0
- package/node_modules/better-sqlite3/src/util/query-macros.cpp +71 -0
- package/node_modules/better-sqlite3/src/util/row-builder.cpp +49 -0
- package/node_modules/bindings/LICENSE.md +22 -0
- package/node_modules/bindings/README.md +98 -0
- package/node_modules/bindings/bindings.js +221 -0
- package/node_modules/bindings/package.json +28 -0
- package/node_modules/bl/.travis.yml +17 -0
- package/node_modules/bl/BufferList.js +396 -0
- package/node_modules/bl/LICENSE.md +13 -0
- package/node_modules/bl/README.md +247 -0
- package/node_modules/bl/bl.js +84 -0
- package/node_modules/bl/package.json +37 -0
- package/node_modules/bl/test/convert.js +21 -0
- package/node_modules/bl/test/indexOf.js +492 -0
- package/node_modules/bl/test/isBufferList.js +32 -0
- package/node_modules/bl/test/test.js +869 -0
- package/node_modules/buffer/AUTHORS.md +70 -0
- package/node_modules/buffer/LICENSE +21 -0
- package/node_modules/buffer/README.md +410 -0
- package/node_modules/buffer/index.d.ts +186 -0
- package/node_modules/buffer/index.js +1817 -0
- package/node_modules/buffer/package.json +96 -0
- package/node_modules/decompress-response/index.d.ts +22 -0
- package/node_modules/decompress-response/index.js +58 -0
- package/node_modules/decompress-response/license +9 -0
- package/node_modules/decompress-response/node_modules/mimic-response/index.d.ts +17 -0
- package/node_modules/decompress-response/node_modules/mimic-response/index.js +77 -0
- package/node_modules/decompress-response/node_modules/mimic-response/license +9 -0
- package/node_modules/decompress-response/node_modules/mimic-response/package.json +42 -0
- package/node_modules/decompress-response/node_modules/mimic-response/readme.md +78 -0
- package/node_modules/decompress-response/package.json +56 -0
- package/node_modules/decompress-response/readme.md +48 -0
- package/node_modules/deep-extend/LICENSE +20 -0
- package/node_modules/deep-extend/README.md +91 -0
- package/node_modules/deep-extend/index.js +1 -0
- package/node_modules/deep-extend/lib/deep-extend.js +150 -0
- package/node_modules/deep-extend/package.json +62 -0
- package/node_modules/end-of-stream/LICENSE +21 -0
- package/node_modules/end-of-stream/README.md +54 -0
- package/node_modules/end-of-stream/index.js +96 -0
- package/node_modules/end-of-stream/package.json +37 -0
- package/node_modules/expand-template/.travis.yml +6 -0
- package/node_modules/expand-template/LICENSE +21 -0
- package/node_modules/expand-template/README.md +43 -0
- package/node_modules/expand-template/index.js +26 -0
- package/node_modules/expand-template/package.json +29 -0
- package/node_modules/expand-template/test.js +67 -0
- package/node_modules/file-uri-to-path/.npmignore +1 -0
- package/node_modules/file-uri-to-path/.travis.yml +30 -0
- package/node_modules/file-uri-to-path/History.md +21 -0
- package/node_modules/file-uri-to-path/LICENSE +20 -0
- package/node_modules/file-uri-to-path/README.md +74 -0
- package/node_modules/file-uri-to-path/index.d.ts +2 -0
- package/node_modules/file-uri-to-path/index.js +66 -0
- package/node_modules/file-uri-to-path/package.json +32 -0
- package/node_modules/file-uri-to-path/test/test.js +24 -0
- package/node_modules/file-uri-to-path/test/tests.json +13 -0
- package/node_modules/fs-constants/LICENSE +21 -0
- package/node_modules/fs-constants/README.md +26 -0
- package/node_modules/fs-constants/browser.js +1 -0
- package/node_modules/fs-constants/index.js +1 -0
- package/node_modules/fs-constants/package.json +19 -0
- package/node_modules/github-from-package/.travis.yml +4 -0
- package/node_modules/github-from-package/LICENSE +18 -0
- package/node_modules/github-from-package/example/package.json +8 -0
- package/node_modules/github-from-package/example/url.js +3 -0
- package/node_modules/github-from-package/index.js +17 -0
- package/node_modules/github-from-package/package.json +30 -0
- package/node_modules/github-from-package/readme.markdown +53 -0
- package/node_modules/github-from-package/test/a.json +8 -0
- package/node_modules/github-from-package/test/b.json +5 -0
- package/node_modules/github-from-package/test/c.json +5 -0
- package/node_modules/github-from-package/test/d.json +7 -0
- package/node_modules/github-from-package/test/e.json +5 -0
- package/node_modules/github-from-package/test/url.js +19 -0
- package/node_modules/ieee754/LICENSE +11 -0
- package/node_modules/ieee754/README.md +51 -0
- package/node_modules/ieee754/index.d.ts +10 -0
- package/node_modules/ieee754/index.js +85 -0
- package/node_modules/ieee754/package.json +52 -0
- package/node_modules/ini/LICENSE +15 -0
- package/node_modules/ini/README.md +102 -0
- package/node_modules/ini/ini.js +206 -0
- package/node_modules/ini/package.json +33 -0
- package/node_modules/minimist/.eslintrc +29 -0
- package/node_modules/minimist/.github/FUNDING.yml +12 -0
- package/node_modules/minimist/.nycrc +14 -0
- package/node_modules/minimist/CHANGELOG.md +298 -0
- package/node_modules/minimist/LICENSE +18 -0
- package/node_modules/minimist/README.md +121 -0
- package/node_modules/minimist/example/parse.js +4 -0
- package/node_modules/minimist/index.js +263 -0
- package/node_modules/minimist/package.json +75 -0
- package/node_modules/minimist/test/all_bool.js +34 -0
- package/node_modules/minimist/test/bool.js +177 -0
- package/node_modules/minimist/test/dash.js +43 -0
- package/node_modules/minimist/test/default_bool.js +37 -0
- package/node_modules/minimist/test/dotted.js +24 -0
- package/node_modules/minimist/test/kv_short.js +32 -0
- package/node_modules/minimist/test/long.js +33 -0
- package/node_modules/minimist/test/num.js +38 -0
- package/node_modules/minimist/test/parse.js +209 -0
- package/node_modules/minimist/test/parse_modified.js +11 -0
- package/node_modules/minimist/test/proto.js +64 -0
- package/node_modules/minimist/test/short.js +69 -0
- package/node_modules/minimist/test/stop_early.js +17 -0
- package/node_modules/minimist/test/unknown.js +104 -0
- package/node_modules/minimist/test/whitespace.js +10 -0
- package/node_modules/mkdirp-classic/LICENSE +21 -0
- package/node_modules/mkdirp-classic/README.md +18 -0
- package/node_modules/mkdirp-classic/index.js +98 -0
- package/node_modules/mkdirp-classic/package.json +18 -0
- package/node_modules/moe-training/DEPLOY_CENTRAL_COMMAND.md +413 -0
- package/node_modules/moe-training/client/consent.js +96 -0
- package/node_modules/moe-training/client/envelope-builder.js +56 -0
- package/node_modules/moe-training/client/index.js +10 -0
- package/node_modules/moe-training/client/parsers/claude-code.js +108 -0
- package/node_modules/moe-training/client/parsers/codex.js +80 -0
- package/node_modules/moe-training/client/parsers/gemini.js +80 -0
- package/node_modules/moe-training/client/parsers/grok.js +16 -0
- package/node_modules/moe-training/client/parsers/index.js +20 -0
- package/node_modules/moe-training/client/scrubber.js +128 -0
- package/node_modules/moe-training/client/session-attestation.js +115 -0
- package/node_modules/moe-training/client/step-classifier.js +68 -0
- package/node_modules/moe-training/client/trajectory-capture.js +307 -0
- package/node_modules/moe-training/client/transmission-queue.js +104 -0
- package/node_modules/moe-training/package.json +21 -0
- package/node_modules/moe-training/server/enrichment.js +24 -0
- package/node_modules/moe-training/server/index.js +119 -0
- package/node_modules/moe-training/server/ledger.js +110 -0
- package/node_modules/moe-training/server/routes/ingest.js +96 -0
- package/node_modules/moe-training/server/routes/sessions.js +43 -0
- package/node_modules/moe-training/server/routes/stats.js +31 -0
- package/node_modules/moe-training/server/scoring.js +63 -0
- package/node_modules/moe-training/server/session-registry.js +156 -0
- package/node_modules/moe-training/server/stats.js +129 -0
- package/node_modules/moe-training/server/stitcher.js +69 -0
- package/node_modules/moe-training/server/storage.js +147 -0
- package/node_modules/moe-training/server/verifier.js +102 -0
- package/node_modules/moe-training/shared/constants.js +30 -0
- package/node_modules/moe-training/shared/crypto.js +45 -0
- package/node_modules/moe-training/shared/envelope-schema.js +220 -0
- package/node_modules/moe-training/test/client/consent.test.js +121 -0
- package/node_modules/moe-training/test/client/envelope-builder.test.js +107 -0
- package/node_modules/moe-training/test/client/parsers/claude-code.test.js +119 -0
- package/node_modules/moe-training/test/client/parsers/codex.test.js +83 -0
- package/node_modules/moe-training/test/client/parsers/gemini.test.js +99 -0
- package/node_modules/moe-training/test/client/scrubber.test.js +143 -0
- package/node_modules/moe-training/test/client/session-attestation-security.test.js +95 -0
- package/node_modules/moe-training/test/client/step-classifier.test.js +135 -0
- package/node_modules/moe-training/test/client/transmission-queue.test.js +33 -0
- package/node_modules/moe-training/test/integration/handshake.test.js +260 -0
- package/node_modules/moe-training/test/server/ingest-security.test.js +166 -0
- package/node_modules/moe-training/test/server/ledger.test.js +131 -0
- package/node_modules/moe-training/test/server/scoring.test.js +242 -0
- package/node_modules/moe-training/test/server/session-registry.test.js +125 -0
- package/node_modules/moe-training/test/server/stitcher.test.js +157 -0
- package/node_modules/moe-training/test/server/verifier.test.js +232 -0
- package/node_modules/moe-training/test/shared/crypto.test.js +87 -0
- package/node_modules/moe-training/test/shared/envelope-schema.test.js +351 -0
- package/node_modules/napi-build-utils/.github/workflows/run-npm-tests.yml +31 -0
- package/node_modules/napi-build-utils/LICENSE +21 -0
- package/node_modules/napi-build-utils/README.md +52 -0
- package/node_modules/napi-build-utils/index.js +214 -0
- package/node_modules/napi-build-utils/index.md +0 -0
- package/node_modules/napi-build-utils/package.json +42 -0
- package/node_modules/prebuild-install/CHANGELOG.md +131 -0
- package/node_modules/prebuild-install/CONTRIBUTING.md +6 -0
- package/node_modules/prebuild-install/LICENSE +21 -0
- package/node_modules/prebuild-install/README.md +163 -0
- package/node_modules/prebuild-install/asset.js +44 -0
- package/node_modules/prebuild-install/bin.js +78 -0
- package/node_modules/prebuild-install/download.js +142 -0
- package/node_modules/prebuild-install/error.js +14 -0
- package/node_modules/prebuild-install/help.txt +16 -0
- package/node_modules/prebuild-install/index.js +1 -0
- package/node_modules/prebuild-install/log.js +33 -0
- package/node_modules/prebuild-install/node_modules/node-abi/LICENSE +21 -0
- package/node_modules/prebuild-install/node_modules/node-abi/README.md +54 -0
- package/node_modules/prebuild-install/node_modules/node-abi/abi_registry.json +439 -0
- package/node_modules/prebuild-install/node_modules/node-abi/index.js +179 -0
- package/node_modules/prebuild-install/node_modules/node-abi/package.json +45 -0
- package/node_modules/prebuild-install/node_modules/semver/LICENSE +15 -0
- package/node_modules/prebuild-install/node_modules/semver/README.md +665 -0
- package/node_modules/prebuild-install/node_modules/semver/bin/semver.js +191 -0
- package/node_modules/prebuild-install/node_modules/semver/classes/comparator.js +143 -0
- package/node_modules/prebuild-install/node_modules/semver/classes/index.js +7 -0
- package/node_modules/prebuild-install/node_modules/semver/classes/range.js +557 -0
- package/node_modules/prebuild-install/node_modules/semver/classes/semver.js +333 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/clean.js +8 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/cmp.js +54 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/coerce.js +62 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/compare-build.js +9 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/compare-loose.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/compare.js +7 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/diff.js +60 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/eq.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/gt.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/gte.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/inc.js +21 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/lt.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/lte.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/major.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/minor.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/neq.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/parse.js +18 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/patch.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/prerelease.js +8 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/rcompare.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/rsort.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/satisfies.js +12 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/sort.js +5 -0
- package/node_modules/prebuild-install/node_modules/semver/functions/valid.js +8 -0
- package/node_modules/prebuild-install/node_modules/semver/index.js +91 -0
- package/node_modules/prebuild-install/node_modules/semver/internal/constants.js +37 -0
- package/node_modules/prebuild-install/node_modules/semver/internal/debug.js +11 -0
- package/node_modules/prebuild-install/node_modules/semver/internal/identifiers.js +29 -0
- package/node_modules/prebuild-install/node_modules/semver/internal/lrucache.js +42 -0
- package/node_modules/prebuild-install/node_modules/semver/internal/parse-options.js +17 -0
- package/node_modules/prebuild-install/node_modules/semver/internal/re.js +223 -0
- package/node_modules/prebuild-install/node_modules/semver/package.json +78 -0
- package/node_modules/prebuild-install/node_modules/semver/preload.js +4 -0
- package/node_modules/prebuild-install/node_modules/semver/range.bnf +16 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/gtr.js +6 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/intersects.js +9 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/ltr.js +6 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/max-satisfying.js +27 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/min-satisfying.js +26 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/min-version.js +63 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/outside.js +82 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/simplify.js +49 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/subset.js +249 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/to-comparators.js +10 -0
- package/node_modules/prebuild-install/node_modules/semver/ranges/valid.js +13 -0
- package/node_modules/prebuild-install/package.json +67 -0
- package/node_modules/prebuild-install/proxy.js +35 -0
- package/node_modules/prebuild-install/rc.js +64 -0
- package/node_modules/prebuild-install/util.js +143 -0
- package/node_modules/pump/.github/FUNDING.yml +2 -0
- package/node_modules/pump/.travis.yml +5 -0
- package/node_modules/pump/LICENSE +21 -0
- package/node_modules/pump/README.md +74 -0
- package/node_modules/pump/SECURITY.md +5 -0
- package/node_modules/pump/empty.js +1 -0
- package/node_modules/pump/index.js +86 -0
- package/node_modules/pump/package.json +30 -0
- package/node_modules/pump/test-browser.js +66 -0
- package/node_modules/pump/test-node.js +53 -0
- package/node_modules/rc/LICENSE.APACHE2 +15 -0
- package/node_modules/rc/LICENSE.BSD +26 -0
- package/node_modules/rc/LICENSE.MIT +24 -0
- package/node_modules/rc/README.md +227 -0
- package/node_modules/rc/browser.js +7 -0
- package/node_modules/rc/cli.js +4 -0
- package/node_modules/rc/index.js +53 -0
- package/node_modules/rc/lib/utils.js +104 -0
- package/node_modules/rc/package.json +29 -0
- package/node_modules/rc/test/ini.js +16 -0
- package/node_modules/rc/test/nested-env-vars.js +50 -0
- package/node_modules/rc/test/test.js +59 -0
- package/node_modules/readable-stream/CONTRIBUTING.md +38 -0
- package/node_modules/readable-stream/GOVERNANCE.md +136 -0
- package/node_modules/readable-stream/LICENSE +47 -0
- package/node_modules/readable-stream/README.md +106 -0
- package/node_modules/readable-stream/errors-browser.js +127 -0
- package/node_modules/readable-stream/errors.js +116 -0
- package/node_modules/readable-stream/experimentalWarning.js +17 -0
- package/node_modules/readable-stream/lib/_stream_duplex.js +126 -0
- package/node_modules/readable-stream/lib/_stream_passthrough.js +37 -0
- package/node_modules/readable-stream/lib/_stream_readable.js +1027 -0
- package/node_modules/readable-stream/lib/_stream_transform.js +190 -0
- package/node_modules/readable-stream/lib/_stream_writable.js +641 -0
- package/node_modules/readable-stream/lib/internal/streams/async_iterator.js +180 -0
- package/node_modules/readable-stream/lib/internal/streams/buffer_list.js +183 -0
- package/node_modules/readable-stream/lib/internal/streams/destroy.js +96 -0
- package/node_modules/readable-stream/lib/internal/streams/end-of-stream.js +86 -0
- package/node_modules/readable-stream/lib/internal/streams/from-browser.js +3 -0
- package/node_modules/readable-stream/lib/internal/streams/from.js +52 -0
- package/node_modules/readable-stream/lib/internal/streams/pipeline.js +86 -0
- package/node_modules/readable-stream/lib/internal/streams/state.js +22 -0
- package/node_modules/readable-stream/lib/internal/streams/stream-browser.js +1 -0
- package/node_modules/readable-stream/lib/internal/streams/stream.js +1 -0
- package/node_modules/readable-stream/package.json +68 -0
- package/node_modules/readable-stream/readable-browser.js +9 -0
- package/node_modules/readable-stream/readable.js +16 -0
- package/node_modules/simple-concat/.travis.yml +3 -0
- package/node_modules/simple-concat/LICENSE +20 -0
- package/node_modules/simple-concat/README.md +44 -0
- package/node_modules/simple-concat/index.js +15 -0
- package/node_modules/simple-concat/package.json +47 -0
- package/node_modules/simple-concat/test/basic.js +41 -0
- package/node_modules/simple-get/.github/dependabot.yml +15 -0
- package/node_modules/simple-get/.github/workflows/ci.yml +23 -0
- package/node_modules/simple-get/LICENSE +20 -0
- package/node_modules/simple-get/README.md +333 -0
- package/node_modules/simple-get/index.js +108 -0
- package/node_modules/simple-get/package.json +67 -0
- package/node_modules/string_decoder/LICENSE +48 -0
- package/node_modules/string_decoder/README.md +47 -0
- package/node_modules/string_decoder/lib/string_decoder.js +296 -0
- package/node_modules/string_decoder/package.json +34 -0
- package/node_modules/strip-json-comments/index.js +70 -0
- package/node_modules/strip-json-comments/license +21 -0
- package/node_modules/strip-json-comments/package.json +42 -0
- package/node_modules/strip-json-comments/readme.md +64 -0
- package/node_modules/tar-fs/.travis.yml +6 -0
- package/node_modules/tar-fs/LICENSE +21 -0
- package/node_modules/tar-fs/README.md +165 -0
- package/node_modules/tar-fs/index.js +363 -0
- package/node_modules/tar-fs/node_modules/chownr/LICENSE +15 -0
- package/node_modules/tar-fs/node_modules/chownr/README.md +3 -0
- package/node_modules/tar-fs/node_modules/chownr/chownr.js +167 -0
- package/node_modules/tar-fs/node_modules/chownr/package.json +29 -0
- package/node_modules/tar-fs/package.json +41 -0
- package/node_modules/tar-fs/test/fixtures/a/hello.txt +1 -0
- package/node_modules/tar-fs/test/fixtures/b/a/test.txt +1 -0
- package/node_modules/tar-fs/test/fixtures/d/file1 +0 -0
- package/node_modules/tar-fs/test/fixtures/d/file2 +0 -0
- package/node_modules/tar-fs/test/fixtures/d/sub-dir/file5 +0 -0
- package/node_modules/tar-fs/test/fixtures/d/sub-files/file3 +0 -0
- package/node_modules/tar-fs/test/fixtures/d/sub-files/file4 +0 -0
- package/node_modules/tar-fs/test/fixtures/e/directory/.ignore +0 -0
- package/node_modules/tar-fs/test/fixtures/e/file +0 -0
- package/node_modules/tar-fs/test/fixtures/invalid.tar +0 -0
- package/node_modules/tar-fs/test/index.js +346 -0
- package/node_modules/tar-stream/LICENSE +21 -0
- package/node_modules/tar-stream/README.md +168 -0
- package/node_modules/tar-stream/extract.js +257 -0
- package/node_modules/tar-stream/headers.js +295 -0
- package/node_modules/tar-stream/index.js +2 -0
- package/node_modules/tar-stream/pack.js +255 -0
- package/node_modules/tar-stream/package.json +58 -0
- package/node_modules/tar-stream/sandbox.js +11 -0
- package/node_modules/tunnel-agent/LICENSE +55 -0
- package/node_modules/tunnel-agent/README.md +4 -0
- package/node_modules/tunnel-agent/index.js +244 -0
- package/node_modules/tunnel-agent/package.json +22 -0
- package/node_modules/util-deprecate/History.md +16 -0
- package/node_modules/util-deprecate/LICENSE +24 -0
- package/node_modules/util-deprecate/README.md +53 -0
- package/node_modules/util-deprecate/browser.js +67 -0
- package/node_modules/util-deprecate/node.js +6 -0
- package/node_modules/util-deprecate/package.json +27 -0
- package/node_modules/uuid/CHANGELOG.md +274 -0
- package/node_modules/uuid/CONTRIBUTING.md +18 -0
- package/node_modules/uuid/LICENSE.md +9 -0
- package/node_modules/uuid/README.md +466 -0
- package/node_modules/uuid/dist/bin/uuid +2 -0
- package/node_modules/uuid/dist/commonjs-browser/index.js +79 -0
- package/node_modules/uuid/dist/commonjs-browser/md5.js +223 -0
- package/node_modules/uuid/dist/commonjs-browser/native.js +11 -0
- package/node_modules/uuid/dist/commonjs-browser/nil.js +8 -0
- package/node_modules/uuid/dist/commonjs-browser/parse.js +45 -0
- package/node_modules/uuid/dist/commonjs-browser/regex.js +8 -0
- package/node_modules/uuid/dist/commonjs-browser/rng.js +25 -0
- package/node_modules/uuid/dist/commonjs-browser/sha1.js +104 -0
- package/node_modules/uuid/dist/commonjs-browser/stringify.js +44 -0
- package/node_modules/uuid/dist/commonjs-browser/v1.js +107 -0
- package/node_modules/uuid/dist/commonjs-browser/v3.js +16 -0
- package/node_modules/uuid/dist/commonjs-browser/v35.js +80 -0
- package/node_modules/uuid/dist/commonjs-browser/v4.js +43 -0
- package/node_modules/uuid/dist/commonjs-browser/v5.js +16 -0
- package/node_modules/uuid/dist/commonjs-browser/validate.js +17 -0
- package/node_modules/uuid/dist/commonjs-browser/version.js +21 -0
- package/node_modules/uuid/dist/esm-browser/index.js +9 -0
- package/node_modules/uuid/dist/esm-browser/md5.js +215 -0
- package/node_modules/uuid/dist/esm-browser/native.js +4 -0
- package/node_modules/uuid/dist/esm-browser/nil.js +1 -0
- package/node_modules/uuid/dist/esm-browser/parse.js +35 -0
- package/node_modules/uuid/dist/esm-browser/regex.js +1 -0
- package/node_modules/uuid/dist/esm-browser/rng.js +18 -0
- package/node_modules/uuid/dist/esm-browser/sha1.js +96 -0
- package/node_modules/uuid/dist/esm-browser/stringify.js +33 -0
- package/node_modules/uuid/dist/esm-browser/v1.js +95 -0
- package/node_modules/uuid/dist/esm-browser/v3.js +4 -0
- package/node_modules/uuid/dist/esm-browser/v35.js +66 -0
- package/node_modules/uuid/dist/esm-browser/v4.js +29 -0
- package/node_modules/uuid/dist/esm-browser/v5.js +4 -0
- package/node_modules/uuid/dist/esm-browser/validate.js +7 -0
- package/node_modules/uuid/dist/esm-browser/version.js +11 -0
- package/node_modules/uuid/dist/esm-node/index.js +9 -0
- package/node_modules/uuid/dist/esm-node/md5.js +13 -0
- package/node_modules/uuid/dist/esm-node/native.js +4 -0
- package/node_modules/uuid/dist/esm-node/nil.js +1 -0
- package/node_modules/uuid/dist/esm-node/parse.js +35 -0
- package/node_modules/uuid/dist/esm-node/regex.js +1 -0
- package/node_modules/uuid/dist/esm-node/rng.js +12 -0
- package/node_modules/uuid/dist/esm-node/sha1.js +13 -0
- package/node_modules/uuid/dist/esm-node/stringify.js +33 -0
- package/node_modules/uuid/dist/esm-node/v1.js +95 -0
- package/node_modules/uuid/dist/esm-node/v3.js +4 -0
- package/node_modules/uuid/dist/esm-node/v35.js +66 -0
- package/node_modules/uuid/dist/esm-node/v4.js +29 -0
- package/node_modules/uuid/dist/esm-node/v5.js +4 -0
- package/node_modules/uuid/dist/esm-node/validate.js +7 -0
- package/node_modules/uuid/dist/esm-node/version.js +11 -0
- package/node_modules/uuid/dist/index.js +79 -0
- package/node_modules/uuid/dist/md5-browser.js +223 -0
- package/node_modules/uuid/dist/md5.js +23 -0
- package/node_modules/uuid/dist/native-browser.js +11 -0
- package/node_modules/uuid/dist/native.js +15 -0
- package/node_modules/uuid/dist/nil.js +8 -0
- package/node_modules/uuid/dist/parse.js +45 -0
- package/node_modules/uuid/dist/regex.js +8 -0
- package/node_modules/uuid/dist/rng-browser.js +25 -0
- package/node_modules/uuid/dist/rng.js +24 -0
- package/node_modules/uuid/dist/sha1-browser.js +104 -0
- package/node_modules/uuid/dist/sha1.js +23 -0
- package/node_modules/uuid/dist/stringify.js +44 -0
- package/node_modules/uuid/dist/uuid-bin.js +85 -0
- package/node_modules/uuid/dist/v1.js +107 -0
- package/node_modules/uuid/dist/v3.js +16 -0
- package/node_modules/uuid/dist/v35.js +80 -0
- package/node_modules/uuid/dist/v4.js +43 -0
- package/node_modules/uuid/dist/v5.js +16 -0
- package/node_modules/uuid/dist/validate.js +17 -0
- package/node_modules/uuid/dist/version.js +21 -0
- package/node_modules/uuid/package.json +135 -0
- package/node_modules/uuid/wrapper.mjs +10 -0
- package/package.json +5 -3
- package/packages/cli/package.json +1 -1
- package/packages/daemon/package.json +1 -1
- package/packages/daemon/src/api.js +39 -23
- package/packages/daemon/src/introducer.js +21 -6
- package/packages/daemon/src/process.js +20 -2
- package/packages/daemon/src/providers/gemini.js +27 -28
- package/packages/gui/dist/assets/{index-VB4_k5Pz.js → index-BgTyFy4f.js} +331 -331
- package/packages/gui/dist/assets/{index-Bo6AeNmM.css → index-QADLyUj5.css} +1 -1
- package/packages/gui/dist/index.html +2 -2
- package/packages/gui/package.json +1 -1
- package/packages/gui/src/components/agents/agent-file-tree.jsx +68 -16
- package/packages/gui/src/components/agents/agent-panel.jsx +2 -0
- package/packages/gui/src/components/agents/code-review.jsx +9 -1
- package/packages/gui/src/components/agents/workspace-mode.jsx +2 -2
- package/packages/gui/src/components/editor/code-editor.jsx +1 -1
- package/packages/gui/src/stores/groove.js +40 -26
- package/packages/gui/src/views/agents.jsx +14 -65
- 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
|
+
});
|