@wlfi-agent/cli 1.4.13 → 1.4.15

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 (289) hide show
  1. package/Cargo.lock +3968 -0
  2. package/Cargo.toml +50 -0
  3. package/README.md +426 -6
  4. package/crates/vault-cli-admin/Cargo.toml +26 -0
  5. package/crates/vault-cli-admin/src/io_utils.rs +500 -0
  6. package/crates/vault-cli-admin/src/main.rs +3990 -0
  7. package/crates/vault-cli-admin/src/shared_config.rs +624 -0
  8. package/crates/vault-cli-admin/src/tui/amounts.rs +180 -0
  9. package/crates/vault-cli-admin/src/tui/token_rpc.rs +250 -0
  10. package/crates/vault-cli-admin/src/tui/utils.rs +82 -0
  11. package/crates/vault-cli-admin/src/tui.rs +3410 -0
  12. package/crates/vault-cli-agent/Cargo.toml +24 -0
  13. package/crates/vault-cli-agent/src/io_utils.rs +576 -0
  14. package/crates/vault-cli-agent/src/main.rs +833 -0
  15. package/crates/vault-cli-daemon/Cargo.toml +28 -0
  16. package/crates/vault-cli-daemon/src/bin/wlfi-agent-system-keychain.rs +216 -0
  17. package/crates/vault-cli-daemon/src/main.rs +644 -0
  18. package/crates/vault-cli-daemon/src/relay_sync.rs +894 -0
  19. package/crates/vault-cli-daemon/tests/system_keychain_helper_acl.rs +167 -0
  20. package/crates/vault-daemon/Cargo.toml +32 -0
  21. package/crates/vault-daemon/src/daemon_parts/api_impl_and_utils.rs +1041 -0
  22. package/crates/vault-daemon/src/daemon_parts/core_helpers.rs +1256 -0
  23. package/crates/vault-daemon/src/daemon_parts/types_api_rpc.rs +622 -0
  24. package/crates/vault-daemon/src/lib.rs +54 -0
  25. package/crates/vault-daemon/src/persistence.rs +441 -0
  26. package/crates/vault-daemon/src/tests.rs +237 -0
  27. package/crates/vault-daemon/src/tests_parts/part1.rs +1224 -0
  28. package/crates/vault-daemon/src/tests_parts/part2.rs +1021 -0
  29. package/crates/vault-daemon/src/tests_parts/part3.rs +835 -0
  30. package/crates/vault-daemon/src/tests_parts/part4.rs +604 -0
  31. package/crates/vault-domain/Cargo.toml +20 -0
  32. package/crates/vault-domain/src/action.rs +849 -0
  33. package/crates/vault-domain/src/address.rs +51 -0
  34. package/crates/vault-domain/src/approval.rs +90 -0
  35. package/crates/vault-domain/src/constants.rs +4 -0
  36. package/crates/vault-domain/src/error.rs +54 -0
  37. package/crates/vault-domain/src/keys.rs +71 -0
  38. package/crates/vault-domain/src/lib.rs +42 -0
  39. package/crates/vault-domain/src/nonce.rs +102 -0
  40. package/crates/vault-domain/src/policy.rs +172 -0
  41. package/crates/vault-domain/src/request.rs +53 -0
  42. package/crates/vault-domain/src/scope.rs +24 -0
  43. package/crates/vault-domain/src/session.rs +50 -0
  44. package/crates/vault-domain/src/signature.rs +34 -0
  45. package/crates/vault-domain/src/tests.rs +651 -0
  46. package/crates/vault-domain/src/u128_as_decimal_string.rs +44 -0
  47. package/crates/vault-policy/Cargo.toml +17 -0
  48. package/crates/vault-policy/src/engine.rs +301 -0
  49. package/crates/vault-policy/src/error.rs +81 -0
  50. package/crates/vault-policy/src/lib.rs +17 -0
  51. package/crates/vault-policy/src/report.rs +34 -0
  52. package/crates/vault-policy/src/tests.rs +891 -0
  53. package/crates/vault-policy/src/tests_explain.rs +78 -0
  54. package/crates/vault-sdk-agent/Cargo.toml +21 -0
  55. package/crates/vault-sdk-agent/src/lib.rs +711 -0
  56. package/crates/vault-signer/Cargo.toml +25 -0
  57. package/crates/vault-signer/src/lib.rs +731 -0
  58. package/crates/vault-signer/tests/secure_enclave_acl.rs +54 -0
  59. package/crates/vault-transport-unix/Cargo.toml +24 -0
  60. package/crates/vault-transport-unix/src/lib.rs +1640 -0
  61. package/crates/vault-transport-xpc/Cargo.toml +25 -0
  62. package/crates/vault-transport-xpc/src/client_codec_api.rs +635 -0
  63. package/crates/vault-transport-xpc/src/lib.rs +680 -0
  64. package/crates/vault-transport-xpc/src/tests.rs +818 -0
  65. package/crates/vault-transport-xpc/tests/e2e_flow.rs +773 -0
  66. package/dist/cli.cjs +35088 -0
  67. package/dist/cli.cjs.map +1 -0
  68. package/package.json +45 -41
  69. package/packages/cache/.turbo/turbo-build.log +52 -0
  70. package/packages/cache/dist/chunk-2QFWMUXT.cjs +43 -0
  71. package/packages/cache/dist/chunk-2QFWMUXT.cjs.map +1 -0
  72. package/packages/cache/dist/chunk-4U63TZTQ.js +43 -0
  73. package/packages/cache/dist/chunk-4U63TZTQ.js.map +1 -0
  74. package/packages/cache/dist/chunk-ALQ6H7KG.cjs +404 -0
  75. package/packages/cache/dist/chunk-ALQ6H7KG.cjs.map +1 -0
  76. package/packages/cache/dist/chunk-FGJEEF5N.js +404 -0
  77. package/packages/cache/dist/chunk-FGJEEF5N.js.map +1 -0
  78. package/packages/cache/dist/chunk-UYNEHZHB.cjs +45 -0
  79. package/packages/cache/dist/chunk-UYNEHZHB.cjs.map +1 -0
  80. package/packages/cache/dist/chunk-VXVMPG3W.js +45 -0
  81. package/packages/cache/dist/chunk-VXVMPG3W.js.map +1 -0
  82. package/packages/cache/dist/client/index.cjs +11 -0
  83. package/packages/cache/dist/client/index.cjs.map +1 -0
  84. package/packages/cache/dist/client/index.d.cts +15 -0
  85. package/packages/cache/dist/client/index.d.ts +15 -0
  86. package/packages/cache/dist/client/index.js +11 -0
  87. package/packages/cache/dist/client/index.js.map +1 -0
  88. package/packages/cache/dist/errors/index.cjs +11 -0
  89. package/packages/cache/dist/errors/index.cjs.map +1 -0
  90. package/packages/cache/dist/errors/index.d.cts +26 -0
  91. package/packages/cache/dist/errors/index.d.ts +26 -0
  92. package/packages/cache/dist/errors/index.js +11 -0
  93. package/packages/cache/dist/errors/index.js.map +1 -0
  94. package/packages/cache/dist/index.cjs +29 -0
  95. package/packages/cache/dist/index.cjs.map +1 -0
  96. package/packages/cache/dist/index.d.cts +4 -0
  97. package/packages/cache/dist/index.d.ts +4 -0
  98. package/packages/cache/dist/index.js +29 -0
  99. package/packages/cache/dist/index.js.map +1 -0
  100. package/packages/cache/dist/service/index.cjs +15 -0
  101. package/packages/cache/dist/service/index.cjs.map +1 -0
  102. package/packages/cache/dist/service/index.d.cts +184 -0
  103. package/packages/cache/dist/service/index.d.ts +184 -0
  104. package/packages/cache/dist/service/index.js +15 -0
  105. package/packages/cache/dist/service/index.js.map +1 -0
  106. package/packages/cache/node_modules/.bin/jiti +17 -0
  107. package/packages/cache/node_modules/.bin/tsc +17 -0
  108. package/packages/cache/node_modules/.bin/tsserver +17 -0
  109. package/packages/cache/node_modules/.bin/tsup +17 -0
  110. package/packages/cache/node_modules/.bin/tsup-node +17 -0
  111. package/packages/cache/node_modules/.bin/tsx +17 -0
  112. package/packages/cache/node_modules/.bin/vitest +17 -0
  113. package/packages/cache/package.json +48 -0
  114. package/packages/cache/src/client/index.ts +56 -0
  115. package/packages/cache/src/errors/index.ts +53 -0
  116. package/packages/cache/src/index.ts +3 -0
  117. package/packages/cache/src/service/index.test.ts +263 -0
  118. package/packages/cache/src/service/index.ts +678 -0
  119. package/packages/cache/tsconfig.json +13 -0
  120. package/packages/cache/tsup.config.ts +13 -0
  121. package/packages/cache/vitest.config.ts +16 -0
  122. package/packages/config/.turbo/turbo-build.log +18 -0
  123. package/packages/config/dist/index.cjs +1037 -0
  124. package/packages/config/dist/index.cjs.map +1 -0
  125. package/packages/config/dist/index.d.ts +131 -0
  126. package/packages/config/node_modules/.bin/jiti +17 -0
  127. package/packages/config/node_modules/.bin/tsc +17 -0
  128. package/packages/config/node_modules/.bin/tsserver +17 -0
  129. package/packages/config/node_modules/.bin/tsup +17 -0
  130. package/packages/config/node_modules/.bin/tsup-node +17 -0
  131. package/packages/config/node_modules/.bin/tsx +17 -0
  132. package/packages/config/package.json +21 -0
  133. package/packages/config/src/index.js +1 -0
  134. package/packages/config/src/index.ts +1282 -0
  135. package/packages/config/tsconfig.json +4 -0
  136. package/packages/rpc/.turbo/turbo-build.log +32 -0
  137. package/packages/rpc/dist/_esm-BCLXDO2R.cjs +3660 -0
  138. package/packages/rpc/dist/_esm-BCLXDO2R.cjs.map +1 -0
  139. package/packages/rpc/dist/ccip-OWJLAW55.cjs +16 -0
  140. package/packages/rpc/dist/ccip-OWJLAW55.cjs.map +1 -0
  141. package/packages/rpc/dist/chunk-APQIFZ3B.cjs +6247 -0
  142. package/packages/rpc/dist/chunk-APQIFZ3B.cjs.map +1 -0
  143. package/packages/rpc/dist/chunk-CDO2GWRD.cjs +410 -0
  144. package/packages/rpc/dist/chunk-CDO2GWRD.cjs.map +1 -0
  145. package/packages/rpc/dist/chunk-QGTNTFJ7.cjs +2249 -0
  146. package/packages/rpc/dist/chunk-QGTNTFJ7.cjs.map +1 -0
  147. package/packages/rpc/dist/chunk-TZDTAHWR.cjs +44 -0
  148. package/packages/rpc/dist/chunk-TZDTAHWR.cjs.map +1 -0
  149. package/packages/rpc/dist/index.cjs +7342 -0
  150. package/packages/rpc/dist/index.cjs.map +1 -0
  151. package/packages/rpc/dist/index.d.ts +3857 -0
  152. package/packages/rpc/dist/secp256k1-WCNM675D.cjs +18 -0
  153. package/packages/rpc/dist/secp256k1-WCNM675D.cjs.map +1 -0
  154. package/packages/rpc/node_modules/.bin/jiti +17 -0
  155. package/packages/rpc/node_modules/.bin/tsc +17 -0
  156. package/packages/rpc/node_modules/.bin/tsserver +17 -0
  157. package/packages/rpc/node_modules/.bin/tsup +17 -0
  158. package/packages/rpc/node_modules/.bin/tsup-node +17 -0
  159. package/packages/rpc/node_modules/.bin/tsx +17 -0
  160. package/packages/rpc/package.json +25 -0
  161. package/packages/rpc/src/index.ts +206 -0
  162. package/packages/rpc/tsconfig.json +4 -0
  163. package/packages/typescript/base.json +36 -0
  164. package/packages/typescript/nextjs.json +17 -0
  165. package/packages/typescript/package.json +10 -0
  166. package/packages/ui/.turbo/turbo-build.log +44 -0
  167. package/packages/ui/dist/chunk-MOAFBKSA.js +11 -0
  168. package/packages/ui/dist/chunk-MOAFBKSA.js.map +1 -0
  169. package/packages/ui/dist/components/badge.d.ts +12 -0
  170. package/packages/ui/dist/components/badge.js +31 -0
  171. package/packages/ui/dist/components/badge.js.map +1 -0
  172. package/packages/ui/dist/components/button.d.ts +13 -0
  173. package/packages/ui/dist/components/button.js +40 -0
  174. package/packages/ui/dist/components/button.js.map +1 -0
  175. package/packages/ui/dist/components/card.d.ts +10 -0
  176. package/packages/ui/dist/components/card.js +39 -0
  177. package/packages/ui/dist/components/card.js.map +1 -0
  178. package/packages/ui/dist/components/input.d.ts +5 -0
  179. package/packages/ui/dist/components/input.js +28 -0
  180. package/packages/ui/dist/components/input.js.map +1 -0
  181. package/packages/ui/dist/components/label.d.ts +5 -0
  182. package/packages/ui/dist/components/label.js +13 -0
  183. package/packages/ui/dist/components/label.js.map +1 -0
  184. package/packages/ui/dist/components/separator.d.ts +5 -0
  185. package/packages/ui/dist/components/separator.js +13 -0
  186. package/packages/ui/dist/components/separator.js.map +1 -0
  187. package/packages/ui/dist/components/textarea.d.ts +5 -0
  188. package/packages/ui/dist/components/textarea.js +27 -0
  189. package/packages/ui/dist/components/textarea.js.map +1 -0
  190. package/packages/ui/dist/tailwind.d.ts +56 -0
  191. package/packages/ui/dist/tailwind.js +60 -0
  192. package/packages/ui/dist/tailwind.js.map +1 -0
  193. package/packages/ui/dist/utils/cn.d.ts +5 -0
  194. package/packages/ui/dist/utils/cn.js +7 -0
  195. package/packages/ui/dist/utils/cn.js.map +1 -0
  196. package/packages/ui/node_modules/.bin/jiti +17 -0
  197. package/packages/ui/node_modules/.bin/tsc +17 -0
  198. package/packages/ui/node_modules/.bin/tsserver +17 -0
  199. package/packages/ui/node_modules/.bin/tsup +17 -0
  200. package/packages/ui/node_modules/.bin/tsup-node +17 -0
  201. package/packages/ui/node_modules/.bin/tsx +17 -0
  202. package/packages/ui/package.json +69 -0
  203. package/packages/ui/src/components/badge.tsx +27 -0
  204. package/packages/ui/src/components/button.tsx +40 -0
  205. package/packages/ui/src/components/card.tsx +31 -0
  206. package/packages/ui/src/components/input.tsx +21 -0
  207. package/packages/ui/src/components/label.tsx +6 -0
  208. package/packages/ui/src/components/separator.tsx +6 -0
  209. package/packages/ui/src/components/textarea.tsx +20 -0
  210. package/packages/ui/src/globals.css +70 -0
  211. package/packages/ui/src/tailwind.ts +56 -0
  212. package/packages/ui/src/utils/cn.ts +6 -0
  213. package/packages/ui/tsconfig.json +20 -0
  214. package/packages/ui/tsup.config.ts +20 -0
  215. package/pnpm-workspace.yaml +4 -0
  216. package/scripts/install-rust-binaries.mjs +84 -0
  217. package/scripts/launchd/install-user-daemon.sh +358 -0
  218. package/scripts/launchd/run-vault-daemon.sh +5 -0
  219. package/scripts/launchd/run-wlfi-agent-daemon.sh +73 -0
  220. package/scripts/launchd/uninstall-user-daemon.sh +103 -0
  221. package/src/cli.ts +2121 -0
  222. package/src/lib/admin-guard.js +1 -0
  223. package/src/lib/admin-guard.ts +185 -0
  224. package/src/lib/admin-passthrough.ts +33 -0
  225. package/src/lib/admin-reset.ts +751 -0
  226. package/src/lib/admin-setup.ts +1612 -0
  227. package/src/lib/agent-auth-clear.js +1 -0
  228. package/src/lib/agent-auth-clear.ts +58 -0
  229. package/src/lib/agent-auth-forwarding.js +1 -0
  230. package/src/lib/agent-auth-forwarding.ts +149 -0
  231. package/src/lib/agent-auth-migrate.js +1 -0
  232. package/src/lib/agent-auth-migrate.ts +150 -0
  233. package/src/lib/agent-auth-revoke.ts +103 -0
  234. package/src/lib/agent-auth-rotate.ts +107 -0
  235. package/src/lib/agent-auth-token.js +1 -0
  236. package/src/lib/agent-auth-token.ts +25 -0
  237. package/src/lib/agent-auth.ts +89 -0
  238. package/src/lib/asset-broadcast.js +1 -0
  239. package/src/lib/asset-broadcast.ts +285 -0
  240. package/src/lib/bootstrap-artifacts.js +1 -0
  241. package/src/lib/bootstrap-artifacts.ts +205 -0
  242. package/src/lib/bootstrap-credentials.js +1 -0
  243. package/src/lib/bootstrap-credentials.ts +832 -0
  244. package/src/lib/config-amounts.js +1 -0
  245. package/src/lib/config-amounts.ts +189 -0
  246. package/src/lib/config-mutation.ts +27 -0
  247. package/src/lib/fs-trust.js +1 -0
  248. package/src/lib/fs-trust.ts +537 -0
  249. package/src/lib/keychain.js +1 -0
  250. package/src/lib/keychain.ts +225 -0
  251. package/src/lib/local-admin-access.ts +106 -0
  252. package/src/lib/network-selection.js +1 -0
  253. package/src/lib/network-selection.ts +71 -0
  254. package/src/lib/passthrough-security.js +1 -0
  255. package/src/lib/passthrough-security.ts +114 -0
  256. package/src/lib/rpc-guard.js +1 -0
  257. package/src/lib/rpc-guard.ts +7 -0
  258. package/src/lib/rust-spawn-options.js +1 -0
  259. package/src/lib/rust-spawn-options.ts +98 -0
  260. package/src/lib/rust.js +1 -0
  261. package/src/lib/rust.ts +143 -0
  262. package/src/lib/signed-tx.js +1 -0
  263. package/src/lib/signed-tx.ts +116 -0
  264. package/src/lib/status-repair-cli.ts +116 -0
  265. package/src/lib/sudo.js +1 -0
  266. package/src/lib/sudo.ts +172 -0
  267. package/src/lib/vault-password-forwarding.js +1 -0
  268. package/src/lib/vault-password-forwarding.ts +155 -0
  269. package/src/lib/wallet-profile.js +1 -0
  270. package/src/lib/wallet-profile.ts +332 -0
  271. package/src/lib/wallet-repair.js +1 -0
  272. package/src/lib/wallet-repair.ts +304 -0
  273. package/src/lib/wallet-setup.js +1 -0
  274. package/src/lib/wallet-setup.ts +1466 -0
  275. package/src/lib/wallet-status.js +1 -0
  276. package/src/lib/wallet-status.ts +640 -0
  277. package/tsconfig.base.json +17 -0
  278. package/tsconfig.json +10 -0
  279. package/tsup.config.ts +25 -0
  280. package/turbo.json +41 -0
  281. package/LICENSE.md +0 -1
  282. package/dist/wlfa/index.cjs +0 -250
  283. package/dist/wlfa/index.d.cts +0 -1
  284. package/dist/wlfa/index.d.ts +0 -1
  285. package/dist/wlfa/index.js +0 -250
  286. package/dist/wlfc/index.cjs +0 -1839
  287. package/dist/wlfc/index.d.cts +0 -1
  288. package/dist/wlfc/index.d.ts +0 -1
  289. package/dist/wlfc/index.js +0 -1839
