arc402-cli 0.2.0

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 (308) hide show
  1. package/README.md +245 -0
  2. package/dist/abis.d.ts +19 -0
  3. package/dist/abis.d.ts.map +1 -0
  4. package/dist/abis.js +177 -0
  5. package/dist/abis.js.map +1 -0
  6. package/dist/bundler.d.ts +65 -0
  7. package/dist/bundler.d.ts.map +1 -0
  8. package/dist/bundler.js +181 -0
  9. package/dist/bundler.js.map +1 -0
  10. package/dist/client.d.ts +14 -0
  11. package/dist/client.d.ts.map +1 -0
  12. package/dist/client.js +24 -0
  13. package/dist/client.js.map +1 -0
  14. package/dist/coinbase-smart-wallet.d.ts +28 -0
  15. package/dist/coinbase-smart-wallet.d.ts.map +1 -0
  16. package/dist/coinbase-smart-wallet.js +38 -0
  17. package/dist/coinbase-smart-wallet.js.map +1 -0
  18. package/dist/commands/accept.d.ts +3 -0
  19. package/dist/commands/accept.d.ts.map +1 -0
  20. package/dist/commands/accept.js +26 -0
  21. package/dist/commands/accept.js.map +1 -0
  22. package/dist/commands/agent-handshake.d.ts +3 -0
  23. package/dist/commands/agent-handshake.d.ts.map +1 -0
  24. package/dist/commands/agent-handshake.js +61 -0
  25. package/dist/commands/agent-handshake.js.map +1 -0
  26. package/dist/commands/agent.d.ts +3 -0
  27. package/dist/commands/agent.d.ts.map +1 -0
  28. package/dist/commands/agent.js +417 -0
  29. package/dist/commands/agent.js.map +1 -0
  30. package/dist/commands/agreements.d.ts +3 -0
  31. package/dist/commands/agreements.d.ts.map +1 -0
  32. package/dist/commands/agreements.js +344 -0
  33. package/dist/commands/agreements.js.map +1 -0
  34. package/dist/commands/arbitrator.d.ts +3 -0
  35. package/dist/commands/arbitrator.d.ts.map +1 -0
  36. package/dist/commands/arbitrator.js +157 -0
  37. package/dist/commands/arbitrator.js.map +1 -0
  38. package/dist/commands/arena-handshake.d.ts +3 -0
  39. package/dist/commands/arena-handshake.d.ts.map +1 -0
  40. package/dist/commands/arena-handshake.js +187 -0
  41. package/dist/commands/arena-handshake.js.map +1 -0
  42. package/dist/commands/cancel.d.ts +3 -0
  43. package/dist/commands/cancel.d.ts.map +1 -0
  44. package/dist/commands/cancel.js +30 -0
  45. package/dist/commands/cancel.js.map +1 -0
  46. package/dist/commands/channel.d.ts +3 -0
  47. package/dist/commands/channel.d.ts.map +1 -0
  48. package/dist/commands/channel.js +238 -0
  49. package/dist/commands/channel.js.map +1 -0
  50. package/dist/commands/coldstart.d.ts +3 -0
  51. package/dist/commands/coldstart.d.ts.map +1 -0
  52. package/dist/commands/coldstart.js +148 -0
  53. package/dist/commands/coldstart.js.map +1 -0
  54. package/dist/commands/config.d.ts +3 -0
  55. package/dist/commands/config.d.ts.map +1 -0
  56. package/dist/commands/config.js +40 -0
  57. package/dist/commands/config.js.map +1 -0
  58. package/dist/commands/contract-interaction.d.ts +3 -0
  59. package/dist/commands/contract-interaction.d.ts.map +1 -0
  60. package/dist/commands/contract-interaction.js +165 -0
  61. package/dist/commands/contract-interaction.js.map +1 -0
  62. package/dist/commands/daemon.d.ts +3 -0
  63. package/dist/commands/daemon.d.ts.map +1 -0
  64. package/dist/commands/daemon.js +891 -0
  65. package/dist/commands/daemon.js.map +1 -0
  66. package/dist/commands/deliver.d.ts +3 -0
  67. package/dist/commands/deliver.d.ts.map +1 -0
  68. package/dist/commands/deliver.js +156 -0
  69. package/dist/commands/deliver.js.map +1 -0
  70. package/dist/commands/discover.d.ts +3 -0
  71. package/dist/commands/discover.d.ts.map +1 -0
  72. package/dist/commands/discover.js +224 -0
  73. package/dist/commands/discover.js.map +1 -0
  74. package/dist/commands/dispute.d.ts +3 -0
  75. package/dist/commands/dispute.d.ts.map +1 -0
  76. package/dist/commands/dispute.js +348 -0
  77. package/dist/commands/dispute.js.map +1 -0
  78. package/dist/commands/endpoint.d.ts +3 -0
  79. package/dist/commands/endpoint.d.ts.map +1 -0
  80. package/dist/commands/endpoint.js +604 -0
  81. package/dist/commands/endpoint.js.map +1 -0
  82. package/dist/commands/hire.d.ts +3 -0
  83. package/dist/commands/hire.d.ts.map +1 -0
  84. package/dist/commands/hire.js +189 -0
  85. package/dist/commands/hire.js.map +1 -0
  86. package/dist/commands/migrate.d.ts +3 -0
  87. package/dist/commands/migrate.d.ts.map +1 -0
  88. package/dist/commands/migrate.js +163 -0
  89. package/dist/commands/migrate.js.map +1 -0
  90. package/dist/commands/negotiate.d.ts +3 -0
  91. package/dist/commands/negotiate.d.ts.map +1 -0
  92. package/dist/commands/negotiate.js +247 -0
  93. package/dist/commands/negotiate.js.map +1 -0
  94. package/dist/commands/openshell.d.ts +3 -0
  95. package/dist/commands/openshell.d.ts.map +1 -0
  96. package/dist/commands/openshell.js +952 -0
  97. package/dist/commands/openshell.js.map +1 -0
  98. package/dist/commands/owner.d.ts +3 -0
  99. package/dist/commands/owner.d.ts.map +1 -0
  100. package/dist/commands/owner.js +32 -0
  101. package/dist/commands/owner.js.map +1 -0
  102. package/dist/commands/policy.d.ts +4 -0
  103. package/dist/commands/policy.d.ts.map +1 -0
  104. package/dist/commands/policy.js +248 -0
  105. package/dist/commands/policy.js.map +1 -0
  106. package/dist/commands/relay.d.ts +3 -0
  107. package/dist/commands/relay.d.ts.map +1 -0
  108. package/dist/commands/relay.js +279 -0
  109. package/dist/commands/relay.js.map +1 -0
  110. package/dist/commands/remediate.d.ts +3 -0
  111. package/dist/commands/remediate.d.ts.map +1 -0
  112. package/dist/commands/remediate.js +42 -0
  113. package/dist/commands/remediate.js.map +1 -0
  114. package/dist/commands/reputation.d.ts +4 -0
  115. package/dist/commands/reputation.d.ts.map +1 -0
  116. package/dist/commands/reputation.js +72 -0
  117. package/dist/commands/reputation.js.map +1 -0
  118. package/dist/commands/setup.d.ts +3 -0
  119. package/dist/commands/setup.d.ts.map +1 -0
  120. package/dist/commands/setup.js +332 -0
  121. package/dist/commands/setup.js.map +1 -0
  122. package/dist/commands/trust.d.ts +3 -0
  123. package/dist/commands/trust.d.ts.map +1 -0
  124. package/dist/commands/trust.js +23 -0
  125. package/dist/commands/trust.js.map +1 -0
  126. package/dist/commands/verify.d.ts +3 -0
  127. package/dist/commands/verify.d.ts.map +1 -0
  128. package/dist/commands/verify.js +88 -0
  129. package/dist/commands/verify.js.map +1 -0
  130. package/dist/commands/wallet.d.ts +3 -0
  131. package/dist/commands/wallet.d.ts.map +1 -0
  132. package/dist/commands/wallet.js +2520 -0
  133. package/dist/commands/wallet.js.map +1 -0
  134. package/dist/commands/watchtower.d.ts +3 -0
  135. package/dist/commands/watchtower.d.ts.map +1 -0
  136. package/dist/commands/watchtower.js +238 -0
  137. package/dist/commands/watchtower.js.map +1 -0
  138. package/dist/commands/workroom.d.ts +3 -0
  139. package/dist/commands/workroom.d.ts.map +1 -0
  140. package/dist/commands/workroom.js +855 -0
  141. package/dist/commands/workroom.js.map +1 -0
  142. package/dist/config.d.ts +62 -0
  143. package/dist/config.d.ts.map +1 -0
  144. package/dist/config.js +141 -0
  145. package/dist/config.js.map +1 -0
  146. package/dist/daemon/config.d.ts +74 -0
  147. package/dist/daemon/config.d.ts.map +1 -0
  148. package/dist/daemon/config.js +271 -0
  149. package/dist/daemon/config.js.map +1 -0
  150. package/dist/daemon/hire-listener.d.ts +31 -0
  151. package/dist/daemon/hire-listener.d.ts.map +1 -0
  152. package/dist/daemon/hire-listener.js +207 -0
  153. package/dist/daemon/hire-listener.js.map +1 -0
  154. package/dist/daemon/index.d.ts +29 -0
  155. package/dist/daemon/index.d.ts.map +1 -0
  156. package/dist/daemon/index.js +535 -0
  157. package/dist/daemon/index.js.map +1 -0
  158. package/dist/daemon/job-lifecycle.d.ts +62 -0
  159. package/dist/daemon/job-lifecycle.d.ts.map +1 -0
  160. package/dist/daemon/job-lifecycle.js +201 -0
  161. package/dist/daemon/job-lifecycle.js.map +1 -0
  162. package/dist/daemon/notify.d.ts +22 -0
  163. package/dist/daemon/notify.d.ts.map +1 -0
  164. package/dist/daemon/notify.js +148 -0
  165. package/dist/daemon/notify.js.map +1 -0
  166. package/dist/daemon/token-metering.d.ts +42 -0
  167. package/dist/daemon/token-metering.d.ts.map +1 -0
  168. package/dist/daemon/token-metering.js +178 -0
  169. package/dist/daemon/token-metering.js.map +1 -0
  170. package/dist/daemon/userops.d.ts +21 -0
  171. package/dist/daemon/userops.d.ts.map +1 -0
  172. package/dist/daemon/userops.js +88 -0
  173. package/dist/daemon/userops.js.map +1 -0
  174. package/dist/daemon/wallet-monitor.d.ts +16 -0
  175. package/dist/daemon/wallet-monitor.d.ts.map +1 -0
  176. package/dist/daemon/wallet-monitor.js +57 -0
  177. package/dist/daemon/wallet-monitor.js.map +1 -0
  178. package/dist/drain-v4.d.ts +2 -0
  179. package/dist/drain-v4.d.ts.map +1 -0
  180. package/dist/drain-v4.js +167 -0
  181. package/dist/drain-v4.js.map +1 -0
  182. package/dist/endpoint-config.d.ts +36 -0
  183. package/dist/endpoint-config.d.ts.map +1 -0
  184. package/dist/endpoint-config.js +96 -0
  185. package/dist/endpoint-config.js.map +1 -0
  186. package/dist/index.d.ts +3 -0
  187. package/dist/index.d.ts.map +1 -0
  188. package/dist/index.js +79 -0
  189. package/dist/index.js.map +1 -0
  190. package/dist/openshell-runtime.d.ts +55 -0
  191. package/dist/openshell-runtime.d.ts.map +1 -0
  192. package/dist/openshell-runtime.js +268 -0
  193. package/dist/openshell-runtime.js.map +1 -0
  194. package/dist/signing.d.ts +2 -0
  195. package/dist/signing.d.ts.map +1 -0
  196. package/dist/signing.js +23 -0
  197. package/dist/signing.js.map +1 -0
  198. package/dist/telegram-notify.d.ts +23 -0
  199. package/dist/telegram-notify.d.ts.map +1 -0
  200. package/dist/telegram-notify.js +106 -0
  201. package/dist/telegram-notify.js.map +1 -0
  202. package/dist/ui/banner.d.ts +7 -0
  203. package/dist/ui/banner.d.ts.map +1 -0
  204. package/dist/ui/banner.js +37 -0
  205. package/dist/ui/banner.js.map +1 -0
  206. package/dist/ui/colors.d.ts +14 -0
  207. package/dist/ui/colors.d.ts.map +1 -0
  208. package/dist/ui/colors.js +29 -0
  209. package/dist/ui/colors.js.map +1 -0
  210. package/dist/ui/format.d.ts +26 -0
  211. package/dist/ui/format.d.ts.map +1 -0
  212. package/dist/ui/format.js +77 -0
  213. package/dist/ui/format.js.map +1 -0
  214. package/dist/ui/spinner.d.ts +8 -0
  215. package/dist/ui/spinner.d.ts.map +1 -0
  216. package/dist/ui/spinner.js +43 -0
  217. package/dist/ui/spinner.js.map +1 -0
  218. package/dist/utils/format.d.ts +10 -0
  219. package/dist/utils/format.d.ts.map +1 -0
  220. package/dist/utils/format.js +61 -0
  221. package/dist/utils/format.js.map +1 -0
  222. package/dist/utils/hash.d.ts +3 -0
  223. package/dist/utils/hash.d.ts.map +1 -0
  224. package/dist/utils/hash.js +43 -0
  225. package/dist/utils/hash.js.map +1 -0
  226. package/dist/utils/time.d.ts +3 -0
  227. package/dist/utils/time.d.ts.map +1 -0
  228. package/dist/utils/time.js +21 -0
  229. package/dist/utils/time.js.map +1 -0
  230. package/dist/wallet-router.d.ts +25 -0
  231. package/dist/wallet-router.d.ts.map +1 -0
  232. package/dist/wallet-router.js +153 -0
  233. package/dist/wallet-router.js.map +1 -0
  234. package/dist/walletconnect-session.d.ts +12 -0
  235. package/dist/walletconnect-session.d.ts.map +1 -0
  236. package/dist/walletconnect-session.js +26 -0
  237. package/dist/walletconnect-session.js.map +1 -0
  238. package/dist/walletconnect.d.ts +46 -0
  239. package/dist/walletconnect.d.ts.map +1 -0
  240. package/dist/walletconnect.js +267 -0
  241. package/dist/walletconnect.js.map +1 -0
  242. package/package.json +38 -0
  243. package/scripts/authorize-machine-key.ts +43 -0
  244. package/scripts/drain-wallet.ts +149 -0
  245. package/scripts/execute-spend-only.ts +81 -0
  246. package/scripts/register-agent-userop.ts +186 -0
  247. package/src/abis.ts +187 -0
  248. package/src/bundler.ts +235 -0
  249. package/src/client.ts +34 -0
  250. package/src/coinbase-smart-wallet.ts +51 -0
  251. package/src/commands/accept.ts +25 -0
  252. package/src/commands/agent-handshake.ts +67 -0
  253. package/src/commands/agent.ts +458 -0
  254. package/src/commands/agreements.ts +324 -0
  255. package/src/commands/arbitrator.ts +129 -0
  256. package/src/commands/arena-handshake.ts +217 -0
  257. package/src/commands/cancel.ts +26 -0
  258. package/src/commands/channel.ts +208 -0
  259. package/src/commands/coldstart.ts +156 -0
  260. package/src/commands/config.ts +35 -0
  261. package/src/commands/contract-interaction.ts +166 -0
  262. package/src/commands/daemon.ts +971 -0
  263. package/src/commands/deliver.ts +116 -0
  264. package/src/commands/discover.ts +295 -0
  265. package/src/commands/dispute.ts +373 -0
  266. package/src/commands/endpoint.ts +619 -0
  267. package/src/commands/hire.ts +200 -0
  268. package/src/commands/migrate.ts +175 -0
  269. package/src/commands/negotiate.ts +270 -0
  270. package/src/commands/openshell.ts +1053 -0
  271. package/src/commands/owner.ts +30 -0
  272. package/src/commands/policy.ts +252 -0
  273. package/src/commands/relay.ts +272 -0
  274. package/src/commands/remediate.ts +22 -0
  275. package/src/commands/reputation.ts +71 -0
  276. package/src/commands/setup.ts +343 -0
  277. package/src/commands/trust.ts +15 -0
  278. package/src/commands/verify.ts +88 -0
  279. package/src/commands/wallet.ts +2892 -0
  280. package/src/commands/watchtower.ts +232 -0
  281. package/src/commands/workroom.ts +889 -0
  282. package/src/config.ts +153 -0
  283. package/src/daemon/config.ts +308 -0
  284. package/src/daemon/hire-listener.ts +226 -0
  285. package/src/daemon/index.ts +609 -0
  286. package/src/daemon/job-lifecycle.ts +215 -0
  287. package/src/daemon/notify.ts +157 -0
  288. package/src/daemon/token-metering.ts +183 -0
  289. package/src/daemon/userops.ts +119 -0
  290. package/src/daemon/wallet-monitor.ts +90 -0
  291. package/src/drain-v4.ts +159 -0
  292. package/src/endpoint-config.ts +83 -0
  293. package/src/index.ts +75 -0
  294. package/src/openshell-runtime.ts +277 -0
  295. package/src/signing.ts +28 -0
  296. package/src/telegram-notify.ts +88 -0
  297. package/src/ui/banner.ts +41 -0
  298. package/src/ui/colors.ts +30 -0
  299. package/src/ui/format.ts +77 -0
  300. package/src/ui/spinner.ts +46 -0
  301. package/src/utils/format.ts +48 -0
  302. package/src/utils/hash.ts +5 -0
  303. package/src/utils/time.ts +15 -0
  304. package/src/wallet-router.ts +178 -0
  305. package/src/walletconnect-session.ts +27 -0
  306. package/src/walletconnect.ts +294 -0
  307. package/test/time.test.js +11 -0
  308. package/tsconfig.json +19 -0
