@wlfi-agent/cli 1.4.12 → 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 -1894
  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 -1894
@@ -0,0 +1,891 @@
1
+ use std::collections::BTreeSet;
2
+ use std::str::FromStr;
3
+
4
+ use time::{Duration, OffsetDateTime};
5
+ use uuid::Uuid;
6
+ use vault_domain::{
7
+ AgentAction, AssetId, BroadcastTx, EntityScope, EvmAddress, PolicyAttachment, PolicyType,
8
+ SpendEvent, SpendingPolicy,
9
+ };
10
+
11
+ use super::{PolicyEngine, PolicyError};
12
+
13
+ fn addr(x: &str) -> EvmAddress {
14
+ EvmAddress::from_str(x).expect("valid test address")
15
+ }
16
+
17
+ fn policy_all_per_tx(max: u128) -> SpendingPolicy {
18
+ SpendingPolicy::new(
19
+ 1,
20
+ PolicyType::PerTxMaxSpending,
21
+ max,
22
+ EntityScope::All,
23
+ EntityScope::All,
24
+ EntityScope::All,
25
+ )
26
+ .expect("policy")
27
+ }
28
+
29
+ fn gas_policy(chain_id: u64, max_gas_wei: u128) -> SpendingPolicy {
30
+ SpendingPolicy::new(
31
+ 0,
32
+ PolicyType::PerChainMaxGasSpend,
33
+ max_gas_wei,
34
+ EntityScope::All,
35
+ EntityScope::All,
36
+ EntityScope::Set(BTreeSet::from([chain_id])),
37
+ )
38
+ .expect("policy")
39
+ }
40
+
41
+ fn broadcast_action_with_fees(
42
+ max_fee_per_gas_wei: u128,
43
+ max_priority_fee_per_gas_wei: u128,
44
+ data_hex: &str,
45
+ ) -> AgentAction {
46
+ AgentAction::BroadcastTx {
47
+ tx: BroadcastTx {
48
+ chain_id: 1,
49
+ nonce: 0,
50
+ to: addr("0x1000000000000000000000000000000000000000"),
51
+ value_wei: 0,
52
+ data_hex: data_hex.to_string(),
53
+ gas_limit: 21_000,
54
+ max_fee_per_gas_wei,
55
+ max_priority_fee_per_gas_wei,
56
+ tx_type: 0x02,
57
+ delegation_enabled: false,
58
+ },
59
+ }
60
+ }
61
+
62
+ #[test]
63
+ fn per_tx_limit_is_enforced() {
64
+ let engine = PolicyEngine;
65
+ let token = addr("0x1000000000000000000000000000000000000000");
66
+ let recipient = addr("0x2000000000000000000000000000000000000000");
67
+ let policy = policy_all_per_tx(100);
68
+
69
+ let action = AgentAction::Transfer {
70
+ chain_id: 1,
71
+ token,
72
+ to: recipient,
73
+ amount_wei: 101,
74
+ };
75
+
76
+ let result = engine.evaluate(
77
+ &[policy],
78
+ &PolicyAttachment::AllPolicies,
79
+ &action,
80
+ &[],
81
+ Uuid::new_v4(),
82
+ OffsetDateTime::now_utc(),
83
+ );
84
+
85
+ assert!(matches!(
86
+ result,
87
+ Err(PolicyError::PerTxLimitExceeded { .. })
88
+ ));
89
+ }
90
+
91
+ #[test]
92
+ fn daily_limit_sums_recent_usage() {
93
+ let engine = PolicyEngine;
94
+ let now = OffsetDateTime::now_utc();
95
+ let token = addr("0x3000000000000000000000000000000000000000");
96
+ let recipient = addr("0x4000000000000000000000000000000000000000");
97
+ let policy = SpendingPolicy::new(
98
+ 1,
99
+ PolicyType::DailyMaxSpending,
100
+ 100,
101
+ EntityScope::All,
102
+ EntityScope::Set(BTreeSet::from([AssetId::Erc20(token.clone())])),
103
+ EntityScope::Set(BTreeSet::from([1_u64])),
104
+ )
105
+ .expect("policy must build");
106
+
107
+ let agent_key_id = Uuid::new_v4();
108
+ let history = vec![SpendEvent {
109
+ agent_key_id,
110
+ chain_id: 1,
111
+ asset: AssetId::Erc20(token.clone()),
112
+ recipient: recipient.clone(),
113
+ amount_wei: 70,
114
+ at: now - time::Duration::hours(2),
115
+ }];
116
+
117
+ let action = AgentAction::Transfer {
118
+ chain_id: 1,
119
+ token,
120
+ to: recipient,
121
+ amount_wei: 31,
122
+ };
123
+
124
+ let result = engine.evaluate(
125
+ &[policy],
126
+ &PolicyAttachment::AllPolicies,
127
+ &action,
128
+ &history,
129
+ agent_key_id,
130
+ now,
131
+ );
132
+
133
+ assert!(matches!(
134
+ result,
135
+ Err(PolicyError::WindowLimitExceeded {
136
+ used_amount_wei: 70,
137
+ requested_amount_wei: 31,
138
+ ..
139
+ })
140
+ ));
141
+ }
142
+
143
+ #[test]
144
+ fn daily_limit_counts_all_in_scope_usage_across_assets_and_chains() {
145
+ let engine = PolicyEngine;
146
+ let now = OffsetDateTime::now_utc();
147
+ let token_a = addr("0x3000000000000000000000000000000000000000");
148
+ let token_b = addr("0x5000000000000000000000000000000000000000");
149
+ let recipient_a = addr("0x4000000000000000000000000000000000000000");
150
+ let recipient_b = addr("0x6000000000000000000000000000000000000000");
151
+ let policy = SpendingPolicy::new(
152
+ 1,
153
+ PolicyType::DailyMaxSpending,
154
+ 100,
155
+ EntityScope::All,
156
+ EntityScope::All,
157
+ EntityScope::All,
158
+ )
159
+ .expect("policy must build");
160
+
161
+ let agent_key_id = Uuid::new_v4();
162
+ let history = vec![
163
+ SpendEvent {
164
+ agent_key_id,
165
+ chain_id: 1,
166
+ asset: AssetId::Erc20(token_a),
167
+ recipient: recipient_a,
168
+ amount_wei: 70,
169
+ at: now - time::Duration::hours(2),
170
+ },
171
+ SpendEvent {
172
+ agent_key_id,
173
+ chain_id: 10,
174
+ asset: AssetId::NativeEth,
175
+ recipient: recipient_b.clone(),
176
+ amount_wei: 20,
177
+ at: now - time::Duration::hours(1),
178
+ },
179
+ ];
180
+
181
+ let action = AgentAction::Transfer {
182
+ chain_id: 137,
183
+ token: token_b,
184
+ to: recipient_b,
185
+ amount_wei: 15,
186
+ };
187
+
188
+ let result = engine.evaluate(
189
+ &[policy],
190
+ &PolicyAttachment::AllPolicies,
191
+ &action,
192
+ &history,
193
+ agent_key_id,
194
+ now,
195
+ );
196
+
197
+ assert!(matches!(
198
+ result,
199
+ Err(PolicyError::WindowLimitExceeded {
200
+ used_amount_wei: 90,
201
+ requested_amount_wei: 15,
202
+ ..
203
+ })
204
+ ));
205
+ }
206
+
207
+ #[test]
208
+ fn network_scope_is_enforced() {
209
+ let engine = PolicyEngine;
210
+ let token = addr("0x5000000000000000000000000000000000000000");
211
+ let recipient = addr("0x6000000000000000000000000000000000000000");
212
+
213
+ let policy = SpendingPolicy::new(
214
+ 1,
215
+ PolicyType::PerTxMaxSpending,
216
+ 100,
217
+ EntityScope::All,
218
+ EntityScope::All,
219
+ EntityScope::Set(BTreeSet::from([1_u64])),
220
+ )
221
+ .expect("policy");
222
+
223
+ let action = AgentAction::Transfer {
224
+ chain_id: 10,
225
+ token,
226
+ to: recipient,
227
+ amount_wei: 1,
228
+ };
229
+
230
+ let result = engine.evaluate(
231
+ &[policy],
232
+ &PolicyAttachment::AllPolicies,
233
+ &action,
234
+ &[],
235
+ Uuid::new_v4(),
236
+ OffsetDateTime::now_utc(),
237
+ );
238
+
239
+ assert!(matches!(result, Err(PolicyError::NoApplicablePolicies)));
240
+ }
241
+
242
+ #[test]
243
+ fn native_eth_scope_is_enforced() {
244
+ let engine = PolicyEngine;
245
+ let recipient = addr("0x7000000000000000000000000000000000000000");
246
+ let policy = SpendingPolicy::new(
247
+ 1,
248
+ PolicyType::PerTxMaxSpending,
249
+ 100,
250
+ EntityScope::All,
251
+ EntityScope::Set(BTreeSet::from([AssetId::NativeEth])),
252
+ EntityScope::All,
253
+ )
254
+ .expect("policy");
255
+
256
+ let action = AgentAction::Transfer {
257
+ chain_id: 1,
258
+ token: addr("0x8000000000000000000000000000000000000000"),
259
+ to: recipient.clone(),
260
+ amount_wei: 1,
261
+ };
262
+
263
+ let result = engine.evaluate(
264
+ std::slice::from_ref(&policy),
265
+ &PolicyAttachment::AllPolicies,
266
+ &action,
267
+ &[],
268
+ Uuid::new_v4(),
269
+ OffsetDateTime::now_utc(),
270
+ );
271
+ assert!(matches!(result, Err(PolicyError::NoApplicablePolicies)));
272
+
273
+ let native_action = AgentAction::TransferNative {
274
+ chain_id: 1,
275
+ to: recipient,
276
+ amount_wei: 1,
277
+ };
278
+
279
+ let result = engine.evaluate(
280
+ &[policy],
281
+ &PolicyAttachment::AllPolicies,
282
+ &native_action,
283
+ &[],
284
+ Uuid::new_v4(),
285
+ OffsetDateTime::now_utc(),
286
+ );
287
+ assert!(result.is_ok());
288
+ }
289
+
290
+ #[test]
291
+ fn policies_are_evaluated_in_priority_order() {
292
+ let engine = PolicyEngine;
293
+ let token = addr("0x5000000000000000000000000000000000000000");
294
+ let recipient = addr("0x6000000000000000000000000000000000000000");
295
+
296
+ let p1 = SpendingPolicy::new(
297
+ 10,
298
+ PolicyType::PerTxMaxSpending,
299
+ 100,
300
+ EntityScope::All,
301
+ EntityScope::All,
302
+ EntityScope::All,
303
+ )
304
+ .expect("policy must build");
305
+
306
+ let p2 = SpendingPolicy::new(
307
+ 1,
308
+ PolicyType::PerTxMaxSpending,
309
+ 100,
310
+ EntityScope::All,
311
+ EntityScope::All,
312
+ EntityScope::All,
313
+ )
314
+ .expect("policy must build");
315
+
316
+ let action = AgentAction::Transfer {
317
+ chain_id: 1,
318
+ token,
319
+ to: recipient,
320
+ amount_wei: 1,
321
+ };
322
+
323
+ let result = engine
324
+ .evaluate(
325
+ &[p1.clone(), p2.clone()],
326
+ &PolicyAttachment::AllPolicies,
327
+ &action,
328
+ &[],
329
+ Uuid::new_v4(),
330
+ OffsetDateTime::now_utc(),
331
+ )
332
+ .expect("must pass");
333
+
334
+ assert_eq!(result.evaluated_policy_ids, vec![p2.id, p1.id]);
335
+ }
336
+
337
+ #[test]
338
+ fn only_attached_policy_ids_are_used() {
339
+ let engine = PolicyEngine;
340
+ let token = addr("0x5000000000000000000000000000000000000000");
341
+ let recipient = addr("0x6000000000000000000000000000000000000000");
342
+
343
+ let included = SpendingPolicy::new(
344
+ 1,
345
+ PolicyType::PerTxMaxSpending,
346
+ 100,
347
+ EntityScope::All,
348
+ EntityScope::All,
349
+ EntityScope::All,
350
+ )
351
+ .expect("policy must build");
352
+
353
+ let excluded = SpendingPolicy::new(
354
+ 0,
355
+ PolicyType::PerTxMaxSpending,
356
+ 10,
357
+ EntityScope::All,
358
+ EntityScope::All,
359
+ EntityScope::All,
360
+ )
361
+ .expect("policy must build");
362
+
363
+ let mut only_included = BTreeSet::new();
364
+ only_included.insert(included.id);
365
+
366
+ let action = AgentAction::Transfer {
367
+ chain_id: 1,
368
+ token,
369
+ to: recipient,
370
+ amount_wei: 20,
371
+ };
372
+
373
+ let result = engine.evaluate(
374
+ &[included.clone(), excluded],
375
+ &PolicyAttachment::PolicySet(only_included),
376
+ &action,
377
+ &[],
378
+ Uuid::new_v4(),
379
+ OffsetDateTime::now_utc(),
380
+ );
381
+
382
+ let evaluation = result.expect("included policy should pass");
383
+ assert_eq!(evaluation.evaluated_policy_ids, vec![included.id]);
384
+ }
385
+
386
+ #[test]
387
+ fn no_applicable_policy_fails_when_scope_does_not_match() {
388
+ let engine = PolicyEngine;
389
+
390
+ let token = addr("0x7777777777777777777777777777777777777777");
391
+ let recipient = addr("0x8888888888888888888888888888888888888888");
392
+
393
+ let policy = SpendingPolicy::new(
394
+ 0,
395
+ PolicyType::PerTxMaxSpending,
396
+ 100,
397
+ EntityScope::All,
398
+ EntityScope::Set(BTreeSet::from([AssetId::Erc20(addr(
399
+ "0x9999999999999999999999999999999999999999",
400
+ ))])),
401
+ EntityScope::All,
402
+ )
403
+ .expect("policy");
404
+
405
+ let action = AgentAction::Transfer {
406
+ chain_id: 1,
407
+ token,
408
+ to: recipient,
409
+ amount_wei: 1,
410
+ };
411
+
412
+ let result = engine.evaluate(
413
+ &[policy],
414
+ &PolicyAttachment::AllPolicies,
415
+ &action,
416
+ &[],
417
+ Uuid::new_v4(),
418
+ OffsetDateTime::now_utc(),
419
+ );
420
+
421
+ assert!(matches!(result, Err(PolicyError::NoApplicablePolicies)));
422
+ }
423
+
424
+ #[test]
425
+ fn equal_priority_policies_are_ordered_by_id() {
426
+ let engine = PolicyEngine;
427
+ let token = addr("0x1111111111111111111111111111111111111111");
428
+ let recipient = addr("0x2222222222222222222222222222222222222222");
429
+
430
+ let mut high_id = SpendingPolicy::new(
431
+ 5,
432
+ PolicyType::PerTxMaxSpending,
433
+ 100,
434
+ EntityScope::All,
435
+ EntityScope::All,
436
+ EntityScope::All,
437
+ )
438
+ .expect("policy must build");
439
+ high_id.id = Uuid::parse_str("ffffffff-ffff-ffff-ffff-ffffffffffff").expect("uuid");
440
+
441
+ let mut low_id = SpendingPolicy::new(
442
+ 5,
443
+ PolicyType::PerTxMaxSpending,
444
+ 100,
445
+ EntityScope::All,
446
+ EntityScope::All,
447
+ EntityScope::All,
448
+ )
449
+ .expect("policy must build");
450
+ low_id.id = Uuid::parse_str("00000000-0000-0000-0000-000000000001").expect("uuid");
451
+
452
+ let action = AgentAction::Transfer {
453
+ chain_id: 1,
454
+ token,
455
+ to: recipient,
456
+ amount_wei: 1,
457
+ };
458
+
459
+ let result = engine
460
+ .evaluate(
461
+ &[high_id.clone(), low_id.clone()],
462
+ &PolicyAttachment::AllPolicies,
463
+ &action,
464
+ &[],
465
+ Uuid::new_v4(),
466
+ OffsetDateTime::now_utc(),
467
+ )
468
+ .expect("must pass");
469
+
470
+ assert_eq!(result.evaluated_policy_ids, vec![low_id.id, high_id.id]);
471
+ }
472
+
473
+ #[test]
474
+ fn window_limit_fails_closed_on_usage_overflow() {
475
+ let engine = PolicyEngine;
476
+ let now = OffsetDateTime::now_utc();
477
+ let token = addr("0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa");
478
+ let recipient = addr("0xbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb");
479
+
480
+ let policy = SpendingPolicy::new(
481
+ 1,
482
+ PolicyType::DailyMaxSpending,
483
+ 100,
484
+ EntityScope::All,
485
+ EntityScope::All,
486
+ EntityScope::All,
487
+ )
488
+ .expect("policy must build");
489
+
490
+ let agent_key_id = Uuid::new_v4();
491
+ let history = vec![
492
+ SpendEvent {
493
+ agent_key_id,
494
+ chain_id: 1,
495
+ asset: AssetId::Erc20(token.clone()),
496
+ recipient: recipient.clone(),
497
+ amount_wei: u128::MAX,
498
+ at: now - time::Duration::hours(2),
499
+ },
500
+ SpendEvent {
501
+ agent_key_id,
502
+ chain_id: 1,
503
+ asset: AssetId::Erc20(token.clone()),
504
+ recipient: recipient.clone(),
505
+ amount_wei: 1,
506
+ at: now - time::Duration::hours(1),
507
+ },
508
+ ];
509
+
510
+ let action = AgentAction::Transfer {
511
+ chain_id: 1,
512
+ token,
513
+ to: recipient,
514
+ amount_wei: 1,
515
+ };
516
+
517
+ let result = engine.evaluate(
518
+ &[policy],
519
+ &PolicyAttachment::AllPolicies,
520
+ &action,
521
+ &history,
522
+ agent_key_id,
523
+ now,
524
+ );
525
+
526
+ assert!(matches!(
527
+ result,
528
+ Err(PolicyError::WindowLimitExceeded {
529
+ used_amount_wei: u128::MAX,
530
+ requested_amount_wei: 1,
531
+ ..
532
+ })
533
+ ));
534
+ }
535
+
536
+ #[test]
537
+ fn per_chain_gas_limit_is_enforced_for_broadcast_tx() {
538
+ let engine = PolicyEngine;
539
+ let now = OffsetDateTime::now_utc();
540
+ let policy = gas_policy(1, 500_000_000_000_000);
541
+
542
+ let action = AgentAction::BroadcastTx {
543
+ tx: BroadcastTx {
544
+ chain_id: 1,
545
+ nonce: 0,
546
+ to: addr("0x1000000000000000000000000000000000000000"),
547
+ value_wei: 0,
548
+ data_hex: "0x".to_string(),
549
+ gas_limit: 1_000_000,
550
+ max_fee_per_gas_wei: 1_000_000_000,
551
+ max_priority_fee_per_gas_wei: 1_000_000_000,
552
+ tx_type: 0x02,
553
+ delegation_enabled: false,
554
+ },
555
+ };
556
+
557
+ let result = engine.evaluate(
558
+ &[policy],
559
+ &PolicyAttachment::AllPolicies,
560
+ &action,
561
+ &[],
562
+ Uuid::new_v4(),
563
+ now,
564
+ );
565
+
566
+ assert!(matches!(result, Err(PolicyError::GasLimitExceeded { .. })));
567
+ }
568
+
569
+ #[test]
570
+ fn per_chain_gas_limit_allows_broadcast_within_cap() {
571
+ let engine = PolicyEngine;
572
+ let now = OffsetDateTime::now_utc();
573
+ let policy = gas_policy(1, 2_000_000_000_000_000);
574
+
575
+ let action = AgentAction::BroadcastTx {
576
+ tx: BroadcastTx {
577
+ chain_id: 1,
578
+ nonce: 0,
579
+ to: addr("0x1000000000000000000000000000000000000000"),
580
+ value_wei: 0,
581
+ data_hex: "0x".to_string(),
582
+ gas_limit: 1_000_000,
583
+ max_fee_per_gas_wei: 1_000_000_000,
584
+ max_priority_fee_per_gas_wei: 1_000_000_000,
585
+ tx_type: 0x02,
586
+ delegation_enabled: false,
587
+ },
588
+ };
589
+
590
+ let result = engine.evaluate(
591
+ &[policy],
592
+ &PolicyAttachment::AllPolicies,
593
+ &action,
594
+ &[],
595
+ Uuid::new_v4(),
596
+ now,
597
+ );
598
+
599
+ assert!(result.is_ok());
600
+ }
601
+
602
+ #[test]
603
+ fn per_tx_max_fee_per_gas_is_enforced() {
604
+ let engine = PolicyEngine;
605
+ let policy = SpendingPolicy::new(
606
+ 0,
607
+ PolicyType::PerTxMaxFeePerGas,
608
+ 1_000_000_000,
609
+ EntityScope::All,
610
+ EntityScope::All,
611
+ EntityScope::All,
612
+ )
613
+ .expect("policy");
614
+ let action = broadcast_action_with_fees(2_000_000_000, 1_000_000_000, "0x");
615
+
616
+ let result = engine.evaluate(
617
+ &[policy],
618
+ &PolicyAttachment::AllPolicies,
619
+ &action,
620
+ &[],
621
+ Uuid::new_v4(),
622
+ OffsetDateTime::now_utc(),
623
+ );
624
+ assert!(matches!(
625
+ result,
626
+ Err(PolicyError::MaxFeePerGasLimitExceeded { .. })
627
+ ));
628
+ }
629
+
630
+ #[test]
631
+ fn per_tx_max_priority_fee_per_gas_is_enforced() {
632
+ let engine = PolicyEngine;
633
+ let policy = SpendingPolicy::new(
634
+ 0,
635
+ PolicyType::PerTxMaxPriorityFeePerGas,
636
+ 500_000_000,
637
+ EntityScope::All,
638
+ EntityScope::All,
639
+ EntityScope::All,
640
+ )
641
+ .expect("policy");
642
+ let action = broadcast_action_with_fees(2_000_000_000, 1_000_000_000, "0x");
643
+
644
+ let result = engine.evaluate(
645
+ &[policy],
646
+ &PolicyAttachment::AllPolicies,
647
+ &action,
648
+ &[],
649
+ Uuid::new_v4(),
650
+ OffsetDateTime::now_utc(),
651
+ );
652
+ assert!(matches!(
653
+ result,
654
+ Err(PolicyError::PriorityFeePerGasLimitExceeded { .. })
655
+ ));
656
+ }
657
+
658
+ #[test]
659
+ fn per_tx_max_calldata_bytes_is_enforced() {
660
+ let engine = PolicyEngine;
661
+ let policy = SpendingPolicy::new(
662
+ 0,
663
+ PolicyType::PerTxMaxCalldataBytes,
664
+ 3,
665
+ EntityScope::All,
666
+ EntityScope::All,
667
+ EntityScope::All,
668
+ )
669
+ .expect("policy");
670
+ let action = broadcast_action_with_fees(2_000_000_000, 1_000_000_000, "0xdeadbeef");
671
+
672
+ let result = engine.evaluate(
673
+ &[policy],
674
+ &PolicyAttachment::AllPolicies,
675
+ &action,
676
+ &[],
677
+ Uuid::new_v4(),
678
+ OffsetDateTime::now_utc(),
679
+ );
680
+ assert!(matches!(
681
+ result,
682
+ Err(PolicyError::CalldataBytesLimitExceeded { .. })
683
+ ));
684
+ }
685
+
686
+ #[test]
687
+ fn daily_tx_count_limit_is_enforced() {
688
+ let engine = PolicyEngine;
689
+ let now = OffsetDateTime::now_utc();
690
+ let recipient = addr("0x2000000000000000000000000000000000000000");
691
+ let token = addr("0x3000000000000000000000000000000000000000");
692
+ let policy = SpendingPolicy::new(
693
+ 0,
694
+ PolicyType::DailyMaxTxCount,
695
+ 2,
696
+ EntityScope::All,
697
+ EntityScope::All,
698
+ EntityScope::All,
699
+ )
700
+ .expect("policy");
701
+ let agent_key_id = Uuid::new_v4();
702
+ let history = vec![
703
+ SpendEvent {
704
+ agent_key_id,
705
+ chain_id: 1,
706
+ asset: AssetId::Erc20(token.clone()),
707
+ recipient: recipient.clone(),
708
+ amount_wei: 1,
709
+ at: now - Duration::hours(2),
710
+ },
711
+ SpendEvent {
712
+ agent_key_id,
713
+ chain_id: 1,
714
+ asset: AssetId::Erc20(token.clone()),
715
+ recipient: recipient.clone(),
716
+ amount_wei: 1,
717
+ at: now - Duration::hours(1),
718
+ },
719
+ ];
720
+ let action = AgentAction::Transfer {
721
+ chain_id: 1,
722
+ token,
723
+ to: recipient,
724
+ amount_wei: 1,
725
+ };
726
+
727
+ let result = engine.evaluate(
728
+ &[policy],
729
+ &PolicyAttachment::AllPolicies,
730
+ &action,
731
+ &history,
732
+ agent_key_id,
733
+ now,
734
+ );
735
+ assert!(matches!(
736
+ result,
737
+ Err(PolicyError::TxCountLimitExceeded { .. })
738
+ ));
739
+ }
740
+
741
+ #[test]
742
+ fn fee_policy_is_not_applicable_when_action_lacks_metadata() {
743
+ let engine = PolicyEngine;
744
+ let policy = SpendingPolicy::new(
745
+ 0,
746
+ PolicyType::PerTxMaxFeePerGas,
747
+ 1_000_000_000,
748
+ EntityScope::All,
749
+ EntityScope::All,
750
+ EntityScope::All,
751
+ )
752
+ .expect("policy");
753
+ let action = AgentAction::Transfer {
754
+ chain_id: 1,
755
+ token: addr("0x1000000000000000000000000000000000000000"),
756
+ to: addr("0x2000000000000000000000000000000000000000"),
757
+ amount_wei: 1,
758
+ };
759
+
760
+ let result = engine.evaluate(
761
+ &[policy],
762
+ &PolicyAttachment::AllPolicies,
763
+ &action,
764
+ &[],
765
+ Uuid::new_v4(),
766
+ OffsetDateTime::now_utc(),
767
+ );
768
+ assert!(result.is_ok());
769
+ }
770
+
771
+ #[test]
772
+ fn gas_policy_is_not_applicable_when_action_lacks_metadata() {
773
+ let engine = PolicyEngine;
774
+ let policy = SpendingPolicy::new(
775
+ 0,
776
+ PolicyType::PerChainMaxGasSpend,
777
+ 1_000_000_000_000_000,
778
+ EntityScope::All,
779
+ EntityScope::All,
780
+ EntityScope::All,
781
+ )
782
+ .expect("policy");
783
+ let action = AgentAction::Transfer {
784
+ chain_id: 1,
785
+ token: addr("0x1000000000000000000000000000000000000000"),
786
+ to: addr("0x2000000000000000000000000000000000000000"),
787
+ amount_wei: 1,
788
+ };
789
+
790
+ let result = engine.evaluate(
791
+ &[policy],
792
+ &PolicyAttachment::AllPolicies,
793
+ &action,
794
+ &[],
795
+ Uuid::new_v4(),
796
+ OffsetDateTime::now_utc(),
797
+ );
798
+ assert!(result.is_ok());
799
+ }
800
+
801
+ #[test]
802
+ fn permit2_per_tx_limit_is_enforced() {
803
+ let engine = PolicyEngine;
804
+ let action = AgentAction::Permit2Permit {
805
+ permit: vault_domain::Permit2Permit {
806
+ chain_id: 1,
807
+ permit2_contract: addr("0x000000000022d473030f116ddee9f6b43ac78ba3"),
808
+ token: addr("0x1000000000000000000000000000000000000000"),
809
+ spender: addr("0x2000000000000000000000000000000000000000"),
810
+ amount_wei: 101,
811
+ expiration: 100,
812
+ nonce: 1,
813
+ sig_deadline: 200,
814
+ },
815
+ };
816
+ let result = engine.evaluate(
817
+ &[policy_all_per_tx(100)],
818
+ &PolicyAttachment::AllPolicies,
819
+ &action,
820
+ &[],
821
+ Uuid::new_v4(),
822
+ OffsetDateTime::now_utc(),
823
+ );
824
+ assert!(matches!(
825
+ result,
826
+ Err(PolicyError::PerTxLimitExceeded { .. })
827
+ ));
828
+ }
829
+
830
+ #[test]
831
+ fn eip3009_broadcast_scope_is_enforced() {
832
+ let engine = PolicyEngine;
833
+ let token = addr("0x3000000000000000000000000000000000000000");
834
+ let allowed_recipient = addr("0x4000000000000000000000000000000000000000");
835
+ let denied_recipient = addr("0x5050505050505050505050505050505050505050");
836
+ let policy = SpendingPolicy::new(
837
+ 1,
838
+ PolicyType::PerTxMaxSpending,
839
+ 1_000,
840
+ EntityScope::Set(BTreeSet::from([allowed_recipient.clone()])),
841
+ EntityScope::Set(BTreeSet::from([AssetId::Erc20(token.clone())])),
842
+ EntityScope::Set(BTreeSet::from([1_u64])),
843
+ )
844
+ .expect("policy");
845
+ let calldata = {
846
+ use alloy_sol_types::{sol, SolCall};
847
+ sol! {
848
+ function transferWithAuthorization(address from, address to, uint256 value, uint256 validAfter, uint256 validBefore, bytes32 nonce, bytes signature);
849
+ }
850
+ transferWithAuthorizationCall {
851
+ from: alloy_primitives::Address::from([0x11; 20]),
852
+ to: alloy_primitives::Address::from([0x50; 20]),
853
+ value: alloy_primitives::U256::from(10u64),
854
+ validAfter: alloy_primitives::U256::from(1u64),
855
+ validBefore: alloy_primitives::U256::from(2u64),
856
+ nonce: [0x77; 32].into(),
857
+ signature: vec![0x12].into(),
858
+ }
859
+ .abi_encode()
860
+ };
861
+ let action = AgentAction::BroadcastTx {
862
+ tx: BroadcastTx {
863
+ chain_id: 1,
864
+ nonce: 0,
865
+ to: token,
866
+ value_wei: 0,
867
+ data_hex: format!(
868
+ "0x{}",
869
+ calldata
870
+ .iter()
871
+ .map(|byte| format!("{byte:02x}"))
872
+ .collect::<String>()
873
+ ),
874
+ gas_limit: 50_000,
875
+ max_fee_per_gas_wei: 1_000_000_000,
876
+ max_priority_fee_per_gas_wei: 1_000_000_000,
877
+ tx_type: 0x02,
878
+ delegation_enabled: false,
879
+ },
880
+ };
881
+ let result = engine.evaluate(
882
+ &[policy],
883
+ &PolicyAttachment::AllPolicies,
884
+ &action,
885
+ &[],
886
+ Uuid::new_v4(),
887
+ OffsetDateTime::now_utc(),
888
+ );
889
+ assert!(matches!(result, Err(PolicyError::NoApplicablePolicies)));
890
+ assert_eq!(action.recipient(), denied_recipient);
891
+ }