@wlfi-agent/cli 1.4.13 → 1.4.14

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 +49 -43
  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,624 @@
1
+ use std::collections::BTreeMap;
2
+ use std::fs::{self, OpenOptions};
3
+ use std::io::Write;
4
+ use std::path::{Path, PathBuf};
5
+
6
+ use anyhow::{bail, Context, Result};
7
+ use serde::{Deserialize, Serialize};
8
+ use serde_json::Value;
9
+ #[cfg(unix)]
10
+ use std::os::unix::fs::{OpenOptionsExt, PermissionsExt};
11
+
12
+ const CONFIG_FILENAME: &str = "config.json";
13
+ #[cfg(unix)]
14
+ const PRIVATE_DIR_MODE: u32 = 0o700;
15
+ #[cfg(unix)]
16
+ const PRIVATE_FILE_MODE: u32 = 0o600;
17
+
18
+ const DEFAULT_ETH_RPC_URL: &str = "https://eth.llamarpc.com";
19
+ const DEFAULT_BSC_RPC_URL: &str = "https://bsc.drpc.org";
20
+ const DEFAULT_USD1_ADDRESS: &str = "0xc83DE66ebA6a91B6F3d167f2ee9F0C42aD70B611";
21
+
22
+ #[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
23
+ #[serde(default, rename_all = "camelCase")]
24
+ pub(crate) struct WlfiConfig {
25
+ pub(crate) rpc_url: Option<String>,
26
+ pub(crate) chain_id: Option<u64>,
27
+ pub(crate) chain_name: Option<String>,
28
+ pub(crate) daemon_socket: Option<String>,
29
+ pub(crate) state_file: Option<String>,
30
+ pub(crate) rust_bin_dir: Option<String>,
31
+ pub(crate) agent_key_id: Option<String>,
32
+ pub(crate) agent_auth_token: Option<String>,
33
+ pub(crate) wallet: Option<WalletProfile>,
34
+ pub(crate) chains: BTreeMap<String, ChainProfile>,
35
+ pub(crate) tokens: BTreeMap<String, TokenProfile>,
36
+ #[serde(flatten)]
37
+ pub(crate) extra: BTreeMap<String, Value>,
38
+ }
39
+
40
+ impl Default for WlfiConfig {
41
+ fn default() -> Self {
42
+ Self {
43
+ rpc_url: None,
44
+ chain_id: None,
45
+ chain_name: None,
46
+ daemon_socket: None,
47
+ state_file: None,
48
+ rust_bin_dir: None,
49
+ agent_key_id: None,
50
+ agent_auth_token: None,
51
+ wallet: None,
52
+ chains: default_chain_profiles(),
53
+ tokens: default_token_profiles(),
54
+ extra: BTreeMap::new(),
55
+ }
56
+ }
57
+ }
58
+
59
+ #[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
60
+ #[serde(default, rename_all = "camelCase")]
61
+ pub(crate) struct WalletProfile {
62
+ pub(crate) vault_key_id: Option<String>,
63
+ pub(crate) vault_public_key: String,
64
+ pub(crate) address: Option<String>,
65
+ pub(crate) agent_key_id: Option<String>,
66
+ pub(crate) policy_attachment: String,
67
+ pub(crate) attached_policy_ids: Vec<String>,
68
+ pub(crate) policy_note: Option<String>,
69
+ pub(crate) network_scope: Option<String>,
70
+ pub(crate) asset_scope: Option<String>,
71
+ pub(crate) recipient_scope: Option<String>,
72
+ #[serde(flatten)]
73
+ pub(crate) extra: BTreeMap<String, Value>,
74
+ }
75
+
76
+ #[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
77
+ #[serde(default, rename_all = "camelCase")]
78
+ pub(crate) struct ChainProfile {
79
+ pub(crate) chain_id: u64,
80
+ pub(crate) name: String,
81
+ pub(crate) rpc_url: Option<String>,
82
+ #[serde(flatten)]
83
+ pub(crate) extra: BTreeMap<String, Value>,
84
+ }
85
+
86
+ #[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
87
+ #[serde(default, rename_all = "camelCase")]
88
+ pub(crate) struct TokenPolicyProfile {
89
+ pub(crate) per_tx_amount: Option<f64>,
90
+ pub(crate) daily_amount: Option<f64>,
91
+ pub(crate) weekly_amount: Option<f64>,
92
+ pub(crate) per_tx_amount_decimal: Option<String>,
93
+ pub(crate) daily_amount_decimal: Option<String>,
94
+ pub(crate) weekly_amount_decimal: Option<String>,
95
+ pub(crate) per_tx_limit: Option<String>,
96
+ pub(crate) daily_limit: Option<String>,
97
+ pub(crate) weekly_limit: Option<String>,
98
+ pub(crate) max_gas_per_chain_wei: Option<String>,
99
+ pub(crate) daily_max_tx_count: Option<String>,
100
+ pub(crate) per_tx_max_fee_per_gas_gwei: Option<String>,
101
+ pub(crate) per_tx_max_fee_per_gas_wei: Option<String>,
102
+ pub(crate) per_tx_max_priority_fee_per_gas_wei: Option<String>,
103
+ pub(crate) per_tx_max_calldata_bytes: Option<String>,
104
+ #[serde(flatten)]
105
+ pub(crate) extra: BTreeMap<String, Value>,
106
+ }
107
+
108
+ #[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
109
+ #[serde(default, rename_all = "camelCase")]
110
+ pub(crate) struct TokenDestinationOverrideProfile {
111
+ pub(crate) recipient: String,
112
+ #[serde(flatten)]
113
+ pub(crate) limits: TokenPolicyProfile,
114
+ }
115
+
116
+ #[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
117
+ #[serde(default, rename_all = "camelCase")]
118
+ pub(crate) struct TokenManualApprovalProfile {
119
+ pub(crate) priority: u32,
120
+ pub(crate) recipient: Option<String>,
121
+ pub(crate) min_amount: Option<f64>,
122
+ pub(crate) max_amount: Option<f64>,
123
+ pub(crate) min_amount_decimal: Option<String>,
124
+ pub(crate) max_amount_decimal: Option<String>,
125
+ pub(crate) min_amount_wei: Option<String>,
126
+ pub(crate) max_amount_wei: Option<String>,
127
+ #[serde(flatten)]
128
+ pub(crate) extra: BTreeMap<String, Value>,
129
+ }
130
+
131
+ #[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
132
+ #[serde(default, rename_all = "camelCase")]
133
+ pub(crate) struct TokenChainProfile {
134
+ pub(crate) chain_id: u64,
135
+ pub(crate) is_native: bool,
136
+ pub(crate) address: Option<String>,
137
+ pub(crate) decimals: u8,
138
+ pub(crate) default_policy: Option<TokenPolicyProfile>,
139
+ #[serde(flatten)]
140
+ pub(crate) extra: BTreeMap<String, Value>,
141
+ }
142
+
143
+ #[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
144
+ #[serde(default, rename_all = "camelCase")]
145
+ pub(crate) struct TokenProfile {
146
+ pub(crate) name: Option<String>,
147
+ pub(crate) symbol: String,
148
+ pub(crate) default_policy: Option<TokenPolicyProfile>,
149
+ pub(crate) destination_overrides: Vec<TokenDestinationOverrideProfile>,
150
+ pub(crate) manual_approval_policies: Vec<TokenManualApprovalProfile>,
151
+ pub(crate) chains: BTreeMap<String, TokenChainProfile>,
152
+ #[serde(flatten)]
153
+ pub(crate) extra: BTreeMap<String, Value>,
154
+ }
155
+
156
+ fn default_chain_profiles() -> BTreeMap<String, ChainProfile> {
157
+ BTreeMap::from([
158
+ (
159
+ "bsc".to_string(),
160
+ ChainProfile {
161
+ chain_id: 56,
162
+ name: "BSC".to_string(),
163
+ rpc_url: Some(DEFAULT_BSC_RPC_URL.to_string()),
164
+ extra: BTreeMap::new(),
165
+ },
166
+ ),
167
+ (
168
+ "eth".to_string(),
169
+ ChainProfile {
170
+ chain_id: 1,
171
+ name: "ETH".to_string(),
172
+ rpc_url: Some(DEFAULT_ETH_RPC_URL.to_string()),
173
+ extra: BTreeMap::new(),
174
+ },
175
+ ),
176
+ ])
177
+ }
178
+
179
+ fn default_token_profiles() -> BTreeMap<String, TokenProfile> {
180
+ let usd1_policy = default_token_policy("10", "100", "700");
181
+ let bnb_policy = default_token_policy("0.01", "0.2", "1.4");
182
+
183
+ BTreeMap::from([
184
+ (
185
+ "bnb".to_string(),
186
+ TokenProfile {
187
+ name: Some("BNB".to_string()),
188
+ symbol: "BNB".to_string(),
189
+ default_policy: Some(bnb_policy.clone()),
190
+ destination_overrides: Vec::new(),
191
+ manual_approval_policies: Vec::new(),
192
+ chains: BTreeMap::from([(
193
+ "bsc".to_string(),
194
+ TokenChainProfile {
195
+ chain_id: 56,
196
+ is_native: true,
197
+ address: None,
198
+ decimals: 18,
199
+ default_policy: Some(bnb_policy),
200
+ extra: BTreeMap::new(),
201
+ },
202
+ )]),
203
+ extra: BTreeMap::new(),
204
+ },
205
+ ),
206
+ (
207
+ "usd1".to_string(),
208
+ TokenProfile {
209
+ name: Some("USD1".to_string()),
210
+ symbol: "USD1".to_string(),
211
+ default_policy: Some(usd1_policy.clone()),
212
+ destination_overrides: Vec::new(),
213
+ manual_approval_policies: Vec::new(),
214
+ chains: BTreeMap::from([
215
+ (
216
+ "bsc".to_string(),
217
+ TokenChainProfile {
218
+ chain_id: 56,
219
+ is_native: false,
220
+ address: Some(DEFAULT_USD1_ADDRESS.to_string()),
221
+ decimals: 18,
222
+ default_policy: Some(usd1_policy.clone()),
223
+ extra: BTreeMap::new(),
224
+ },
225
+ ),
226
+ (
227
+ "eth".to_string(),
228
+ TokenChainProfile {
229
+ chain_id: 1,
230
+ is_native: false,
231
+ address: Some(DEFAULT_USD1_ADDRESS.to_string()),
232
+ decimals: 18,
233
+ default_policy: Some(usd1_policy),
234
+ extra: BTreeMap::new(),
235
+ },
236
+ ),
237
+ ]),
238
+ extra: BTreeMap::new(),
239
+ },
240
+ ),
241
+ ])
242
+ }
243
+
244
+ fn default_token_policy(per_tx: &str, daily: &str, weekly: &str) -> TokenPolicyProfile {
245
+ TokenPolicyProfile {
246
+ per_tx_amount: None,
247
+ daily_amount: None,
248
+ weekly_amount: None,
249
+ per_tx_amount_decimal: Some(per_tx.to_string()),
250
+ daily_amount_decimal: Some(daily.to_string()),
251
+ weekly_amount_decimal: Some(weekly.to_string()),
252
+ per_tx_limit: None,
253
+ daily_limit: None,
254
+ weekly_limit: None,
255
+ max_gas_per_chain_wei: None,
256
+ daily_max_tx_count: None,
257
+ per_tx_max_fee_per_gas_gwei: None,
258
+ per_tx_max_fee_per_gas_wei: None,
259
+ per_tx_max_priority_fee_per_gas_wei: None,
260
+ per_tx_max_calldata_bytes: None,
261
+ extra: BTreeMap::new(),
262
+ }
263
+ }
264
+
265
+ #[derive(Debug, Clone, PartialEq)]
266
+ pub(crate) struct LoadedConfig {
267
+ pub(crate) path: PathBuf,
268
+ pub(crate) config: WlfiConfig,
269
+ }
270
+
271
+ impl LoadedConfig {
272
+ pub(crate) fn load_default() -> Result<Self> {
273
+ let path = default_config_path()?;
274
+ let config = WlfiConfig::read_from_path(&path)?;
275
+ Ok(Self { path, config })
276
+ }
277
+
278
+ pub(crate) fn save(&self) -> Result<()> {
279
+ self.config.write_to_path(&self.path)
280
+ }
281
+ }
282
+
283
+ impl WlfiConfig {
284
+ fn apply_seed_defaults_if_legacy_empty(mut self) -> Self {
285
+ if self.chains.is_empty() && self.tokens.is_empty() {
286
+ self.chains = default_chain_profiles();
287
+ self.tokens = default_token_profiles();
288
+ }
289
+ self
290
+ }
291
+
292
+ pub(crate) fn read_from_path(path: &Path) -> Result<Self> {
293
+ match fs::read_to_string(path) {
294
+ Ok(raw) => serde_json::from_str::<Self>(&raw)
295
+ .map(Self::apply_seed_defaults_if_legacy_empty)
296
+ .with_context(|| format!("failed to parse config file '{}'", path.display())),
297
+ Err(error) if error.kind() == std::io::ErrorKind::NotFound => Ok(Self::default()),
298
+ Err(error) => Err(error)
299
+ .with_context(|| format!("failed to read config file '{}'", path.display())),
300
+ }
301
+ }
302
+
303
+ pub(crate) fn write_to_path(&self, path: &Path) -> Result<()> {
304
+ if let Some(parent) = path.parent() {
305
+ fs::create_dir_all(parent).with_context(|| {
306
+ format!("failed to create config directory '{}'", parent.display())
307
+ })?;
308
+ #[cfg(unix)]
309
+ {
310
+ fs::set_permissions(parent, fs::Permissions::from_mode(PRIVATE_DIR_MODE))
311
+ .with_context(|| {
312
+ format!("failed to secure config directory '{}'", parent.display())
313
+ })?;
314
+ }
315
+ }
316
+
317
+ let rendered = serde_json::to_string_pretty(self).context("failed to serialize config")?;
318
+ let mut options = OpenOptions::new();
319
+ options.create(true).truncate(true).write(true);
320
+ #[cfg(unix)]
321
+ options.mode(PRIVATE_FILE_MODE);
322
+ let mut file = options
323
+ .open(path)
324
+ .with_context(|| format!("failed to open config file '{}'", path.display()))?;
325
+ file.write_all(rendered.as_bytes())
326
+ .with_context(|| format!("failed to write config file '{}'", path.display()))?;
327
+ file.write_all(b"\n")
328
+ .with_context(|| format!("failed to finalize config file '{}'", path.display()))?;
329
+ #[cfg(unix)]
330
+ fs::set_permissions(path, fs::Permissions::from_mode(PRIVATE_FILE_MODE))
331
+ .with_context(|| format!("failed to secure config file '{}'", path.display()))?;
332
+ Ok(())
333
+ }
334
+ }
335
+
336
+ pub(crate) fn default_config_path() -> Result<PathBuf> {
337
+ Ok(wlfi_home_dir()?.join(CONFIG_FILENAME))
338
+ }
339
+
340
+ fn wlfi_home_dir() -> Result<PathBuf> {
341
+ if let Some(path) = std::env::var_os("WLFI_HOME") {
342
+ let candidate = PathBuf::from(path);
343
+ if candidate.as_os_str().is_empty() {
344
+ bail!("WLFI_HOME must not be empty");
345
+ }
346
+ return Ok(candidate);
347
+ }
348
+
349
+ let Some(home) = std::env::var_os("HOME") else {
350
+ bail!("HOME is not set; use WLFI_HOME to choose config directory");
351
+ };
352
+ Ok(PathBuf::from(home).join(".wlfi_agent"))
353
+ }
354
+
355
+ #[cfg(test)]
356
+ mod tests {
357
+ use super::{
358
+ default_config_path, LoadedConfig, TokenChainProfile, TokenDestinationOverrideProfile,
359
+ TokenManualApprovalProfile, TokenPolicyProfile, TokenProfile, WlfiConfig,
360
+ };
361
+ use std::collections::BTreeMap;
362
+ use std::fs;
363
+ use std::path::PathBuf;
364
+
365
+ fn temp_root(label: &str) -> PathBuf {
366
+ let unique = format!(
367
+ "wlfi-agent-admin-shared-config-{label}-{}-{}",
368
+ std::process::id(),
369
+ std::time::SystemTime::now()
370
+ .duration_since(std::time::UNIX_EPOCH)
371
+ .expect("unix time")
372
+ .as_nanos()
373
+ );
374
+ std::env::temp_dir().join(unique)
375
+ }
376
+
377
+ #[test]
378
+ fn read_from_missing_path_returns_default_config() {
379
+ let root = temp_root("missing");
380
+ let path = root.join("config.json");
381
+ let config = WlfiConfig::read_from_path(&path).expect("missing path should default");
382
+ assert_eq!(config, WlfiConfig::default());
383
+ }
384
+
385
+ #[test]
386
+ fn read_from_legacy_empty_config_reseeds_defaults() {
387
+ let root = temp_root("legacy-empty");
388
+ fs::create_dir_all(&root).expect("create root");
389
+ let path = root.join("config.json");
390
+ fs::write(
391
+ &path,
392
+ r#"{
393
+ "chains": {},
394
+ "tokens": {},
395
+ "chainId": 11155111,
396
+ "chainName": "sepolia",
397
+ "rpcUrl": "https://rpc.sepolia.example"
398
+ }
399
+ "#,
400
+ )
401
+ .expect("write legacy config");
402
+
403
+ let config = WlfiConfig::read_from_path(&path).expect("read config");
404
+ assert!(config.chains.contains_key("eth"));
405
+ assert!(config.chains.contains_key("bsc"));
406
+ assert!(config.tokens.contains_key("usd1"));
407
+ assert!(config.tokens.contains_key("bnb"));
408
+ assert_eq!(config.chain_id, Some(11155111));
409
+ assert_eq!(config.chain_name.as_deref(), Some("sepolia"));
410
+ assert_eq!(
411
+ config.rpc_url.as_deref(),
412
+ Some("https://rpc.sepolia.example")
413
+ );
414
+
415
+ fs::remove_dir_all(&root).expect("cleanup temp root");
416
+ }
417
+
418
+ #[test]
419
+ fn default_config_seeds_eth_bsc_usd1_and_bnb() {
420
+ let config = WlfiConfig::default();
421
+
422
+ assert_eq!(config.chains["eth"].chain_id, 1);
423
+ assert_eq!(
424
+ config.chains["eth"].rpc_url.as_deref(),
425
+ Some("https://eth.llamarpc.com")
426
+ );
427
+ assert_eq!(config.chains["bsc"].chain_id, 56);
428
+ assert_eq!(
429
+ config.chains["bsc"].rpc_url.as_deref(),
430
+ Some("https://bsc.drpc.org")
431
+ );
432
+
433
+ let usd1 = config.tokens.get("usd1").expect("usd1 default");
434
+ assert_eq!(usd1.symbol, "USD1");
435
+ assert_eq!(
436
+ usd1.default_policy
437
+ .as_ref()
438
+ .and_then(|policy| policy.per_tx_amount_decimal.as_deref()),
439
+ Some("10")
440
+ );
441
+ assert_eq!(
442
+ usd1.default_policy
443
+ .as_ref()
444
+ .and_then(|policy| policy.daily_amount_decimal.as_deref()),
445
+ Some("100")
446
+ );
447
+ assert_eq!(
448
+ usd1.default_policy
449
+ .as_ref()
450
+ .and_then(|policy| policy.weekly_amount_decimal.as_deref()),
451
+ Some("700")
452
+ );
453
+ assert_eq!(
454
+ usd1.chains["eth"].address.as_deref(),
455
+ Some("0xc83DE66ebA6a91B6F3d167f2ee9F0C42aD70B611")
456
+ );
457
+ assert_eq!(
458
+ usd1.chains["bsc"].address.as_deref(),
459
+ Some("0xc83DE66ebA6a91B6F3d167f2ee9F0C42aD70B611")
460
+ );
461
+
462
+ let bnb = config.tokens.get("bnb").expect("bnb default");
463
+ assert_eq!(bnb.symbol, "BNB");
464
+ assert!(bnb.chains["bsc"].is_native);
465
+ assert_eq!(
466
+ bnb.default_policy
467
+ .as_ref()
468
+ .and_then(|policy| policy.per_tx_amount_decimal.as_deref()),
469
+ Some("0.01")
470
+ );
471
+ assert_eq!(
472
+ bnb.default_policy
473
+ .as_ref()
474
+ .and_then(|policy| policy.daily_amount_decimal.as_deref()),
475
+ Some("0.2")
476
+ );
477
+ assert_eq!(
478
+ bnb.default_policy
479
+ .as_ref()
480
+ .and_then(|policy| policy.weekly_amount_decimal.as_deref()),
481
+ Some("1.4")
482
+ );
483
+ }
484
+
485
+ #[test]
486
+ fn write_and_read_round_trip_preserves_chain_and_token_profiles() {
487
+ let root = temp_root("roundtrip");
488
+ fs::create_dir_all(&root).expect("create root");
489
+ let path = root.join("config.json");
490
+
491
+ let mut config = WlfiConfig {
492
+ chain_id: Some(11155111),
493
+ chain_name: Some("sepolia".to_string()),
494
+ rpc_url: Some("https://rpc.sepolia.example".to_string()),
495
+ ..WlfiConfig::default()
496
+ };
497
+ config.chains.insert(
498
+ "sepolia".to_string(),
499
+ super::ChainProfile {
500
+ chain_id: 11155111,
501
+ name: "sepolia".to_string(),
502
+ rpc_url: Some("https://rpc.sepolia.example".to_string()),
503
+ extra: BTreeMap::new(),
504
+ },
505
+ );
506
+ config.tokens.insert(
507
+ "usdc".to_string(),
508
+ TokenProfile {
509
+ name: Some("USD Coin".to_string()),
510
+ symbol: "USDC".to_string(),
511
+ default_policy: Some(TokenPolicyProfile {
512
+ per_tx_amount: Some(25.0),
513
+ daily_amount: Some(100.0),
514
+ weekly_amount: Some(500.0),
515
+ per_tx_amount_decimal: Some("25".to_string()),
516
+ daily_amount_decimal: Some("100".to_string()),
517
+ weekly_amount_decimal: Some("500".to_string()),
518
+ per_tx_limit: None,
519
+ daily_limit: None,
520
+ weekly_limit: None,
521
+ max_gas_per_chain_wei: Some("1000000000000000".to_string()),
522
+ daily_max_tx_count: Some("0".to_string()),
523
+ per_tx_max_fee_per_gas_gwei: Some("25".to_string()),
524
+ per_tx_max_fee_per_gas_wei: Some("25000000000".to_string()),
525
+ per_tx_max_priority_fee_per_gas_wei: Some("0".to_string()),
526
+ per_tx_max_calldata_bytes: Some("0".to_string()),
527
+ extra: BTreeMap::new(),
528
+ }),
529
+ destination_overrides: vec![TokenDestinationOverrideProfile {
530
+ recipient: "0x1000000000000000000000000000000000000001".to_string(),
531
+ limits: TokenPolicyProfile {
532
+ per_tx_amount: None,
533
+ daily_amount: None,
534
+ weekly_amount: None,
535
+ per_tx_amount_decimal: Some("10".to_string()),
536
+ daily_amount_decimal: Some("50".to_string()),
537
+ weekly_amount_decimal: Some("200".to_string()),
538
+ per_tx_limit: None,
539
+ daily_limit: None,
540
+ weekly_limit: None,
541
+ max_gas_per_chain_wei: Some("100000000000000".to_string()),
542
+ daily_max_tx_count: Some("0".to_string()),
543
+ per_tx_max_fee_per_gas_gwei: Some("15".to_string()),
544
+ per_tx_max_fee_per_gas_wei: Some("15000000000".to_string()),
545
+ per_tx_max_priority_fee_per_gas_wei: Some("0".to_string()),
546
+ per_tx_max_calldata_bytes: Some("0".to_string()),
547
+ extra: BTreeMap::new(),
548
+ },
549
+ }],
550
+ manual_approval_policies: vec![TokenManualApprovalProfile {
551
+ priority: 100,
552
+ recipient: None,
553
+ min_amount: Some(250.0),
554
+ max_amount: Some(500.0),
555
+ min_amount_decimal: Some("250".to_string()),
556
+ max_amount_decimal: Some("500".to_string()),
557
+ min_amount_wei: None,
558
+ max_amount_wei: None,
559
+ extra: BTreeMap::new(),
560
+ }],
561
+ chains: BTreeMap::from([(
562
+ "sepolia".to_string(),
563
+ TokenChainProfile {
564
+ chain_id: 11155111,
565
+ is_native: false,
566
+ address: Some("0x1000000000000000000000000000000000000000".to_string()),
567
+ decimals: 6,
568
+ default_policy: Some(TokenPolicyProfile {
569
+ per_tx_amount: Some(25.0),
570
+ daily_amount: Some(100.0),
571
+ weekly_amount: Some(500.0),
572
+ per_tx_amount_decimal: Some("25".to_string()),
573
+ daily_amount_decimal: Some("100".to_string()),
574
+ weekly_amount_decimal: Some("500".to_string()),
575
+ per_tx_limit: Some("25".to_string()),
576
+ daily_limit: Some("100".to_string()),
577
+ weekly_limit: Some("500".to_string()),
578
+ max_gas_per_chain_wei: Some("1000000000000000".to_string()),
579
+ daily_max_tx_count: Some("0".to_string()),
580
+ per_tx_max_fee_per_gas_gwei: Some("25".to_string()),
581
+ per_tx_max_fee_per_gas_wei: Some("0".to_string()),
582
+ per_tx_max_priority_fee_per_gas_wei: Some("0".to_string()),
583
+ per_tx_max_calldata_bytes: Some("0".to_string()),
584
+ extra: BTreeMap::new(),
585
+ }),
586
+ extra: BTreeMap::new(),
587
+ },
588
+ )]),
589
+ extra: BTreeMap::new(),
590
+ },
591
+ );
592
+
593
+ config.write_to_path(&path).expect("write config");
594
+ let loaded = WlfiConfig::read_from_path(&path).expect("read config");
595
+ assert_eq!(loaded, config);
596
+
597
+ let loaded_document = LoadedConfig {
598
+ path: path.clone(),
599
+ config,
600
+ };
601
+ loaded_document.save().expect("save loaded document");
602
+ assert!(path.exists());
603
+
604
+ fs::remove_dir_all(&root).expect("cleanup temp root");
605
+ }
606
+
607
+ #[test]
608
+ fn default_config_path_uses_wlfi_home_when_present() {
609
+ let root = temp_root("wlfi-home");
610
+ fs::create_dir_all(&root).expect("create root");
611
+ let previous = std::env::var_os("WLFI_HOME");
612
+ std::env::set_var("WLFI_HOME", &root);
613
+
614
+ let path = default_config_path().expect("resolve default config path");
615
+ assert_eq!(path, root.join("config.json"));
616
+
617
+ if let Some(value) = previous {
618
+ std::env::set_var("WLFI_HOME", value);
619
+ } else {
620
+ std::env::remove_var("WLFI_HOME");
621
+ }
622
+ fs::remove_dir_all(&root).expect("cleanup temp root");
623
+ }
624
+ }