@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,25 @@
1
+ [package]
2
+ name = "vault-transport-xpc"
3
+ version.workspace = true
4
+ edition.workspace = true
5
+ license.workspace = true
6
+ authors.workspace = true
7
+
8
+ [dependencies]
9
+ async-trait.workspace = true
10
+ block.workspace = true
11
+ libc.workspace = true
12
+ serde.workspace = true
13
+ serde_json.workspace = true
14
+ thiserror.workspace = true
15
+ time.workspace = true
16
+ tokio.workspace = true
17
+ uuid.workspace = true
18
+ zeroize.workspace = true
19
+ vault-daemon = { path = "../vault-daemon" }
20
+ vault-domain = { path = "../vault-domain" }
21
+ vault-policy = { path = "../vault-policy" }
22
+ vault-signer = { path = "../vault-signer" }
23
+
24
+ [target.'cfg(target_os = "macos")'.dependencies]
25
+ security-framework.workspace = true
@@ -0,0 +1,635 @@
1
+ pub struct XpcDaemonClient {
2
+ connection: XpcConnection,
3
+ responses: Mutex<Receiver<IncomingWireMessage>>,
4
+ timeout: Duration,
5
+ _client_block: RcBlock<(XpcObject,), ()>,
6
+ }
7
+
8
+ // SAFETY: Apple XPC connections are reference-counted OS objects designed for
9
+ // cross-thread use. Access to response receiver is synchronized via `Mutex`.
10
+ #[cfg(target_os = "macos")]
11
+ unsafe impl Send for XpcDaemonClient {}
12
+ // SAFETY: methods requiring interior mutation synchronize shared state.
13
+ #[cfg(target_os = "macos")]
14
+ unsafe impl Sync for XpcDaemonClient {}
15
+
16
+ #[cfg(target_os = "macos")]
17
+ impl Drop for XpcDaemonClient {
18
+ fn drop(&mut self) {
19
+ // SAFETY: connection is owned by this instance.
20
+ unsafe {
21
+ xpc_release(self.connection);
22
+ }
23
+ }
24
+ }
25
+
26
+ #[cfg(target_os = "macos")]
27
+ impl XpcDaemonClient {
28
+ /// Connects to daemon endpoint.
29
+ pub fn connect(endpoint: &XpcEndpoint, timeout: Duration) -> Result<Self, XpcTransportError> {
30
+ // SAFETY: endpoint comes from xpc_endpoint_create.
31
+ unsafe {
32
+ let connection = xpc_connection_create_from_endpoint(endpoint.raw);
33
+ if connection.is_null() {
34
+ return Err(XpcTransportError::Internal(
35
+ "xpc_connection_create_from_endpoint returned null".to_string(),
36
+ ));
37
+ }
38
+
39
+ let (tx, rx) = mpsc::channel::<IncomingWireMessage>();
40
+ let client_block = ConcreteBlock::new(move |message: XpcObject| {
41
+ if !ptr::eq(xpc_get_type(message), &_xpc_type_dictionary) {
42
+ return;
43
+ }
44
+ let incoming = match decode_wire_response(message) {
45
+ Ok(resp) => IncomingWireMessage::Response(resp),
46
+ Err(err) => IncomingWireMessage::DecodeError(err),
47
+ };
48
+ if tx.send(incoming).is_err() {
49
+ // Receiver is dropped; nothing to do.
50
+ }
51
+ })
52
+ .copy();
53
+
54
+ xpc_connection_set_event_handler(
55
+ connection,
56
+ &*client_block as *const Block<_, _> as *mut c_void,
57
+ );
58
+ xpc_connection_resume(connection);
59
+
60
+ xpc_retain(connection);
61
+
62
+ Ok(Self {
63
+ connection,
64
+ responses: Mutex::new(rx),
65
+ timeout,
66
+ _client_block: client_block,
67
+ })
68
+ }
69
+ }
70
+
71
+ /// Performs a single RPC call.
72
+ pub fn call_rpc(
73
+ &self,
74
+ request: DaemonRpcRequest,
75
+ ) -> Result<DaemonRpcResponse, XpcTransportError> {
76
+ let mut request = request;
77
+ let responses = self
78
+ .responses
79
+ .lock()
80
+ .map_err(|_| XpcTransportError::Internal("responses lock poisoned".to_string()))?;
81
+
82
+ let request_id = Uuid::new_v4().to_string();
83
+ let body_json = serde_json::to_string(&request)
84
+ .map_err(|err| XpcTransportError::Serialization(err.to_string()));
85
+ request.zeroize_secrets();
86
+ let mut body_json = body_json?;
87
+ if let Err(err) = validate_wire_lengths(&request_id, &body_json) {
88
+ body_json.zeroize();
89
+ return Err(err);
90
+ }
91
+
92
+ let mut wire_request = WireRequest {
93
+ request_id: request_id.clone(),
94
+ body_json,
95
+ };
96
+ let xpc_request = encode_wire_request(&wire_request);
97
+ wire_request.body_json.zeroize();
98
+ let xpc_request = xpc_request?;
99
+
100
+ // SAFETY: connection is resumed and message is a valid dictionary.
101
+ unsafe {
102
+ xpc_connection_send_message(self.connection, xpc_request);
103
+ xpc_release(xpc_request);
104
+ }
105
+
106
+ // Keep send+receive in a single critical section so only one request is
107
+ // in flight per connection; this prevents response/request-id mixups.
108
+ let deadline = std::time::Instant::now() + self.timeout;
109
+ let mut response = recv_matching_response(&responses, &request_id, deadline)?;
110
+
111
+ if response.ok {
112
+ let parsed = serde_json::from_str(&response.body_json)
113
+ .map_err(|err| XpcTransportError::Serialization(err.to_string()));
114
+ response.body_json.zeroize();
115
+ parsed
116
+ } else {
117
+ let daemon_error = match serde_json::from_str::<WireDaemonError>(&response.body_json) {
118
+ Ok(err) => Err(XpcTransportError::Daemon(err.into_daemon_error())),
119
+ Err(_) => Err(XpcTransportError::Daemon(DaemonError::Transport(
120
+ response.body_json.clone(),
121
+ ))),
122
+ };
123
+ response.body_json.zeroize();
124
+ daemon_error
125
+ }
126
+ }
127
+ }
128
+
129
+ #[cfg(target_os = "macos")]
130
+ fn recv_matching_response(
131
+ responses: &Receiver<IncomingWireMessage>,
132
+ request_id: &str,
133
+ deadline: std::time::Instant,
134
+ ) -> Result<WireResponse, XpcTransportError> {
135
+ loop {
136
+ let now = std::time::Instant::now();
137
+ if now >= deadline {
138
+ return Err(XpcTransportError::Timeout);
139
+ }
140
+
141
+ let remaining = deadline.saturating_duration_since(now);
142
+ let incoming = responses
143
+ .recv_timeout(remaining)
144
+ .map_err(|_| XpcTransportError::Timeout)?;
145
+
146
+ match incoming {
147
+ IncomingWireMessage::DecodeError(err) => return Err(err),
148
+ IncomingWireMessage::Response(response) => {
149
+ if response.request_id == request_id {
150
+ return Ok(response);
151
+ }
152
+ }
153
+ }
154
+ }
155
+ }
156
+
157
+ fn serialize_wire_daemon_error(err: DaemonError) -> String {
158
+ match serde_json::to_string(&WireDaemonError::from(err)) {
159
+ Ok(json) => json,
160
+ Err(serialize_err) => format!("failed to serialize daemon error: {serialize_err}"),
161
+ }
162
+ }
163
+
164
+ #[cfg(target_os = "macos")]
165
+ fn encode_wire_request(req: &WireRequest) -> Result<XpcObject, XpcTransportError> {
166
+ let dict = unsafe { xpc_dictionary_create(ptr::null(), ptr::null(), 0) };
167
+ if dict.is_null() {
168
+ return Err(XpcTransportError::Internal(
169
+ "unable to allocate xpc dictionary".to_string(),
170
+ ));
171
+ }
172
+
173
+ set_dict_string(dict, "wlfi_kind", "request")?;
174
+ set_dict_string(dict, "wlfi_request_id", &req.request_id)?;
175
+ set_dict_string(dict, "wlfi_body", &req.body_json)?;
176
+ Ok(dict)
177
+ }
178
+
179
+ #[cfg(target_os = "macos")]
180
+ fn decode_wire_request(message: XpcObject) -> Result<WireRequest, XpcTransportError> {
181
+ let kind = get_dict_string(message, "wlfi_kind")?;
182
+ if kind != "request" {
183
+ return Err(XpcTransportError::Protocol(
184
+ "unexpected request kind".to_string(),
185
+ ));
186
+ }
187
+
188
+ let request_id = get_dict_string(message, "wlfi_request_id")?;
189
+ let body_json = get_dict_string(message, "wlfi_body")?;
190
+ validate_wire_lengths(&request_id, &body_json)?;
191
+
192
+ Ok(WireRequest {
193
+ request_id,
194
+ body_json,
195
+ })
196
+ }
197
+
198
+ #[cfg(target_os = "macos")]
199
+ fn encode_wire_response(resp: &WireResponse) -> Result<XpcObject, XpcTransportError> {
200
+ let dict = unsafe { xpc_dictionary_create(ptr::null(), ptr::null(), 0) };
201
+ if dict.is_null() {
202
+ return Err(XpcTransportError::Internal(
203
+ "unable to allocate xpc dictionary".to_string(),
204
+ ));
205
+ }
206
+
207
+ set_dict_string(dict, "wlfi_kind", "response")?;
208
+ set_dict_string(dict, "wlfi_request_id", &resp.request_id)?;
209
+ set_dict_bool(dict, "wlfi_ok", resp.ok)?;
210
+ set_dict_string(dict, "wlfi_body", &resp.body_json)?;
211
+ Ok(dict)
212
+ }
213
+
214
+ #[cfg(target_os = "macos")]
215
+ fn decode_wire_response(message: XpcObject) -> Result<WireResponse, XpcTransportError> {
216
+ let kind = get_dict_string(message, "wlfi_kind")?;
217
+ if kind != "response" {
218
+ return Err(XpcTransportError::Protocol(
219
+ "unexpected response kind".to_string(),
220
+ ));
221
+ }
222
+
223
+ let request_id = get_dict_string(message, "wlfi_request_id")?;
224
+ let body_json = get_dict_string(message, "wlfi_body")?;
225
+ validate_wire_lengths(&request_id, &body_json)?;
226
+
227
+ Ok(WireResponse {
228
+ request_id,
229
+ ok: get_dict_bool(message, "wlfi_ok")?,
230
+ body_json,
231
+ })
232
+ }
233
+
234
+ fn validate_wire_lengths(request_id: &str, body_json: &str) -> Result<(), XpcTransportError> {
235
+ if request_id.len() > MAX_WIRE_REQUEST_ID_BYTES {
236
+ return Err(XpcTransportError::Protocol(format!(
237
+ "wire request id exceeds max bytes ({MAX_WIRE_REQUEST_ID_BYTES})"
238
+ )));
239
+ }
240
+ if body_json.len() > MAX_WIRE_BODY_BYTES {
241
+ return Err(XpcTransportError::Protocol(format!(
242
+ "wire body exceeds max bytes ({MAX_WIRE_BODY_BYTES})"
243
+ )));
244
+ }
245
+ Ok(())
246
+ }
247
+
248
+ fn enforce_wire_response_limits(response: WireResponse) -> WireResponse {
249
+ let validation_err = match validate_wire_lengths(&response.request_id, &response.body_json) {
250
+ Ok(()) => return response,
251
+ Err(err) => err,
252
+ };
253
+
254
+ let request_id = if response.request_id.len() <= MAX_WIRE_REQUEST_ID_BYTES {
255
+ response.request_id
256
+ } else {
257
+ Uuid::nil().to_string()
258
+ };
259
+
260
+ let body_json = serde_json::to_string(&WireDaemonError::Transport(validation_err.to_string()))
261
+ .unwrap_or_else(|_| {
262
+ r#"{"kind":"Transport","data":"protocol error: oversized response"}"#.to_string()
263
+ });
264
+
265
+ WireResponse {
266
+ request_id,
267
+ ok: false,
268
+ body_json,
269
+ }
270
+ }
271
+
272
+ #[cfg(target_os = "macos")]
273
+ fn extract_safe_request_id(message: XpcObject) -> String {
274
+ get_dict_string(message, "wlfi_request_id")
275
+ .ok()
276
+ .filter(|id| id.len() <= MAX_WIRE_REQUEST_ID_BYTES)
277
+ .unwrap_or_else(|| Uuid::nil().to_string())
278
+ }
279
+
280
+ #[cfg(target_os = "macos")]
281
+ fn set_dict_string(dict: XpcObject, key: &str, value: &str) -> Result<(), XpcTransportError> {
282
+ let key = CString::new(key).map_err(|err| XpcTransportError::Protocol(err.to_string()))?;
283
+ let value = CString::new(value).map_err(|err| XpcTransportError::Protocol(err.to_string()))?;
284
+ unsafe {
285
+ xpc_dictionary_set_string(dict, key.as_ptr(), value.as_ptr());
286
+ }
287
+ Ok(())
288
+ }
289
+
290
+ #[cfg(target_os = "macos")]
291
+ fn set_dict_bool(dict: XpcObject, key: &str, value: bool) -> Result<(), XpcTransportError> {
292
+ let key = CString::new(key).map_err(|err| XpcTransportError::Protocol(err.to_string()))?;
293
+ unsafe {
294
+ xpc_dictionary_set_bool(dict, key.as_ptr(), value);
295
+ }
296
+ Ok(())
297
+ }
298
+
299
+ #[cfg(target_os = "macos")]
300
+ fn get_dict_string(dict: XpcObject, key: &str) -> Result<String, XpcTransportError> {
301
+ let key = CString::new(key).map_err(|err| XpcTransportError::Protocol(err.to_string()))?;
302
+ let value = unsafe { xpc_dictionary_get_string(dict, key.as_ptr()) };
303
+ if value.is_null() {
304
+ return Err(XpcTransportError::Protocol(format!("missing key: {key:?}")));
305
+ }
306
+
307
+ Ok(unsafe { CStr::from_ptr(value) }
308
+ .to_string_lossy()
309
+ .to_string())
310
+ }
311
+
312
+ #[cfg(target_os = "macos")]
313
+ fn get_dict_bool(dict: XpcObject, key: &str) -> Result<bool, XpcTransportError> {
314
+ let key = CString::new(key).map_err(|err| XpcTransportError::Protocol(err.to_string()))?;
315
+ Ok(unsafe { xpc_dictionary_get_bool(dict, key.as_ptr()) })
316
+ }
317
+
318
+ /// Agent/client-side adapter backed by XPC transport.
319
+ #[cfg(target_os = "macos")]
320
+ #[async_trait]
321
+ impl KeyManagerDaemonApi for XpcDaemonClient {
322
+ async fn issue_lease(&self, vault_password: &str) -> Result<Lease, DaemonError> {
323
+ match self.call_rpc(DaemonRpcRequest::IssueLease {
324
+ vault_password: vault_password.to_string(),
325
+ }) {
326
+ Ok(DaemonRpcResponse::Lease(lease)) => Ok(lease),
327
+ Ok(_) => Err(DaemonError::Transport(
328
+ "unexpected response type".to_string(),
329
+ )),
330
+ Err(XpcTransportError::Daemon(err)) => Err(err),
331
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
332
+ }
333
+ }
334
+
335
+ async fn add_policy(
336
+ &self,
337
+ session: &AdminSession,
338
+ policy: SpendingPolicy,
339
+ ) -> Result<(), DaemonError> {
340
+ match self.call_rpc(DaemonRpcRequest::AddPolicy {
341
+ session: session.clone(),
342
+ policy,
343
+ }) {
344
+ Ok(DaemonRpcResponse::Unit) => Ok(()),
345
+ Ok(_) => Err(DaemonError::Transport(
346
+ "unexpected response type".to_string(),
347
+ )),
348
+ Err(XpcTransportError::Daemon(err)) => Err(err),
349
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
350
+ }
351
+ }
352
+
353
+ async fn list_policies(
354
+ &self,
355
+ session: &AdminSession,
356
+ ) -> Result<Vec<SpendingPolicy>, DaemonError> {
357
+ match self.call_rpc(DaemonRpcRequest::ListPolicies {
358
+ session: session.clone(),
359
+ }) {
360
+ Ok(DaemonRpcResponse::Policies(policies)) => Ok(policies),
361
+ Ok(_) => Err(DaemonError::Transport(
362
+ "unexpected response type".to_string(),
363
+ )),
364
+ Err(XpcTransportError::Daemon(err)) => Err(err),
365
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
366
+ }
367
+ }
368
+
369
+ async fn disable_policy(
370
+ &self,
371
+ session: &AdminSession,
372
+ policy_id: Uuid,
373
+ ) -> Result<(), DaemonError> {
374
+ match self.call_rpc(DaemonRpcRequest::DisablePolicy {
375
+ session: session.clone(),
376
+ policy_id,
377
+ }) {
378
+ Ok(DaemonRpcResponse::Unit) => Ok(()),
379
+ Ok(_) => Err(DaemonError::Transport(
380
+ "unexpected response type".to_string(),
381
+ )),
382
+ Err(XpcTransportError::Daemon(err)) => Err(err),
383
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
384
+ }
385
+ }
386
+
387
+ async fn create_vault_key(
388
+ &self,
389
+ session: &AdminSession,
390
+ request: KeyCreateRequest,
391
+ ) -> Result<VaultKey, DaemonError> {
392
+ match self.call_rpc(DaemonRpcRequest::CreateVaultKey {
393
+ session: session.clone(),
394
+ request,
395
+ }) {
396
+ Ok(DaemonRpcResponse::VaultKey(key)) => Ok(key),
397
+ Ok(_) => Err(DaemonError::Transport(
398
+ "unexpected response type".to_string(),
399
+ )),
400
+ Err(XpcTransportError::Daemon(err)) => Err(err),
401
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
402
+ }
403
+ }
404
+
405
+ async fn create_agent_key(
406
+ &self,
407
+ session: &AdminSession,
408
+ vault_key_id: Uuid,
409
+ attachment: PolicyAttachment,
410
+ ) -> Result<AgentCredentials, DaemonError> {
411
+ match self.call_rpc(DaemonRpcRequest::CreateAgentKey {
412
+ session: session.clone(),
413
+ vault_key_id,
414
+ attachment,
415
+ }) {
416
+ Ok(DaemonRpcResponse::AgentCredentials(agent_credentials)) => Ok(agent_credentials),
417
+ Ok(_) => Err(DaemonError::Transport(
418
+ "unexpected response type".to_string(),
419
+ )),
420
+ Err(XpcTransportError::Daemon(err)) => Err(err),
421
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
422
+ }
423
+ }
424
+
425
+ async fn export_vault_private_key(
426
+ &self,
427
+ session: &AdminSession,
428
+ vault_key_id: Uuid,
429
+ ) -> Result<Option<String>, DaemonError> {
430
+ match self.call_rpc(DaemonRpcRequest::ExportVaultPrivateKey {
431
+ session: session.clone(),
432
+ vault_key_id,
433
+ }) {
434
+ Ok(DaemonRpcResponse::PrivateKey(private_key)) => Ok(private_key),
435
+ Ok(_) => Err(DaemonError::Transport(
436
+ "unexpected response type".to_string(),
437
+ )),
438
+ Err(XpcTransportError::Daemon(err)) => Err(err),
439
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
440
+ }
441
+ }
442
+
443
+ async fn rotate_agent_auth_token(
444
+ &self,
445
+ session: &AdminSession,
446
+ agent_key_id: Uuid,
447
+ ) -> Result<String, DaemonError> {
448
+ match self.call_rpc(DaemonRpcRequest::RotateAgentAuthToken {
449
+ session: session.clone(),
450
+ agent_key_id,
451
+ }) {
452
+ Ok(DaemonRpcResponse::AuthToken(token)) => Ok(token),
453
+ Ok(_) => Err(DaemonError::Transport(
454
+ "unexpected response type".to_string(),
455
+ )),
456
+ Err(XpcTransportError::Daemon(err)) => Err(err),
457
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
458
+ }
459
+ }
460
+
461
+ async fn revoke_agent_key(
462
+ &self,
463
+ session: &AdminSession,
464
+ agent_key_id: Uuid,
465
+ ) -> Result<(), DaemonError> {
466
+ match self.call_rpc(DaemonRpcRequest::RevokeAgentKey {
467
+ session: session.clone(),
468
+ agent_key_id,
469
+ }) {
470
+ Ok(DaemonRpcResponse::Unit) => Ok(()),
471
+ Ok(_) => Err(DaemonError::Transport(
472
+ "unexpected response type".to_string(),
473
+ )),
474
+ Err(XpcTransportError::Daemon(err)) => Err(err),
475
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
476
+ }
477
+ }
478
+
479
+ async fn list_manual_approval_requests(
480
+ &self,
481
+ session: &AdminSession,
482
+ ) -> Result<Vec<ManualApprovalRequest>, DaemonError> {
483
+ match self.call_rpc(DaemonRpcRequest::ListManualApprovalRequests {
484
+ session: session.clone(),
485
+ }) {
486
+ Ok(DaemonRpcResponse::ManualApprovalRequests(requests)) => Ok(requests),
487
+ Ok(_) => Err(DaemonError::Transport(
488
+ "unexpected response type".to_string(),
489
+ )),
490
+ Err(XpcTransportError::Daemon(err)) => Err(err),
491
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
492
+ }
493
+ }
494
+
495
+ async fn decide_manual_approval_request(
496
+ &self,
497
+ session: &AdminSession,
498
+ approval_request_id: Uuid,
499
+ decision: ManualApprovalDecision,
500
+ rejection_reason: Option<String>,
501
+ ) -> Result<ManualApprovalRequest, DaemonError> {
502
+ match self.call_rpc(DaemonRpcRequest::DecideManualApprovalRequest {
503
+ session: session.clone(),
504
+ approval_request_id,
505
+ decision,
506
+ rejection_reason,
507
+ }) {
508
+ Ok(DaemonRpcResponse::ManualApprovalRequest(request)) => Ok(request),
509
+ Ok(_) => Err(DaemonError::Transport(
510
+ "unexpected response type".to_string(),
511
+ )),
512
+ Err(XpcTransportError::Daemon(err)) => Err(err),
513
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
514
+ }
515
+ }
516
+
517
+ async fn set_relay_config(
518
+ &self,
519
+ session: &AdminSession,
520
+ relay_url: Option<String>,
521
+ frontend_url: Option<String>,
522
+ ) -> Result<RelayConfig, DaemonError> {
523
+ match self.call_rpc(DaemonRpcRequest::SetRelayConfig {
524
+ session: session.clone(),
525
+ relay_url,
526
+ frontend_url,
527
+ }) {
528
+ Ok(DaemonRpcResponse::RelayConfig(relay_config)) => Ok(relay_config),
529
+ Ok(_) => Err(DaemonError::Transport(
530
+ "unexpected response type".to_string(),
531
+ )),
532
+ Err(XpcTransportError::Daemon(err)) => Err(err),
533
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
534
+ }
535
+ }
536
+
537
+ async fn get_relay_config(&self, session: &AdminSession) -> Result<RelayConfig, DaemonError> {
538
+ match self.call_rpc(DaemonRpcRequest::GetRelayConfig {
539
+ session: session.clone(),
540
+ }) {
541
+ Ok(DaemonRpcResponse::RelayConfig(relay_config)) => Ok(relay_config),
542
+ Ok(_) => Err(DaemonError::Transport(
543
+ "unexpected response type".to_string(),
544
+ )),
545
+ Err(XpcTransportError::Daemon(err)) => Err(err),
546
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
547
+ }
548
+ }
549
+
550
+ async fn evaluate_for_agent(
551
+ &self,
552
+ request: SignRequest,
553
+ ) -> Result<PolicyEvaluation, DaemonError> {
554
+ match self.call_rpc(DaemonRpcRequest::EvaluateForAgent { request }) {
555
+ Ok(DaemonRpcResponse::PolicyEvaluation(evaluation)) => Ok(evaluation),
556
+ Ok(_) => Err(DaemonError::Transport(
557
+ "unexpected response type".to_string(),
558
+ )),
559
+ Err(XpcTransportError::Daemon(err)) => Err(err),
560
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
561
+ }
562
+ }
563
+
564
+ async fn explain_for_agent(
565
+ &self,
566
+ request: SignRequest,
567
+ ) -> Result<PolicyExplanation, DaemonError> {
568
+ match self.call_rpc(DaemonRpcRequest::ExplainForAgent { request }) {
569
+ Ok(DaemonRpcResponse::PolicyExplanation(explanation)) => Ok(explanation),
570
+ Ok(_) => Err(DaemonError::Transport(
571
+ "unexpected response type".to_string(),
572
+ )),
573
+ Err(XpcTransportError::Daemon(err)) => Err(err),
574
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
575
+ }
576
+ }
577
+
578
+ async fn reserve_nonce(
579
+ &self,
580
+ request: NonceReservationRequest,
581
+ ) -> Result<NonceReservation, DaemonError> {
582
+ match self.call_rpc(DaemonRpcRequest::ReserveNonce { request }) {
583
+ Ok(DaemonRpcResponse::NonceReservation(reservation)) => Ok(reservation),
584
+ Ok(_) => Err(DaemonError::Transport(
585
+ "unexpected response type".to_string(),
586
+ )),
587
+ Err(XpcTransportError::Daemon(err)) => Err(err),
588
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
589
+ }
590
+ }
591
+
592
+ async fn release_nonce(&self, request: NonceReleaseRequest) -> Result<(), DaemonError> {
593
+ match self.call_rpc(DaemonRpcRequest::ReleaseNonce { request }) {
594
+ Ok(DaemonRpcResponse::Unit) => Ok(()),
595
+ Ok(_) => Err(DaemonError::Transport(
596
+ "unexpected response type".to_string(),
597
+ )),
598
+ Err(XpcTransportError::Daemon(err)) => Err(err),
599
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
600
+ }
601
+ }
602
+
603
+ async fn sign_for_agent(&self, request: SignRequest) -> Result<Signature, DaemonError> {
604
+ match self.call_rpc(DaemonRpcRequest::SignForAgent { request }) {
605
+ Ok(DaemonRpcResponse::Signature(sig)) => Ok(sig),
606
+ Ok(_) => Err(DaemonError::Transport(
607
+ "unexpected response type".to_string(),
608
+ )),
609
+ Err(XpcTransportError::Daemon(err)) => Err(err),
610
+ Err(err) => Err(DaemonError::Transport(err.to_string())),
611
+ }
612
+ }
613
+ }
614
+
615
+ /// Non-macOS fallback server type.
616
+ #[cfg(not(target_os = "macos"))]
617
+ pub struct XpcDaemonServer;
618
+
619
+ /// Non-macOS fallback client type.
620
+ #[cfg(not(target_os = "macos"))]
621
+ pub struct XpcDaemonClient;
622
+
623
+ #[cfg(not(target_os = "macos"))]
624
+ impl XpcDaemonServer {
625
+ /// Always returns unsupported on non-macOS.
626
+ pub fn start_inmemory<B>(
627
+ _daemon: Arc<InMemoryDaemon<B>>,
628
+ _runtime_handle: Handle,
629
+ ) -> Result<Self, XpcTransportError>
630
+ where
631
+ B: VaultSignerBackend + 'static,
632
+ {
633
+ Err(XpcTransportError::UnsupportedPlatform)
634
+ }
635
+ }