@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,285 @@
1
+ import { setTimeout as sleep } from 'node:timers/promises';
2
+ import { encodeFunctionData, erc20Abi, type Address, type Hex, type TransactionReceipt } from 'viem';
3
+ import type { ResolvedAssetMetadata, RustAmountOutputShape } from './config-amounts.js';
4
+ import { normalizeAgentAmountOutput } from './config-amounts.js';
5
+
6
+ export interface AssetBroadcastPlanInput {
7
+ rpcUrl: string;
8
+ chainId: number;
9
+ from: Address;
10
+ to: Address;
11
+ valueWei: bigint;
12
+ dataHex: Hex;
13
+ nonce?: number;
14
+ gasLimit?: bigint;
15
+ maxFeePerGasWei?: bigint;
16
+ maxPriorityFeePerGasWei?: bigint;
17
+ txType: string;
18
+ }
19
+
20
+ export interface AssetBroadcastPlan {
21
+ rpcUrl: string;
22
+ chainId: number;
23
+ from: Address;
24
+ to: Address;
25
+ valueWei: bigint;
26
+ dataHex: Hex;
27
+ nonce: number;
28
+ gasLimit: bigint;
29
+ maxFeePerGasWei: bigint;
30
+ maxPriorityFeePerGasWei: bigint;
31
+ txType: string;
32
+ }
33
+
34
+ export interface SignedBroadcastResult {
35
+ raw_tx_hex?: string;
36
+ tx_hash_hex?: string;
37
+ r_hex?: string;
38
+ s_hex?: string;
39
+ v?: number;
40
+ }
41
+
42
+ export interface CompletedAssetBroadcast {
43
+ signedTxHash: string | null;
44
+ networkTxHash: Hex;
45
+ signedNonce: number;
46
+ }
47
+
48
+ export interface WaitForOnchainReceiptResult {
49
+ receipt: TransactionReceipt | null;
50
+ timedOut: boolean;
51
+ }
52
+
53
+ interface BroadcastFeeEstimate {
54
+ gasPrice: bigint | null;
55
+ maxFeePerGas: bigint | null;
56
+ maxPriorityFeePerGas: bigint | null;
57
+ }
58
+
59
+ export interface ResolveAssetBroadcastPlanDeps {
60
+ getChainInfo: (rpcUrl: string) => Promise<{ chainId: number }>;
61
+ assertRpcChainIdMatches: (expectedChainId: number, actualChainId: number) => void;
62
+ getNonce: (rpcUrl: string, address: Address) => Promise<number>;
63
+ estimateGas: (args: {
64
+ rpcUrl: string;
65
+ from: Address;
66
+ to: Address;
67
+ value?: bigint;
68
+ data?: Hex;
69
+ }) => Promise<bigint>;
70
+ estimateFees: (rpcUrl: string) => Promise<BroadcastFeeEstimate>;
71
+ }
72
+
73
+ export interface CompleteAssetBroadcastDeps {
74
+ assertSignedBroadcastTransactionMatchesRequest: (expected: {
75
+ rawTxHex: Hex;
76
+ from: Address;
77
+ to: Address;
78
+ chainId: number;
79
+ nonce: number;
80
+ allowHigherNonce?: boolean;
81
+ value: bigint;
82
+ data: Hex;
83
+ gasLimit: bigint;
84
+ maxFeePerGas: bigint;
85
+ maxPriorityFeePerGas: bigint;
86
+ txType: string;
87
+ }) => Promise<{ nonce: number }>;
88
+ broadcastRawTransaction: (rpcUrl: string, rawTransaction: Hex) => Promise<Hex>;
89
+ }
90
+
91
+ export interface WaitForOnchainReceiptDeps {
92
+ getTransactionReceiptByHash: (rpcUrl: string, hash: Hex) => Promise<TransactionReceipt>;
93
+ now?: () => number;
94
+ sleep?: (ms: number) => Promise<void>;
95
+ }
96
+
97
+ export function encodeErc20TransferData(recipient: Address, amountWei: bigint): Hex {
98
+ return encodeFunctionData({
99
+ abi: erc20Abi,
100
+ functionName: 'transfer',
101
+ args: [recipient, amountWei],
102
+ });
103
+ }
104
+
105
+ export function encodeErc20ApproveData(spender: Address, amountWei: bigint): Hex {
106
+ return encodeFunctionData({
107
+ abi: erc20Abi,
108
+ functionName: 'approve',
109
+ args: [spender, amountWei],
110
+ });
111
+ }
112
+
113
+ export async function resolveAssetBroadcastPlan(
114
+ input: AssetBroadcastPlanInput,
115
+ deps: ResolveAssetBroadcastPlanDeps,
116
+ ): Promise<AssetBroadcastPlan> {
117
+ const chainInfo = await deps.getChainInfo(input.rpcUrl);
118
+ deps.assertRpcChainIdMatches(input.chainId, chainInfo.chainId);
119
+
120
+ const nonce = input.nonce ?? await deps.getNonce(input.rpcUrl, input.from);
121
+ const gasLimit = input.gasLimit ?? await deps.estimateGas({
122
+ rpcUrl: input.rpcUrl,
123
+ from: input.from,
124
+ to: input.to,
125
+ value: input.valueWei,
126
+ data: input.dataHex,
127
+ });
128
+ const fees = await deps.estimateFees(input.rpcUrl);
129
+ const resolvedFees = fees as BroadcastFeeEstimate;
130
+ const maxFeePerGasWei = input.maxFeePerGasWei ?? (resolvedFees.maxFeePerGas ?? resolvedFees.gasPrice);
131
+ const maxPriorityFeePerGasWei =
132
+ input.maxPriorityFeePerGasWei
133
+ ?? (resolvedFees.maxPriorityFeePerGas ?? resolvedFees.gasPrice ?? 0n);
134
+
135
+ if (maxFeePerGasWei === null || maxFeePerGasWei <= 0n) {
136
+ throw new Error('Could not determine maxFeePerGas; pass --max-fee-per-gas-wei');
137
+ }
138
+
139
+ return {
140
+ rpcUrl: input.rpcUrl,
141
+ chainId: input.chainId,
142
+ from: input.from,
143
+ to: input.to,
144
+ valueWei: input.valueWei,
145
+ dataHex: input.dataHex,
146
+ nonce,
147
+ gasLimit,
148
+ maxFeePerGasWei,
149
+ maxPriorityFeePerGasWei,
150
+ txType: input.txType,
151
+ };
152
+ }
153
+
154
+ export async function completeAssetBroadcast(
155
+ plan: AssetBroadcastPlan,
156
+ signed: SignedBroadcastResult,
157
+ deps: CompleteAssetBroadcastDeps,
158
+ ): Promise<CompletedAssetBroadcast> {
159
+ if (!signed.raw_tx_hex) {
160
+ throw new Error('Rust agent did not return raw_tx_hex for broadcast signing');
161
+ }
162
+
163
+ const inspected = await deps.assertSignedBroadcastTransactionMatchesRequest({
164
+ rawTxHex: signed.raw_tx_hex as Hex,
165
+ from: plan.from,
166
+ to: plan.to,
167
+ chainId: plan.chainId,
168
+ nonce: plan.nonce,
169
+ allowHigherNonce: true,
170
+ value: plan.valueWei,
171
+ data: plan.dataHex,
172
+ gasLimit: plan.gasLimit,
173
+ maxFeePerGas: plan.maxFeePerGasWei,
174
+ maxPriorityFeePerGas: plan.maxPriorityFeePerGasWei,
175
+ txType: plan.txType,
176
+ });
177
+
178
+ return {
179
+ signedTxHash: signed.tx_hash_hex ?? null,
180
+ networkTxHash: await deps.broadcastRawTransaction(plan.rpcUrl, signed.raw_tx_hex as Hex),
181
+ signedNonce: inspected.nonce,
182
+ };
183
+ }
184
+
185
+ function isPendingTransactionReceiptError(error: unknown): boolean {
186
+ if (!(error instanceof Error)) {
187
+ return false;
188
+ }
189
+
190
+ const message = error.message.toLowerCase();
191
+ return (
192
+ message.includes('transaction receipt')
193
+ && (
194
+ message.includes('not found')
195
+ || message.includes('could not be found')
196
+ || message.includes('was not found')
197
+ || message.includes('does not exist')
198
+ )
199
+ );
200
+ }
201
+
202
+ export async function waitForOnchainReceipt(
203
+ input: {
204
+ rpcUrl: string;
205
+ txHash: Hex;
206
+ timeoutMs?: number;
207
+ intervalMs?: number;
208
+ },
209
+ deps: WaitForOnchainReceiptDeps,
210
+ ): Promise<WaitForOnchainReceiptResult> {
211
+ const timeoutMs = input.timeoutMs ?? 30_000;
212
+ const intervalMs = input.intervalMs ?? 2_000;
213
+ const now = deps.now ?? Date.now;
214
+ const pause = deps.sleep ?? sleep;
215
+ const started = now();
216
+
217
+ while (true) {
218
+ try {
219
+ return {
220
+ receipt: await deps.getTransactionReceiptByHash(input.rpcUrl, input.txHash),
221
+ timedOut: false,
222
+ };
223
+ } catch (error) {
224
+ if (!isPendingTransactionReceiptError(error)) {
225
+ throw error;
226
+ }
227
+ }
228
+
229
+ if (now() - started >= timeoutMs) {
230
+ return {
231
+ receipt: null,
232
+ timedOut: true,
233
+ };
234
+ }
235
+
236
+ await pause(intervalMs);
237
+ }
238
+ }
239
+
240
+ export function formatBroadcastedAssetOutput(input: {
241
+ command: string;
242
+ counterparty: Address;
243
+ asset: ResolvedAssetMetadata;
244
+ signed: RustAmountOutputShape;
245
+ plan: AssetBroadcastPlan;
246
+ signedNonce?: number;
247
+ networkTxHash: Hex;
248
+ revealRawTx: boolean;
249
+ revealSignature: boolean;
250
+ }) {
251
+ const {
252
+ raw_tx_hex: _rawTxHex,
253
+ tx_hash_hex: _txHashHex,
254
+ ...normalized
255
+ } = normalizeAgentAmountOutput(
256
+ {
257
+ ...input.signed,
258
+ command: input.command,
259
+ network: String(input.plan.chainId),
260
+ counterparty: input.counterparty,
261
+ },
262
+ input.asset,
263
+ );
264
+
265
+ return {
266
+ ...normalized,
267
+ rpcUrl: input.plan.rpcUrl,
268
+ chainId: input.plan.chainId,
269
+ from: input.plan.from,
270
+ nonce: input.signedNonce ?? input.plan.nonce,
271
+ gasLimit: input.plan.gasLimit.toString(),
272
+ maxFeePerGasWei: input.plan.maxFeePerGasWei.toString(),
273
+ maxPriorityFeePerGasWei: input.plan.maxPriorityFeePerGasWei.toString(),
274
+ signedTxHash: input.signed.tx_hash_hex ?? null,
275
+ networkTxHash: input.networkTxHash,
276
+ rawTxHex: input.revealRawTx ? (input.signed.raw_tx_hex ?? null) : '<redacted>',
277
+ signer: input.revealSignature
278
+ ? {
279
+ r: input.signed.r_hex ?? null,
280
+ s: input.signed.s_hex ?? null,
281
+ v: input.signed.v ?? null,
282
+ }
283
+ : '<redacted>',
284
+ };
285
+ }
@@ -0,0 +1 @@
1
+ export * from './bootstrap-artifacts.ts';
@@ -0,0 +1,205 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ import { resolveWlfiHome } from '../../packages/config/src/index.js';
4
+ import {
5
+ assertBootstrapSetupSummaryLeaseIsActive,
6
+ cleanupBootstrapAgentCredentialsFile,
7
+ readBootstrapAgentCredentialsFile,
8
+ readBootstrapSetupSummaryFile
9
+ } from './bootstrap-credentials.js';
10
+ import { assertTrustedDirectoryPath, assertTrustedPrivateFilePath } from './fs-trust.js';
11
+
12
+ const AUTO_GENERATED_BOOTSTRAP_FILE_PATTERN = /^bootstrap-\d+-\d+\.json$/u;
13
+ const MAX_AUTO_GENERATED_BOOTSTRAP_FILES = 128;
14
+
15
+ export type BootstrapArtifactStatus = 'plaintext' | 'redacted' | 'invalid' | 'untrusted';
16
+
17
+ export interface BootstrapArtifactCheck {
18
+ path: string;
19
+ status: BootstrapArtifactStatus;
20
+ agentKeyId: string | null;
21
+ leaseExpiresAt: string | null;
22
+ leaseExpired: boolean;
23
+ error: string | null;
24
+ }
25
+
26
+ export interface CleanupBootstrapArtifactResult extends BootstrapArtifactCheck {
27
+ cleanup: 'deleted' | 'redacted' | 'skipped';
28
+ }
29
+
30
+ export interface CleanupBootstrapArtifactsResult {
31
+ wlfiHome: string;
32
+ action: 'deleted' | 'redacted';
33
+ files: CleanupBootstrapArtifactResult[];
34
+ }
35
+
36
+ interface BootstrapArtifactsDeps {
37
+ resolveWlfiHome?: () => string;
38
+ readdirSync?: typeof fs.readdirSync;
39
+ }
40
+
41
+ function renderError(error: unknown): string {
42
+ return error instanceof Error ? error.message : String(error);
43
+ }
44
+
45
+ function resolveAutoGeneratedBootstrapPaths(
46
+ wlfiHome: string,
47
+ deps: BootstrapArtifactsDeps = {}
48
+ ): string[] {
49
+ const readdirSync = deps.readdirSync ?? fs.readdirSync;
50
+ const resolvedHome = path.resolve(wlfiHome);
51
+ try {
52
+ assertTrustedDirectoryPath(resolvedHome, 'WLFI home');
53
+ } catch {
54
+ return [];
55
+ }
56
+
57
+ let entries: fs.Dirent[];
58
+ try {
59
+ entries = readdirSync(resolvedHome, { withFileTypes: true }) as fs.Dirent[];
60
+ } catch (error) {
61
+ if ((error as NodeJS.ErrnoException).code === 'ENOENT') {
62
+ return [];
63
+ }
64
+ throw error;
65
+ }
66
+
67
+ return entries
68
+ .filter((entry) => AUTO_GENERATED_BOOTSTRAP_FILE_PATTERN.test(entry.name))
69
+ .map((entry) => path.join(resolvedHome, entry.name))
70
+ .sort((left, right) => right.localeCompare(left))
71
+ .slice(0, MAX_AUTO_GENERATED_BOOTSTRAP_FILES);
72
+ }
73
+
74
+ export function inspectBootstrapArtifact(targetPath: string): BootstrapArtifactCheck {
75
+ const resolvedPath = path.resolve(targetPath);
76
+
77
+ try {
78
+ assertTrustedPrivateFilePath(resolvedPath, 'bootstrap file');
79
+ } catch (error) {
80
+ return {
81
+ path: resolvedPath,
82
+ status: 'untrusted',
83
+ agentKeyId: null,
84
+ leaseExpiresAt: null,
85
+ leaseExpired: false,
86
+ error: renderError(error)
87
+ };
88
+ }
89
+
90
+ let summary: ReturnType<typeof readBootstrapSetupSummaryFile>;
91
+ try {
92
+ summary = readBootstrapSetupSummaryFile(resolvedPath);
93
+ } catch (error) {
94
+ return {
95
+ path: resolvedPath,
96
+ status: 'invalid',
97
+ agentKeyId: null,
98
+ leaseExpiresAt: null,
99
+ leaseExpired: false,
100
+ error: renderError(error)
101
+ };
102
+ }
103
+
104
+ let leaseExpired = false;
105
+ try {
106
+ assertBootstrapSetupSummaryLeaseIsActive(summary);
107
+ } catch (error) {
108
+ const message = renderError(error);
109
+ if (message.includes('lease has expired')) {
110
+ leaseExpired = true;
111
+ } else {
112
+ return {
113
+ path: resolvedPath,
114
+ status: 'invalid',
115
+ agentKeyId: summary.agentKeyId,
116
+ leaseExpiresAt: summary.leaseExpiresAt,
117
+ leaseExpired: false,
118
+ error: message
119
+ };
120
+ }
121
+ }
122
+
123
+ try {
124
+ const credentials = readBootstrapAgentCredentialsFile(resolvedPath);
125
+ return {
126
+ path: resolvedPath,
127
+ status: 'plaintext',
128
+ agentKeyId: credentials.agentKeyId,
129
+ leaseExpiresAt: summary.leaseExpiresAt,
130
+ leaseExpired,
131
+ error: null
132
+ };
133
+ } catch (error) {
134
+ const message = renderError(error);
135
+ if (message.includes('contains a redacted agent auth token')) {
136
+ if (summary.vaultPrivateKey !== null) {
137
+ return {
138
+ path: resolvedPath,
139
+ status: 'plaintext',
140
+ agentKeyId: summary.agentKeyId,
141
+ leaseExpiresAt: summary.leaseExpiresAt,
142
+ leaseExpired,
143
+ error: null
144
+ };
145
+ }
146
+ return {
147
+ path: resolvedPath,
148
+ status: 'redacted',
149
+ agentKeyId: summary.agentKeyId,
150
+ leaseExpiresAt: summary.leaseExpiresAt,
151
+ leaseExpired,
152
+ error: null
153
+ };
154
+ }
155
+
156
+ return {
157
+ path: resolvedPath,
158
+ status: 'invalid',
159
+ agentKeyId: summary.agentKeyId,
160
+ leaseExpiresAt: summary.leaseExpiresAt,
161
+ leaseExpired,
162
+ error: message
163
+ };
164
+ }
165
+ }
166
+
167
+ export function listAutoGeneratedBootstrapArtifacts(
168
+ deps: BootstrapArtifactsDeps = {}
169
+ ): BootstrapArtifactCheck[] {
170
+ const wlfiHome = (deps.resolveWlfiHome ?? resolveWlfiHome)();
171
+ return resolveAutoGeneratedBootstrapPaths(wlfiHome, deps).map((targetPath) =>
172
+ inspectBootstrapArtifact(targetPath)
173
+ );
174
+ }
175
+
176
+ export function cleanupAutoGeneratedBootstrapArtifacts(
177
+ action: 'deleted' | 'redacted' = 'deleted',
178
+ deps: BootstrapArtifactsDeps = {}
179
+ ): CleanupBootstrapArtifactsResult {
180
+ const wlfiHome = path.resolve((deps.resolveWlfiHome ?? resolveWlfiHome)());
181
+ assertTrustedDirectoryPath(wlfiHome, 'WLFI home');
182
+
183
+ const files = resolveAutoGeneratedBootstrapPaths(wlfiHome, deps).map((targetPath) => {
184
+ const artifact = inspectBootstrapArtifact(targetPath);
185
+ if (artifact.status !== 'plaintext' && artifact.status !== 'redacted') {
186
+ return {
187
+ ...artifact,
188
+ cleanup: 'skipped' as const
189
+ };
190
+ }
191
+
192
+ const result = cleanupBootstrapAgentCredentialsFile(targetPath, action);
193
+ return {
194
+ ...artifact,
195
+ path: result.sourcePath,
196
+ cleanup: action
197
+ };
198
+ });
199
+
200
+ return {
201
+ wlfiHome,
202
+ action,
203
+ files
204
+ };
205
+ }
@@ -0,0 +1 @@
1
+ export * from './bootstrap-credentials.ts';