@@ -0,0 +1,15 @@
1
+ import {
2
+ RelayCacheService,
3
+ createRelayCacheService,
4
+ relayApprovalStatuses,
5
+ relayUpdateStatuses
6
+ } from "../chunk-FGJEEF5N.js";
7
+ import "../chunk-VXVMPG3W.js";
8
+ import "../chunk-4U63TZTQ.js";
9
+ export {
10
+ RelayCacheService,
11
+ createRelayCacheService,
12
+ relayApprovalStatuses,
13
+ relayUpdateStatuses
14
+ };
15
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
@@ -0,0 +1,17 @@
1
+ #!/bin/sh
2
+ basedir=$(dirname "$(echo "$0" | sed -e 's,\\,/,g')")
3
+
4
+ case `uname` in
5
+ *CYGWIN*) basedir=`cygpath -w "$basedir"`;;
6
+ esac
7
+
8
+ if [ -z "$NODE_PATH" ]; then
9
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/jiti@1.21.7/node_modules/jiti/bin/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/jiti@1.21.7/node_modules/jiti/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/jiti@1.21.7/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules"
10
+ else
11
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/jiti@1.21.7/node_modules/jiti/bin/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/jiti@1.21.7/node_modules/jiti/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/jiti@1.21.7/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules:$NODE_PATH"
12
+ fi
13
+ if [ -x "$basedir/node" ]; then
14
+ exec "$basedir/node" "$basedir/../../../../node_modules/.pnpm/jiti@1.21.7/node_modules/jiti/bin/jiti.js" "$@"
15
+ else
16
+ exec node "$basedir/../../../../node_modules/.pnpm/jiti@1.21.7/node_modules/jiti/bin/jiti.js" "$@"
17
+ fi
@@ -0,0 +1,17 @@
1
+ #!/bin/sh
2
+ basedir=$(dirname "$(echo "$0" | sed -e 's,\\,/,g')")
3
+
4
+ case `uname` in
5
+ *CYGWIN*) basedir=`cygpath -w "$basedir"`;;
6
+ esac
7
+
8
+ if [ -z "$NODE_PATH" ]; then
9
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/bin/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules"
10
+ else
11
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/bin/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules:$NODE_PATH"
12
+ fi
13
+ if [ -x "$basedir/node" ]; then
14
+ exec "$basedir/node" "$basedir/../../../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/bin/tsc" "$@"
15
+ else
16
+ exec node "$basedir/../../../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/bin/tsc" "$@"
17
+ fi
@@ -0,0 +1,17 @@
1
+ #!/bin/sh
2
+ basedir=$(dirname "$(echo "$0" | sed -e 's,\\,/,g')")
3
+
4
+ case `uname` in
5
+ *CYGWIN*) basedir=`cygpath -w "$basedir"`;;
6
+ esac
7
+
8
+ if [ -z "$NODE_PATH" ]; then
9
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/bin/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules"
10
+ else
11
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/bin/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/typescript@5.9.3/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules:$NODE_PATH"
12
+ fi
13
+ if [ -x "$basedir/node" ]; then
14
+ exec "$basedir/node" "$basedir/../../../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/bin/tsserver" "$@"
15
+ else
16
+ exec node "$basedir/../../../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/bin/tsserver" "$@"
17
+ fi
@@ -0,0 +1,17 @@
1
+ #!/bin/sh
2
+ basedir=$(dirname "$(echo "$0" | sed -e 's,\\,/,g')")
3
+
4
+ case `uname` in
5
+ *CYGWIN*) basedir=`cygpath -w "$basedir"`;;
6
+ esac
7
+
8
+ if [ -z "$NODE_PATH" ]; then
9
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/dist/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules"
10
+ else
11
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/dist/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules:$NODE_PATH"
12
+ fi
13
+ if [ -x "$basedir/node" ]; then
14
+ exec "$basedir/node" "$basedir/../../../../node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/dist/cli-default.js" "$@"
15
+ else
16
+ exec node "$basedir/../../../../node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/dist/cli-default.js" "$@"
17
+ fi
@@ -0,0 +1,17 @@
1
+ #!/bin/sh
2
+ basedir=$(dirname "$(echo "$0" | sed -e 's,\\,/,g')")
3
+
4
+ case `uname` in
5
+ *CYGWIN*) basedir=`cygpath -w "$basedir"`;;
6
+ esac
7
+
8
+ if [ -z "$NODE_PATH" ]; then
9
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/dist/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules"
10
+ else
11
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/dist/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules:$NODE_PATH"
12
+ fi
13
+ if [ -x "$basedir/node" ]; then
14
+ exec "$basedir/node" "$basedir/../../../../node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/dist/cli-node.js" "$@"
15
+ else
16
+ exec node "$basedir/../../../../node_modules/.pnpm/tsup@8.5.1_jiti@1.21.7_postcss@8.5.8_tsx@4.21.0_typescript@5.9.3/node_modules/tsup/dist/cli-node.js" "$@"
17
+ fi
@@ -0,0 +1,17 @@
1
+ #!/bin/sh
2
+ basedir=$(dirname "$(echo "$0" | sed -e 's,\\,/,g')")
3
+
4
+ case `uname` in
5
+ *CYGWIN*) basedir=`cygpath -w "$basedir"`;;
6
+ esac
7
+
8
+ if [ -z "$NODE_PATH" ]; then
9
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsx@4.21.0/node_modules/tsx/dist/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsx@4.21.0/node_modules/tsx/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsx@4.21.0/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules"
10
+ else
11
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsx@4.21.0/node_modules/tsx/dist/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsx@4.21.0/node_modules/tsx/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/tsx@4.21.0/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules:$NODE_PATH"
12
+ fi
13
+ if [ -x "$basedir/node" ]; then
14
+ exec "$basedir/node" "$basedir/../../../../node_modules/.pnpm/tsx@4.21.0/node_modules/tsx/dist/cli.mjs" "$@"
15
+ else
16
+ exec node "$basedir/../../../../node_modules/.pnpm/tsx@4.21.0/node_modules/tsx/dist/cli.mjs" "$@"
17
+ fi
@@ -0,0 +1,17 @@
1
+ #!/bin/sh
2
+ basedir=$(dirname "$(echo "$0" | sed -e 's,\\,/,g')")
3
+
4
+ case `uname` in
5
+ *CYGWIN*) basedir=`cygpath -w "$basedir"`;;
6
+ esac
7
+
8
+ if [ -z "$NODE_PATH" ]; then
9
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/vitest@3.2.4_@types+node@24.12.0_jiti@1.21.7_tsx@4.21.0/node_modules/vitest/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/vitest@3.2.4_@types+node@24.12.0_jiti@1.21.7_tsx@4.21.0/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules"
10
+ else
11
+ export NODE_PATH="/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/vitest@3.2.4_@types+node@24.12.0_jiti@1.21.7_tsx@4.21.0/node_modules/vitest/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/vitest@3.2.4_@types+node@24.12.0_jiti@1.21.7_tsx@4.21.0/node_modules:/Users/hanzhi/Documents/WLFI/wlfi-agent-sdk/node_modules/.pnpm/node_modules:$NODE_PATH"
12
+ fi
13
+ if [ -x "$basedir/node" ]; then
14
+ exec "$basedir/node" "$basedir/../../../../node_modules/.pnpm/vitest@3.2.4_@types+node@24.12.0_jiti@1.21.7_tsx@4.21.0/node_modules/vitest/vitest.mjs" "$@"
15
+ else
16
+ exec node "$basedir/../../../../node_modules/.pnpm/vitest@3.2.4_@types+node@24.12.0_jiti@1.21.7_tsx@4.21.0/node_modules/vitest/vitest.mjs" "$@"
17
+ fi
@@ -0,0 +1,48 @@
1
+ {
2
+ "name": "@wlfi-agent/cache",
3
+ "private": true,
4
+ "version": "0.0.0",
5
+ "type": "module",
6
+ "exports": {
7
+ ".": {
8
+ "types": "./src/index.ts",
9
+ "import": "./dist/index.js",
10
+ "require": "./dist/index.cjs"
11
+ },
12
+ "./client": {
13
+ "types": "./src/client/index.ts",
14
+ "import": "./dist/client/index.js",
15
+ "require": "./dist/client/index.cjs"
16
+ },
17
+ "./errors": {
18
+ "types": "./src/errors/index.ts",
19
+ "import": "./dist/errors/index.js",
20
+ "require": "./dist/errors/index.cjs"
21
+ },
22
+ "./service": {
23
+ "types": "./src/service/index.ts",
24
+ "import": "./dist/service/index.js",
25
+ "require": "./dist/service/index.cjs"
26
+ }
27
+ },
28
+ "scripts": {
29
+ "build": "tsup",
30
+ "clean": "rm -rf .turbo node_modules dist coverage junit",
31
+ "dev": "tsup --watch",
32
+ "lint": "biome check .",
33
+ "lint:fix": "biome check --write .",
34
+ "test:unit": "vitest run",
35
+ "typecheck": "tsc --noEmit",
36
+ "check": "pnpm run typecheck"
37
+ },
38
+ "types": "./src/index.ts",
39
+ "dependencies": {
40
+ "ioredis": "^5.7.0"
41
+ },
42
+ "devDependencies": {
43
+ "@types/node": "^24.0.0",
44
+ "tsup": "^8.5.1",
45
+ "typescript": "^5.9.0",
46
+ "vitest": "^3.2.4"
47
+ }
48
+ }
@@ -0,0 +1,56 @@
1
+ /** biome-ignore-all lint/style/noProcessEnv: relay cache config is environment driven */
2
+ import Redis from 'ioredis';
3
+
4
+ export interface CacheClientOptions {
5
+ connectTimeoutMs?: number;
6
+ enableOfflineQueue?: boolean;
7
+ keyPrefix?: string;
8
+ lazyConnect?: boolean;
9
+ maxRetriesPerRequest?: number | null;
10
+ url?: string;
11
+ }
12
+
13
+ let singletonClient: Redis | null = null;
14
+
15
+ const getDefaultCacheUrl = (): string => {
16
+ const explicitUrl = process.env.CACHE_URL?.trim();
17
+ if (explicitUrl) {
18
+ return explicitUrl;
19
+ }
20
+
21
+ const host = process.env.CACHE_HOST?.trim() || '127.0.0.1';
22
+ const port = Number(process.env.CACHE_PORT?.trim() || '6379');
23
+
24
+ return `redis://${host}:${port}`;
25
+ };
26
+
27
+ export const createCacheClient = (options: CacheClientOptions = {}): Redis => {
28
+ return new Redis(options.url ?? getDefaultCacheUrl(), {
29
+ connectTimeout: options.connectTimeoutMs ?? 5_000,
30
+ enableOfflineQueue: options.enableOfflineQueue ?? true,
31
+ keyPrefix: options.keyPrefix,
32
+ lazyConnect: options.lazyConnect ?? false,
33
+ maxRetriesPerRequest: options.maxRetriesPerRequest ?? 2,
34
+ reconnectOnError(error) {
35
+ return error.message.includes('READONLY') || error.message.includes('ETIMEDOUT');
36
+ },
37
+ });
38
+ };
39
+
40
+ export const getCacheClient = (options: CacheClientOptions = {}): Redis => {
41
+ if (!singletonClient) {
42
+ singletonClient = createCacheClient(options);
43
+ }
44
+
45
+ return singletonClient;
46
+ };
47
+
48
+ export const closeCacheClient = async (): Promise<void> => {
49
+ if (!singletonClient) {
50
+ return;
51
+ }
52
+
53
+ const client = singletonClient;
54
+ singletonClient = null;
55
+ await client.quit();
56
+ };
@@ -0,0 +1,53 @@
1
+ export const cacheErrorCodes = {
2
+ connectionFailed: 'CACHE_CONNECTION_FAILED',
3
+ invalidPayload: 'CACHE_INVALID_PAYLOAD',
4
+ notFound: 'CACHE_NOT_FOUND',
5
+ unknown: 'CACHE_UNKNOWN',
6
+ } as const;
7
+
8
+ export type CacheErrorCode = (typeof cacheErrorCodes)[keyof typeof cacheErrorCodes];
9
+
10
+ export class CacheError extends Error {
11
+ readonly code: CacheErrorCode;
12
+ readonly key?: string;
13
+ readonly operation?: string;
14
+ override readonly cause?: unknown;
15
+
16
+ constructor(options: {
17
+ cause?: unknown;
18
+ code: CacheErrorCode;
19
+ key?: string;
20
+ message: string;
21
+ operation?: string;
22
+ }) {
23
+ super(options.message);
24
+ this.name = 'CacheError';
25
+ this.code = options.code;
26
+ this.key = options.key;
27
+ this.operation = options.operation;
28
+ this.cause = options.cause;
29
+ }
30
+ }
31
+
32
+ export const toCacheError = (
33
+ error: unknown,
34
+ context: { key?: string; operation?: string } = {},
35
+ ): CacheError => {
36
+ if (error instanceof CacheError) {
37
+ return error;
38
+ }
39
+
40
+ const message = error instanceof Error ? error.message : 'Unknown cache error';
41
+ const normalizedMessage = message.toLowerCase();
42
+ const code = normalizedMessage.includes('connect')
43
+ ? cacheErrorCodes.connectionFailed
44
+ : cacheErrorCodes.unknown;
45
+
46
+ return new CacheError({
47
+ cause: error,
48
+ code,
49
+ key: context.key,
50
+ message,
51
+ operation: context.operation,
52
+ });
53
+ };
@@ -0,0 +1,3 @@
1
+ export * from './client/index.js';
2
+ export * from './errors/index.js';
3
+ export * from './service/index.js';
@@ -0,0 +1,263 @@
1
+ import { createHash } from 'node:crypto';
2
+ import { describe, expect, it } from 'vitest';
3
+ import { RelayCacheService } from './index.js';
4
+
5
+ class InMemoryCacheClient {
6
+ private readonly store = new Map<string, string>();
7
+ private readonly sets = new Map<string, Set<string>>();
8
+ private readonly zsets = new Map<string, Array<{ member: string; score: number }>>();
9
+
10
+ async del(key: string): Promise<number> {
11
+ const existed = this.store.delete(key);
12
+ this.sets.delete(key);
13
+ this.zsets.delete(key);
14
+ return existed ? 1 : 0;
15
+ }
16
+
17
+ async get(key: string): Promise<string | null> {
18
+ return this.store.get(key) ?? null;
19
+ }
20
+
21
+ async ping(): Promise<string> {
22
+ return 'PONG';
23
+ }
24
+
25
+ async quit(): Promise<string> {
26
+ return 'OK';
27
+ }
28
+
29
+ async sadd(key: string, ...members: string[]): Promise<number> {
30
+ const bucket = this.sets.get(key) ?? new Set<string>();
31
+ let added = 0;
32
+ for (const member of members) {
33
+ if (!bucket.has(member)) {
34
+ bucket.add(member);
35
+ added += 1;
36
+ }
37
+ }
38
+ this.sets.set(key, bucket);
39
+ return added;
40
+ }
41
+
42
+ async set(key: string, value: string, mode?: 'NX' | 'XX'): Promise<'OK' | null> {
43
+ if (mode === 'NX' && this.store.has(key)) {
44
+ return null;
45
+ }
46
+ if (mode === 'XX' && !this.store.has(key)) {
47
+ return null;
48
+ }
49
+ this.store.set(key, value);
50
+ return 'OK';
51
+ }
52
+
53
+ async smembers(key: string): Promise<string[]> {
54
+ return [...(this.sets.get(key) ?? new Set<string>())];
55
+ }
56
+
57
+ async zadd(key: string, ...args: (string | number)[]): Promise<number> {
58
+ const bucket = this.zsets.get(key) ?? [];
59
+ for (let index = 0; index < args.length; index += 2) {
60
+ bucket.push({ score: Number(args[index]), member: String(args[index + 1]) });
61
+ }
62
+ this.zsets.set(key, bucket);
63
+ return args.length / 2;
64
+ }
65
+
66
+ async zrange(key: string, start: number, stop: number, ...args: string[]): Promise<string[]> {
67
+ const bucket = [...(this.zsets.get(key) ?? [])].sort((left, right) => left.score - right.score);
68
+ const ordered = args.includes('REV') ? bucket.reverse() : bucket;
69
+ const normalizedStop = stop < 0 ? ordered.length + stop : stop;
70
+ return ordered.slice(start, normalizedStop + 1).map((entry) => entry.member);
71
+ }
72
+
73
+ async zrem(key: string, ...members: string[]): Promise<number> {
74
+ const bucket = this.zsets.get(key) ?? [];
75
+ const filtered = bucket.filter((entry) => !members.includes(entry.member));
76
+ this.zsets.set(key, filtered);
77
+ return bucket.length - filtered.length;
78
+ }
79
+ }
80
+
81
+ describe('RelayCacheService approval capability guards', () => {
82
+ it('consumes an approval capability only once', async () => {
83
+ const service = new RelayCacheService({
84
+ client: new InMemoryCacheClient() as never,
85
+ namespace: 'test:relay',
86
+ });
87
+
88
+ await expect(service.consumeApprovalCapability('approval-1', 'hash-1')).resolves.toBe(true);
89
+ await expect(service.consumeApprovalCapability('approval-1', 'hash-1')).resolves.toBe(false);
90
+ });
91
+
92
+ it('rate limits repeated invalid approval capability attempts', async () => {
93
+ const service = new RelayCacheService({
94
+ client: new InMemoryCacheClient() as never,
95
+ namespace: 'test:relay',
96
+ });
97
+
98
+ for (let attempt = 1; attempt < 5; attempt += 1) {
99
+ await expect(service.recordApprovalCapabilityFailure('approval-2')).resolves.toMatchObject({
100
+ attempts: attempt,
101
+ blocked: false,
102
+ blockedUntil: null,
103
+ });
104
+ }
105
+
106
+ await expect(service.recordApprovalCapabilityFailure('approval-2')).resolves.toMatchObject({
107
+ attempts: 5,
108
+ blocked: true,
109
+ });
110
+ });
111
+
112
+ it('rotates approval capabilities for pending approvals and clears prior failure state', async () => {
113
+ const service = new RelayCacheService({
114
+ client: new InMemoryCacheClient() as never,
115
+ namespace: 'test:relay',
116
+ });
117
+
118
+ const daemonId = '11'.repeat(32);
119
+ const originalToken = 'aa'.repeat(32);
120
+ const originalHash = createHash('sha256').update(originalToken).digest('hex');
121
+
122
+ await service.syncDaemonRegistration({
123
+ daemon: {
124
+ daemonId,
125
+ daemonPublicKey: '22'.repeat(32),
126
+ ethereumAddress: '0x3333333333333333333333333333333333333333',
127
+ lastSeenAt: new Date().toISOString(),
128
+ registeredAt: new Date().toISOString(),
129
+ status: 'active',
130
+ updatedAt: new Date().toISOString(),
131
+ },
132
+ approvalRequests: [
133
+ {
134
+ approvalRequestId: 'aaaaaaaa-aaaa-4aaa-8aaa-aaaaaaaaaaaa',
135
+ daemonId,
136
+ destination: '0x4444444444444444444444444444444444444444',
137
+ metadata: {
138
+ approvalCapabilityHash: originalHash,
139
+ approvalCapabilityToken: originalToken,
140
+ source: 'daemon',
141
+ },
142
+ requestedAt: new Date().toISOString(),
143
+ status: 'pending',
144
+ transactionType: 'transfer_native',
145
+ updatedAt: new Date().toISOString(),
146
+ },
147
+ ],
148
+ });
149
+
150
+ for (let attempt = 1; attempt <= 2; attempt += 1) {
151
+ await expect(
152
+ service.recordApprovalCapabilityFailure('aaaaaaaa-aaaa-4aaa-8aaa-aaaaaaaaaaaa'),
153
+ ).resolves.toMatchObject({
154
+ attempts: attempt,
155
+ blocked: false,
156
+ });
157
+ }
158
+
159
+ const rotated = await service.rotateApprovalCapability('aaaaaaaa-aaaa-4aaa-8aaa-aaaaaaaaaaaa');
160
+
161
+ expect(rotated.metadata?.source).toBe('daemon');
162
+ expect(rotated.metadata?.approvalCapabilityToken).toMatch(/^[a-f0-9]{64}$/u);
163
+ expect(rotated.metadata?.approvalCapabilityToken).not.toBe(originalToken);
164
+ expect(rotated.metadata?.approvalCapabilityHash).toMatch(/^[a-f0-9]{64}$/u);
165
+ expect(rotated.metadata?.approvalCapabilityHash).not.toBe(originalHash);
166
+ expect(rotated.metadata?.approvalCapabilityHash).toBe(
167
+ createHash('sha256')
168
+ .update(rotated.metadata?.approvalCapabilityToken ?? '')
169
+ .digest('hex'),
170
+ );
171
+
172
+ await expect(
173
+ service.recordApprovalCapabilityFailure('aaaaaaaa-aaaa-4aaa-8aaa-aaaaaaaaaaaa'),
174
+ ).resolves.toMatchObject({
175
+ attempts: 1,
176
+ blocked: false,
177
+ blockedUntil: null,
178
+ });
179
+ });
180
+
181
+ it('issues a secure approval capability for pending approvals that were synced without one', async () => {
182
+ const service = new RelayCacheService({
183
+ client: new InMemoryCacheClient() as never,
184
+ namespace: 'test:relay',
185
+ });
186
+
187
+ const daemonId = '55'.repeat(32);
188
+
189
+ await service.syncDaemonRegistration({
190
+ daemon: {
191
+ daemonId,
192
+ daemonPublicKey: '66'.repeat(32),
193
+ ethereumAddress: '0x7777777777777777777777777777777777777777',
194
+ lastSeenAt: new Date().toISOString(),
195
+ registeredAt: new Date().toISOString(),
196
+ status: 'active',
197
+ updatedAt: new Date().toISOString(),
198
+ },
199
+ approvalRequests: [
200
+ {
201
+ approvalRequestId: 'bbbbbbbb-bbbb-4bbb-8bbb-bbbbbbbbbbbb',
202
+ daemonId,
203
+ destination: '0x8888888888888888888888888888888888888888',
204
+ metadata: {
205
+ source: 'legacy-daemon',
206
+ },
207
+ requestedAt: new Date().toISOString(),
208
+ status: 'pending',
209
+ transactionType: 'transfer_native',
210
+ updatedAt: new Date().toISOString(),
211
+ },
212
+ ],
213
+ });
214
+
215
+ const rotated = await service.rotateApprovalCapability('bbbbbbbb-bbbb-4bbb-8bbb-bbbbbbbbbbbb');
216
+
217
+ expect(rotated.metadata).toMatchObject({
218
+ source: 'legacy-daemon',
219
+ });
220
+ expect(rotated.metadata?.approvalCapabilityToken).toMatch(/^[a-f0-9]{64}$/u);
221
+ expect(rotated.metadata?.approvalCapabilityHash).toBe(
222
+ createHash('sha256')
223
+ .update(rotated.metadata?.approvalCapabilityToken ?? '')
224
+ .digest('hex'),
225
+ );
226
+ });
227
+
228
+ it('rejects secure approval capability rotation for non-pending approvals', async () => {
229
+ const service = new RelayCacheService({
230
+ client: new InMemoryCacheClient() as never,
231
+ namespace: 'test:relay',
232
+ });
233
+
234
+ const daemonId = '99'.repeat(32);
235
+
236
+ await service.syncDaemonRegistration({
237
+ daemon: {
238
+ daemonId,
239
+ daemonPublicKey: 'aa'.repeat(32),
240
+ ethereumAddress: '0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb',
241
+ lastSeenAt: new Date().toISOString(),
242
+ registeredAt: new Date().toISOString(),
243
+ status: 'active',
244
+ updatedAt: new Date().toISOString(),
245
+ },
246
+ approvalRequests: [
247
+ {
248
+ approvalRequestId: 'cccccccc-cccc-4ccc-8ccc-cccccccccccc',
249
+ daemonId,
250
+ destination: '0xdddddddddddddddddddddddddddddddddddddddd',
251
+ requestedAt: new Date().toISOString(),
252
+ status: 'completed',
253
+ transactionType: 'transfer_native',
254
+ updatedAt: new Date().toISOString(),
255
+ },
256
+ ],
257
+ });
258
+
259
+ await expect(
260
+ service.rotateApprovalCapability('cccccccc-cccc-4ccc-8ccc-cccccccccccc'),
261
+ ).rejects.toThrow(/cannot accept a new secure approval link/);
262
+ });
263
+ });