@@ -0,0 +1,324 @@
1
+ import { Command } from "commander";
2
+ import { ethers } from "ethers";
3
+ import * as readline from "readline";
4
+ import { AgreementStatus, ServiceAgreementClient } from "@arc402/sdk";
5
+ import { loadConfig } from "../config";
6
+ import { getClient } from "../client";
7
+ import { agreementStatusLabel, formatDate, printTable, truncateAddress } from "../utils/format";
8
+ import { formatDeadline } from "../utils/time";
9
+ import { hashString } from "../utils/hash";
10
+
11
+ // ─── AgreementTree minimal ABI ────────────────────────────────────────────────
12
+
13
+ const AGREEMENT_TREE_ABI = [
14
+ "function registerSubAgreement(uint256 parentAgreementId, uint256 childAgreementId) external",
15
+ "function getChildren(uint256 agreementId) external view returns (uint256[])",
16
+ "function getRoot(uint256 agreementId) external view returns (uint256)",
17
+ "function getPath(uint256 agreementId) external view returns (uint256[])",
18
+ "function allChildrenSettled(uint256 agreementId) external view returns (bool)",
19
+ "function getDepth(uint256 agreementId) external view returns (uint256)",
20
+ ];
21
+
22
+ export function registerAgreementsCommands(program: Command): void {
23
+
24
+ // ── arc402 agreements ────────────────────────────────────────────────────────
25
+ program
26
+ .command("agreements")
27
+ .description("List agreements for the configured wallet")
28
+ .option("--as <role>", "client or provider", "client")
29
+ .option("--json")
30
+ .action(async (opts) => {
31
+ const config = loadConfig();
32
+ if (!config.serviceAgreementAddress) throw new Error("serviceAgreementAddress missing in config");
33
+ const { provider, address } = await getClient(config);
34
+ if (!address) throw new Error("No wallet configured");
35
+ const client = new ServiceAgreementClient(config.serviceAgreementAddress, provider);
36
+ const agreements = opts.as === "provider"
37
+ ? await client.getProviderAgreements(address)
38
+ : await client.getClientAgreements(address);
39
+ if (opts.json) return console.log(JSON.stringify(agreements, (_k, value) => typeof value === "bigint" ? value.toString() : value, 2));
40
+ printTable(
41
+ ["ID", "COUNTERPARTY", "SERVICE", "DEADLINE", "STATUS"],
42
+ agreements.map((agreement) => [
43
+ agreement.id.toString(),
44
+ truncateAddress(opts.as === "provider" ? agreement.client : agreement.provider),
45
+ agreement.serviceType,
46
+ formatDeadline(Number(agreement.deadline)),
47
+ agreementStatusLabel(agreement.status),
48
+ ])
49
+ );
50
+ });
51
+
52
+ // ── arc402 agreement <id> ────────────────────────────────────────────────────
53
+ program
54
+ .command("agreement <id>")
55
+ .description("Show agreement detail, including remediation/dispute fields")
56
+ .option("--json")
57
+ .action(async (id, opts) => {
58
+ const config = loadConfig();
59
+ if (!config.serviceAgreementAddress) throw new Error("serviceAgreementAddress missing in config");
60
+ const { provider } = await getClient(config);
61
+ const client = new ServiceAgreementClient(config.serviceAgreementAddress, provider);
62
+ const agreement = await client.getAgreement(BigInt(id));
63
+ if (opts.json) return console.log(JSON.stringify(agreement, (_k, value) => typeof value === "bigint" ? value.toString() : value, 2));
64
+ console.log(`agreement #${agreement.id}\nclient=${agreement.client}\nprovider=${agreement.provider}\nstatus=${agreementStatusLabel(agreement.status)}\ncreated=${formatDate(Number(agreement.createdAt))}\ndeadline=${formatDate(Number(agreement.deadline))}\nverifyWindowEnd=${Number(agreement.verifyWindowEnd) ? formatDate(Number(agreement.verifyWindowEnd)) : "n/a"}\ncommittedHash=${agreement.committedHash}`);
65
+ if ([AgreementStatus.REVISION_REQUESTED, AgreementStatus.REVISED, AgreementStatus.PARTIAL_SETTLEMENT, AgreementStatus.ESCALATED_TO_HUMAN, AgreementStatus.DISPUTED, AgreementStatus.ESCALATED_TO_ARBITRATION].includes(agreement.status)) {
66
+ const remediation = await client.getRemediationCase(agreement.id);
67
+ const dispute = await client.getDisputeCase(agreement.id);
68
+ console.log(`remediationActive=${remediation.active} cycles=${remediation.cycleCount} disputeOutcome=${dispute.outcome}`);
69
+ }
70
+ });
71
+
72
+ // ── arc402 agreements-sub-register ──────────────────────────────────────────
73
+ // Spec 19: arc402 agreements sub-register --parent <id> --child <id>
74
+ program
75
+ .command("agreements-sub-register")
76
+ .description("Link a child agreement to its parent in the AgreementTree (Spec 19)")
77
+ .requiredOption("--parent <id>", "Parent agreement ID")
78
+ .requiredOption("--child <id>", "Child agreement ID to register under the parent")
79
+ .option("--json", "Machine-parseable output")
80
+ .action(async (opts) => {
81
+ const config = loadConfig();
82
+ if (!config.agreementTreeAddress) throw new Error("agreementTreeAddress missing in config");
83
+ const { signer } = await getClient(config);
84
+ if (!signer) { console.error("No private key configured."); process.exit(1); }
85
+
86
+ const tree = new ethers.Contract(config.agreementTreeAddress, AGREEMENT_TREE_ABI, signer);
87
+ const tx = await tree.registerSubAgreement(BigInt(opts.parent), BigInt(opts.child));
88
+ const receipt = await tx.wait();
89
+
90
+ if (opts.json) {
91
+ console.log(JSON.stringify({ txHash: receipt.hash, parentId: opts.parent, childId: opts.child }));
92
+ } else {
93
+ console.log(`Sub-agreement registered.`);
94
+ console.log(` Parent: ${opts.parent} Child: ${opts.child}`);
95
+ console.log(` tx: ${receipt.hash}`);
96
+ }
97
+ });
98
+
99
+ // ── arc402 agreements-tree <id> ──────────────────────────────────────────────
100
+ // Spec 19: arc402 agreements tree <agreementId>
101
+ program
102
+ .command("agreements-tree <agreementId>")
103
+ .description("View full agreement tree for an agreement (Spec 19)")
104
+ .option("--json", "Machine-parseable output")
105
+ .action(async (agreementId, opts) => {
106
+ const config = loadConfig();
107
+ if (!config.agreementTreeAddress) throw new Error("agreementTreeAddress missing in config");
108
+ const { provider } = await getClient(config);
109
+
110
+ const tree = new ethers.Contract(config.agreementTreeAddress, AGREEMENT_TREE_ABI, provider);
111
+
112
+ const id = BigInt(agreementId);
113
+ const [root, path, children, depth] = await Promise.all([
114
+ tree.getRoot(id),
115
+ tree.getPath(id),
116
+ tree.getChildren(id),
117
+ tree.getDepth(id),
118
+ ]);
119
+
120
+ const result = {
121
+ agreementId: agreementId,
122
+ root: root.toString(),
123
+ path: (path as bigint[]).map(String),
124
+ children: (children as bigint[]).map(String),
125
+ depth: Number(depth),
126
+ };
127
+
128
+ if (opts.json) {
129
+ console.log(JSON.stringify(result));
130
+ return;
131
+ }
132
+
133
+ console.log(`Agreement Tree — node #${agreementId}`);
134
+ console.log(` Root: #${result.root}`);
135
+ console.log(` Depth: ${result.depth}`);
136
+ console.log(` Path: ${result.path.map((p) => "#" + p).join(" → ")}`);
137
+ console.log(` Children: ${result.children.length > 0 ? result.children.map((c) => "#" + c).join(", ") : "(none)"}`);
138
+ });
139
+
140
+ // ── arc402 agreements-tree-status <id> ───────────────────────────────────────
141
+ // Spec 19: arc402 agreements tree-status <agreementId>
142
+ program
143
+ .command("agreements-tree-status <agreementId>")
144
+ .description("Check whether all sub-agreements are settled before delivering (Spec 19)")
145
+ .option("--json", "Machine-parseable output")
146
+ .action(async (agreementId, opts) => {
147
+ const config = loadConfig();
148
+ if (!config.agreementTreeAddress) throw new Error("agreementTreeAddress missing in config");
149
+ const { provider } = await getClient(config);
150
+
151
+ const tree = new ethers.Contract(config.agreementTreeAddress, AGREEMENT_TREE_ABI, provider);
152
+
153
+ const id = BigInt(agreementId);
154
+ const [allSettled, children] = await Promise.all([
155
+ tree.allChildrenSettled(id),
156
+ tree.getChildren(id),
157
+ ]);
158
+
159
+ const result = {
160
+ agreementId: agreementId,
161
+ childCount: (children as bigint[]).length,
162
+ allChildrenSettled: Boolean(allSettled),
163
+ readyToDeliver: Boolean(allSettled),
164
+ };
165
+
166
+ if (opts.json) {
167
+ console.log(JSON.stringify(result));
168
+ return;
169
+ }
170
+
171
+ console.log(`Agreement #${agreementId} tree status:`);
172
+ console.log(` Sub-agreements: ${result.childCount}`);
173
+ console.log(` All settled: ${result.allChildrenSettled ? "YES" : "NO"}`);
174
+ console.log(` Ready to deliver to parent: ${result.readyToDeliver ? "YES" : "NO"}`);
175
+ });
176
+
177
+ // ── arc402 agreements-create-tree ────────────────────────────────────────────
178
+ // Spec 19: interactive multi-party agreement setup
179
+ // Usage: arc402 agreements-create-tree \
180
+ // --provider <addr> --task <desc> --service-type <type> --price <wei> --deadline <s> \
181
+ // --sub <addr>,<task>,<type>,<wei>,<deadline> [--sub ...]
182
+ program
183
+ .command("agreements-create-tree")
184
+ .description("Create a multi-party agreement tree: one root agreement plus sub-agreements (Spec 19)")
185
+ .requiredOption("--provider <address>", "Root provider (Agent B)")
186
+ .requiredOption("--task <description>", "Root task description")
187
+ .requiredOption("--service-type <type>", "Root service type")
188
+ .requiredOption("--price <wei>", "Root price in wei")
189
+ .requiredOption("--deadline <seconds>", "Root deadline offset in seconds from now")
190
+ .option(
191
+ "--sub <spec>",
192
+ "Sub-agreement in format <provider>,<task>,<serviceType>,<priceWei>,<deadlineSeconds>. Repeatable.",
193
+ (val: string, acc: string[]) => { acc.push(val); return acc; },
194
+ [] as string[],
195
+ )
196
+ .option("--interactive", "Prompt interactively for sub-agreements")
197
+ .option("--json", "Machine-parseable output")
198
+ .action(async (opts) => {
199
+ const config = loadConfig();
200
+ if (!config.serviceAgreementAddress) throw new Error("serviceAgreementAddress missing in config");
201
+ if (!config.agreementTreeAddress) throw new Error("agreementTreeAddress missing in config");
202
+ const { signer } = await getClient(config);
203
+ if (!signer) { console.error("No private key configured."); process.exit(1); }
204
+
205
+ type SubSpec = { provider: string; task: string; serviceType: string; price: bigint; deadline: number };
206
+ const subSpecs: SubSpec[] = [];
207
+
208
+ // Parse --sub flags
209
+ for (const raw of opts.sub as string[]) {
210
+ const parts = raw.split(",");
211
+ if (parts.length < 5) {
212
+ console.error(`--sub format: <provider>,<task>,<serviceType>,<priceWei>,<deadlineSeconds> got: ${raw}`);
213
+ process.exit(1);
214
+ }
215
+ subSpecs.push({
216
+ provider: parts[0],
217
+ task: parts.slice(1, parts.length - 3).join(",") || parts[1],
218
+ serviceType: parts[parts.length - 3],
219
+ price: BigInt(parts[parts.length - 2]),
220
+ deadline: Number(parts[parts.length - 1]),
221
+ });
222
+ }
223
+
224
+ // Interactive mode: prompt for sub-agreements if no --sub flags provided
225
+ if (opts.interactive || subSpecs.length === 0) {
226
+ const rl = readline.createInterface({ input: process.stdin, output: process.stdout });
227
+ const ask = (q: string) => new Promise<string>((res) => rl.question(q, res));
228
+
229
+ console.log("\nMulti-party agreement setup — add sub-agreements (leave provider blank to finish)");
230
+ while (true) {
231
+ const provider = await ask(" Sub-provider address (blank to finish): ");
232
+ if (!provider.trim()) break;
233
+ const task = await ask(" Task description: ");
234
+ const serviceType = await ask(" Service type: ");
235
+ const priceRaw = await ask(" Price (wei): ");
236
+ const deadlineRaw = await ask(" Deadline offset (seconds): ");
237
+ subSpecs.push({
238
+ provider: provider.trim(),
239
+ task: task.trim(),
240
+ serviceType: serviceType.trim(),
241
+ price: BigInt(priceRaw.trim()),
242
+ deadline: Number(deadlineRaw.trim()),
243
+ });
244
+ }
245
+ rl.close();
246
+ }
247
+
248
+ const saContract = new ethers.Contract(config.serviceAgreementAddress, [
249
+ "function propose(address provider, string serviceType, string description, uint256 price, address token, uint256 deadline, bytes32 deliverablesHash) external payable returns (uint256)",
250
+ "event AgreementProposed(uint256 indexed id, address indexed client, address indexed provider, string serviceType, uint256 price, address token, uint256 deadline)",
251
+ ], signer);
252
+ const treeContract = new ethers.Contract(config.agreementTreeAddress, AGREEMENT_TREE_ABI, signer);
253
+
254
+ const rootPrice = BigInt(opts.price);
255
+ const rootDeadline = Number(opts.deadline);
256
+ const rootHash = hashString(opts.task);
257
+
258
+ if (!opts.json) console.log(`\nProposing root agreement to ${opts.provider}...`);
259
+ const rootTx = await saContract.propose(
260
+ opts.provider, opts.serviceType, opts.task,
261
+ rootPrice, ethers.ZeroAddress, rootDeadline, rootHash,
262
+ { value: rootPrice },
263
+ );
264
+ const rootReceipt = await rootTx.wait();
265
+ const iface = new ethers.Interface([
266
+ "event AgreementProposed(uint256 indexed id, address indexed client, address indexed provider, string serviceType, uint256 price, address token, uint256 deadline)",
267
+ ]);
268
+ let rootAgreementId: bigint | undefined;
269
+ for (const log of rootReceipt.logs) {
270
+ try {
271
+ const parsed = iface.parseLog(log);
272
+ if (parsed?.name === "AgreementProposed") { rootAgreementId = BigInt(parsed.args.id); break; }
273
+ } catch { /* skip */ }
274
+ }
275
+ if (rootAgreementId === undefined) throw new Error("Could not parse root AgreementProposed event");
276
+
277
+ if (!opts.json) console.log(` Root agreement ID: ${rootAgreementId}`);
278
+
279
+ const childAgreementIds: bigint[] = [];
280
+ for (const sub of subSpecs) {
281
+ if (!opts.json) console.log(` Proposing sub-agreement to ${sub.provider}...`);
282
+ const subHash = hashString(sub.task);
283
+ const subTx = await saContract.propose(
284
+ sub.provider, sub.serviceType, sub.task,
285
+ sub.price, ethers.ZeroAddress, sub.deadline, subHash,
286
+ { value: sub.price },
287
+ );
288
+ const subReceipt = await subTx.wait();
289
+ let childId: bigint | undefined;
290
+ for (const log of subReceipt.logs) {
291
+ try {
292
+ const parsed = iface.parseLog(log);
293
+ if (parsed?.name === "AgreementProposed") { childId = BigInt(parsed.args.id); break; }
294
+ } catch { /* skip */ }
295
+ }
296
+ if (childId === undefined) throw new Error(`Could not parse AgreementProposed for sub-agreement to ${sub.provider}`);
297
+ if (!opts.json) console.log(` Sub-agreement ID: ${childId} — registering in tree...`);
298
+ await (await treeContract.registerSubAgreement(rootAgreementId, childId)).wait();
299
+ childAgreementIds.push(childId);
300
+ }
301
+
302
+ const result = {
303
+ rootAgreementId: rootAgreementId.toString(),
304
+ childAgreementIds: childAgreementIds.map(String),
305
+ txHash: rootReceipt.hash,
306
+ };
307
+
308
+ if (opts.json) {
309
+ console.log(JSON.stringify(result, null, 2));
310
+ } else {
311
+ console.log(`\nAgreement tree created.`);
312
+ console.log(` Root: #${result.rootAgreementId}`);
313
+ if (result.childAgreementIds.length > 0) {
314
+ console.log(` Children: ${result.childAgreementIds.map((id) => "#" + id).join(", ")}`);
315
+ } else {
316
+ console.log(` Children: (none — sub-agreements can be added with agreements-sub-register)`);
317
+ }
318
+ }
319
+ });
320
+
321
+ // ── arc402 agreements-tree <id> (alias) already exists above ─────────────────
322
+ // The spec also refers to this as "arc402 agreements tree <id>" — the flat
323
+ // command agreements-tree <agreementId> above fulfils that requirement.
324
+ }
@@ -0,0 +1,129 @@
1
+ import { Command } from "commander";
2
+ import { DisputeArbitrationClient } from "@arc402/sdk";
3
+ import { loadConfig } from "../config";
4
+ import { getClient, requireSigner } from "../client";
5
+
6
+ export function registerArbitratorCommand(program: Command): void {
7
+ const arbitrator = program
8
+ .command("arbitrator")
9
+ .description("Arbitrator panel operations: bonds, eligibility, status");
10
+
11
+ // Bond subcommands — nest under 'bond' to avoid duplicate registration
12
+ const bond = arbitrator
13
+ .command("bond")
14
+ .description("Arbitrator bond operations: status, accept, fallback");
15
+
16
+ bond
17
+ .command("status <arbitratorAddress> [agreementId]")
18
+ .description("Check arbitrator bond status for an agreement (or general eligibility)")
19
+ .action(async (arbitratorAddress, agreementId, _opts) => {
20
+ const config = loadConfig();
21
+ if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
22
+ const { provider } = await getClient(config);
23
+ const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, provider);
24
+
25
+ const eligible = await client.isEligibleArbitrator(arbitratorAddress);
26
+ console.log(`Arbitrator ${arbitratorAddress} eligible: ${eligible}`);
27
+
28
+ if (agreementId) {
29
+ const bondState = await client.getArbitratorBondState(arbitratorAddress, BigInt(agreementId));
30
+ console.log(`Bond state for agreement ${agreementId}:`);
31
+ console.log(` Amount: ${bondState.bondAmount.toString()} tokens`);
32
+ console.log(` Locked: ${bondState.locked}`);
33
+ console.log(` Slashed: ${bondState.slashed}`);
34
+ console.log(` Returned: ${bondState.returned}`);
35
+ console.log(` Locked at: ${new Date(Number(bondState.lockedAt) * 1000).toISOString()}`);
36
+ }
37
+ });
38
+
39
+ bond
40
+ .command("accept <agreementId>")
41
+ .description("Accept panel assignment and post bond")
42
+ .option("--bond <bond>", "Bond amount in wei (for ETH agreements)", "0")
43
+ .action(async (agreementId, opts) => {
44
+ const config = loadConfig();
45
+ if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
46
+ const { signer } = await requireSigner(config);
47
+ const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, signer);
48
+
49
+ const bondAmount = BigInt(opts.bond);
50
+ if (bondAmount === 0n) {
51
+ console.warn("Warning: bond amount is 0. For ERC-20 agreements, pre-approve the DisputeArbitration contract.");
52
+ }
53
+
54
+ await client.acceptAssignment(BigInt(agreementId), bondAmount);
55
+ console.log(`accepted assignment for agreement ${agreementId}`);
56
+ });
57
+
58
+ bond
59
+ .command("fallback <agreementId>")
60
+ .description("Trigger fallback to human backstop (mutual unfunded or panel incomplete)")
61
+ .action(async (agreementId, _opts) => {
62
+ const config = loadConfig();
63
+ if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
64
+ const { signer } = await requireSigner(config);
65
+ const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, signer);
66
+ await client.triggerFallback(BigInt(agreementId));
67
+ console.log(`fallback triggered for ${agreementId}`);
68
+ });
69
+
70
+ // Rate subcommands — nest under 'rate'
71
+ const rate = arbitrator
72
+ .command("rate")
73
+ .description("Token USD rate management (owner only)");
74
+
75
+ rate
76
+ .command("set <tokenAddress> <usdPerToken>")
77
+ .description("Set USD rate for a token. Rate in USD with 18 decimals. e.g. 2000e18 for $2000")
78
+ .action(async (tokenAddress, usdPerToken, _opts) => {
79
+ const config = loadConfig();
80
+ if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
81
+ const { signer } = await requireSigner(config);
82
+ const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, signer);
83
+
84
+ const rateVal = BigInt(usdPerToken);
85
+ await client.setTokenUsdRate(tokenAddress, rateVal);
86
+ console.log(`token rate set: ${tokenAddress} = ${rateVal.toString()} USD/token`);
87
+ });
88
+
89
+ // Admin: slash arbitrator (manual rules violation)
90
+ arbitrator
91
+ .command("slash <agreementId> <arbitratorAddress> <reason>")
92
+ .description("Owner only: manually slash an arbitrator for rules violation")
93
+ .action(async (agreementId, arbitratorAddress, reason, _opts) => {
94
+ const config = loadConfig();
95
+ if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
96
+ const { signer } = await requireSigner(config);
97
+ const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, signer);
98
+ await client.slashArbitrator(BigInt(agreementId), arbitratorAddress, reason);
99
+ console.log(`slashed arbitrator ${arbitratorAddress} for ${reason}`);
100
+ });
101
+
102
+ arbitrator
103
+ .command("reclaim-bond <agreementId>")
104
+ .description("Reclaim an arbitrator bond after 45 days if dispute was never resolved via resolveDisputeFee")
105
+ .action(async (agreementId, _opts) => {
106
+ const config = loadConfig();
107
+ if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
108
+ const { signer } = await requireSigner(config);
109
+ const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, signer);
110
+ await client.reclaimExpiredBond(BigInt(agreementId));
111
+ console.log(`bond reclaimed for agreement ${agreementId}`);
112
+ });
113
+
114
+ arbitrator
115
+ .command("resolve-from <agreementId>")
116
+ .description("Resolve a dispute from arbitration with split amounts (calls ServiceAgreement.resolveFromArbitration)")
117
+ .requiredOption("--recipient <address>", "Winning recipient address")
118
+ .requiredOption("--provider-amount <amount>", "Provider payout in wei/tokens")
119
+ .requiredOption("--client-amount <amount>", "Client refund in wei/tokens")
120
+ .action(async (agreementId, opts) => {
121
+ const { ServiceAgreementClient } = await import("@arc402/sdk");
122
+ const config = loadConfig();
123
+ if (!config.serviceAgreementAddress) throw new Error("serviceAgreementAddress missing in config");
124
+ const { signer } = await requireSigner(config);
125
+ const client = new ServiceAgreementClient(config.serviceAgreementAddress, signer);
126
+ await client.resolveFromArbitration(BigInt(agreementId), opts.recipient, BigInt(opts.providerAmount), BigInt(opts.clientAmount));
127
+ console.log(`resolved from arbitration: agreement ${agreementId}, recipient ${opts.recipient}`);
128
+ });
129
+ }
@@ -0,0 +1,217 @@
1
+ import { Command } from "commander";
2
+ import { ethers } from "ethers";
3
+ import { loadConfig, getUsdcAddress } from "../config";
4
+ import { requireSigner } from "../client";
5
+
6
+ // ─── Handshake Contract ABI (from Handshake.sol) ─────────────────────────────
7
+
8
+ const HANDSHAKE_ABI = [
9
+ "function sendHandshake(address to, uint8 hsType, string note) payable",
10
+ "function sendHandshakeWithToken(address to, uint8 hsType, string note, address token, uint256 tokenAmount)",
11
+ "function sendBatch(address[] recipients, uint8[] hsTypes, string[] notes)",
12
+ "function hasConnection(address from, address to) view returns (bool)",
13
+ "function isMutual(address a, address b) view returns (bool)",
14
+ "function getStats(address agent) view returns (uint256 sent, uint256 received, uint256 uniqueInbound)",
15
+ "function sentCount(address) view returns (uint256)",
16
+ "function receivedCount(address) view returns (uint256)",
17
+ "function uniqueSenders(address) view returns (uint256)",
18
+ "function totalHandshakes() view returns (uint256)",
19
+ "function allowedTokens(address) view returns (bool)",
20
+ "event HandshakeSent(uint256 indexed handshakeId, address indexed from, address indexed to, uint8 hsType, address token, uint256 amount, string note, uint256 timestamp)",
21
+ "event NewConnection(address indexed from, address indexed to, uint256 handshakeId)",
22
+ ];
23
+
24
+ const POLICY_ENGINE_ABI = [
25
+ "function isContractWhitelisted(address wallet, address target) view returns (bool)",
26
+ "function whitelistContract(address wallet, address target)",
27
+ ];
28
+
29
+ const HANDSHAKE_TYPES: Record<string, number> = {
30
+ respect: 0,
31
+ curiosity: 1,
32
+ endorsement: 2,
33
+ thanks: 3,
34
+ collaboration: 4,
35
+ challenge: 5,
36
+ referral: 6,
37
+ hello: 7,
38
+ };
39
+
40
+ // ─── Auto-Whitelist ──────────────────────────────────────────────────────────
41
+
42
+ async function ensureWhitelisted(
43
+ signer: ethers.Signer,
44
+ provider: ethers.Provider,
45
+ walletAddress: string,
46
+ policyEngineAddress: string,
47
+ handshakeAddress: string
48
+ ): Promise<void> {
49
+ const pe = new ethers.Contract(policyEngineAddress, POLICY_ENGINE_ABI, provider);
50
+ const isWhitelisted = await pe.isContractWhitelisted(walletAddress, handshakeAddress);
51
+
52
+ if (!isWhitelisted) {
53
+ console.log("Handshake contract not yet whitelisted on your wallet.");
54
+ console.log("Whitelisting now (one-time setup)...");
55
+
56
+ const peSigner = new ethers.Contract(policyEngineAddress, POLICY_ENGINE_ABI, signer);
57
+ const tx = await peSigner.whitelistContract(walletAddress, handshakeAddress);
58
+ console.log(` tx: ${tx.hash}`);
59
+ await tx.wait();
60
+ console.log(" ✓ Handshake contract whitelisted\n");
61
+ }
62
+ }
63
+
64
+ // ─── Commands ────────────────────────────────────────────────────────────────
65
+
66
+ export function registerArenaHandshakeCommands(program: Command): void {
67
+ const hs = program
68
+ .command("shake")
69
+ .description("ARC Arena social handshake — send a typed trust signal to another agent.");
70
+
71
+ // ── send ──────────────────────────────────────────────────────────────────
72
+ hs.command("send <agentAddress>")
73
+ .description("Send a handshake to another agent.")
74
+ .option("--type <type>", "Handshake type: respect, curiosity, endorsement, thanks, collaboration, challenge, referral, hello", "hello")
75
+ .option("--note <note>", "Short message (max 280 chars)", "")
76
+ .option("--tip <amount>", "ETH tip to attach (e.g. 0.01)")
77
+ .option("--usdc <amount>", "USDC tip to attach (e.g. 5.00)")
78
+ .option("--json", "Output as JSON")
79
+ .action(async (agentAddress: string, opts) => {
80
+ const config = loadConfig();
81
+ const { signer, provider } = await requireSigner(config);
82
+ const myAddress = await signer.getAddress();
83
+
84
+ if (!config.handshakeAddress) {
85
+ console.error("handshakeAddress not configured. Run: arc402 config set handshakeAddress <address>");
86
+ process.exit(1);
87
+ }
88
+ if (!config.policyEngineAddress) {
89
+ console.error("policyEngineAddress not configured.");
90
+ process.exit(1);
91
+ }
92
+
93
+ // Auto-whitelist check
94
+ await ensureWhitelisted(signer, provider, myAddress, config.policyEngineAddress, config.handshakeAddress);
95
+
96
+ const hsType = HANDSHAKE_TYPES[opts.type.toLowerCase()];
97
+ if (hsType === undefined) {
98
+ console.error(`Unknown handshake type: ${opts.type}`);
99
+ console.error(`Valid types: ${Object.keys(HANDSHAKE_TYPES).join(", ")}`);
100
+ process.exit(1);
101
+ }
102
+
103
+ const handshake = new ethers.Contract(config.handshakeAddress, HANDSHAKE_ABI, signer);
104
+
105
+ let tx;
106
+ if (opts.usdc) {
107
+ // USDC handshake
108
+ const usdcAddress = getUsdcAddress(config);
109
+ const amount = ethers.parseUnits(opts.usdc, 6);
110
+ tx = await handshake.sendHandshakeWithToken(agentAddress, hsType, opts.note, usdcAddress, amount);
111
+ } else {
112
+ // ETH handshake (with optional tip)
113
+ const value = opts.tip ? ethers.parseEther(opts.tip) : 0n;
114
+ tx = await handshake.sendHandshake(agentAddress, hsType, opts.note, { value });
115
+ }
116
+
117
+ if (opts.json) {
118
+ console.log(JSON.stringify({ tx: tx.hash, from: myAddress, to: agentAddress, type: opts.type, note: opts.note }));
119
+ } else {
120
+ console.log(`✓ Handshake sent`);
121
+ console.log(` From: ${myAddress}`);
122
+ console.log(` To: ${agentAddress}`);
123
+ console.log(` Type: ${opts.type}`);
124
+ if (opts.note) console.log(` Note: ${opts.note}`);
125
+ if (opts.tip) console.log(` Tip: ${opts.tip} ETH`);
126
+ if (opts.usdc) console.log(` Tip: ${opts.usdc} USDC`);
127
+ console.log(` tx: ${tx.hash}`);
128
+ }
129
+ });
130
+
131
+ // ── batch ─────────────────────────────────────────────────────────────────
132
+ hs.command("batch")
133
+ .description("Send handshakes to multiple agents at once (onboarding ritual).")
134
+ .argument("<agents...>", "Agent addresses to handshake (up to 10)")
135
+ .option("--type <type>", "Handshake type for all", "hello")
136
+ .option("--note <note>", "Note for all", "")
137
+ .action(async (agents: string[], opts) => {
138
+ const config = loadConfig();
139
+ const { signer, provider } = await requireSigner(config);
140
+ const myAddress = await signer.getAddress();
141
+
142
+ if (!config.handshakeAddress || !config.policyEngineAddress) {
143
+ console.error("handshakeAddress or policyEngineAddress not configured.");
144
+ process.exit(1);
145
+ }
146
+
147
+ await ensureWhitelisted(signer, provider, myAddress, config.policyEngineAddress, config.handshakeAddress);
148
+
149
+ const hsType = HANDSHAKE_TYPES[opts.type.toLowerCase()];
150
+ if (hsType === undefined) {
151
+ console.error(`Unknown type: ${opts.type}. Valid: ${Object.keys(HANDSHAKE_TYPES).join(", ")}`);
152
+ process.exit(1);
153
+ }
154
+
155
+ if (agents.length > 10) {
156
+ console.error("Max 10 agents per batch.");
157
+ process.exit(1);
158
+ }
159
+
160
+ const handshake = new ethers.Contract(config.handshakeAddress, HANDSHAKE_ABI, signer);
161
+ const types = agents.map(() => hsType);
162
+ const notes = agents.map(() => opts.note);
163
+
164
+ const tx = await handshake.sendBatch(agents, types, notes);
165
+ console.log(`✓ Batch handshake sent to ${agents.length} agents`);
166
+ agents.forEach(a => console.log(` → ${a}`));
167
+ console.log(` tx: ${tx.hash}`);
168
+ });
169
+
170
+ // ── stats ─────────────────────────────────────────────────────────────────
171
+ hs.command("stats [address]")
172
+ .description("View handshake stats for an agent.")
173
+ .action(async (address?: string) => {
174
+ const config = loadConfig();
175
+ const { signer, provider } = await requireSigner(config);
176
+ const target = address || await signer.getAddress();
177
+
178
+ if (!config.handshakeAddress) {
179
+ console.error("handshakeAddress not configured.");
180
+ process.exit(1);
181
+ }
182
+
183
+ const handshake = new ethers.Contract(config.handshakeAddress, HANDSHAKE_ABI, provider);
184
+ const [sent, received, unique] = await handshake.getStats(target);
185
+ const total = await handshake.totalHandshakes();
186
+
187
+ console.log(`Handshake Stats: ${target}`);
188
+ console.log(` Sent: ${sent}`);
189
+ console.log(` Received: ${received}`);
190
+ console.log(` Unique senders: ${unique}`);
191
+ console.log(` Network total: ${total}`);
192
+ });
193
+
194
+ // ── check ─────────────────────────────────────────────────────────────────
195
+ hs.command("check <agentAddress>")
196
+ .description("Check if a connection or mutual handshake exists with an agent.")
197
+ .action(async (agentAddress: string) => {
198
+ const config = loadConfig();
199
+ const { signer, provider } = await requireSigner(config);
200
+ const myAddress = await signer.getAddress();
201
+
202
+ if (!config.handshakeAddress) {
203
+ console.error("handshakeAddress not configured.");
204
+ process.exit(1);
205
+ }
206
+
207
+ const handshake = new ethers.Contract(config.handshakeAddress, HANDSHAKE_ABI, provider);
208
+ const iSent = await handshake.hasConnection(myAddress, agentAddress);
209
+ const theySent = await handshake.hasConnection(agentAddress, myAddress);
210
+ const mutual = await handshake.isMutual(myAddress, agentAddress);
211
+
212
+ console.log(`Connection: ${myAddress} ↔ ${agentAddress}`);
213
+ console.log(` You → them: ${iSent ? "✓ handshaked" : "✗ no handshake"}`);
214
+ console.log(` Them → you: ${theySent ? "✓ handshaked" : "✗ no handshake"}`);
215
+ console.log(` Mutual: ${mutual ? "✓ yes" : "✗ no"}`);
216
+ });
217
+ }