arc402-cli 1.0.0-rc.1 → 1.0.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 (253) hide show
  1. package/README.md +41 -2
  2. package/dist/abis.d.ts +1 -0
  3. package/dist/abis.d.ts.map +1 -1
  4. package/dist/abis.js +29 -1
  5. package/dist/abis.js.map +1 -1
  6. package/dist/commands/backup.d.ts +3 -0
  7. package/dist/commands/backup.d.ts.map +1 -0
  8. package/dist/commands/backup.js +106 -0
  9. package/dist/commands/backup.js.map +1 -0
  10. package/dist/commands/compute.d.ts +14 -0
  11. package/dist/commands/compute.d.ts.map +1 -0
  12. package/dist/commands/compute.js +466 -0
  13. package/dist/commands/compute.js.map +1 -0
  14. package/dist/commands/config.d.ts.map +1 -1
  15. package/dist/commands/config.js +11 -1
  16. package/dist/commands/config.js.map +1 -1
  17. package/dist/commands/daemon.d.ts.map +1 -1
  18. package/dist/commands/daemon.js +67 -0
  19. package/dist/commands/daemon.js.map +1 -1
  20. package/dist/commands/discover.d.ts.map +1 -1
  21. package/dist/commands/discover.js +60 -15
  22. package/dist/commands/discover.js.map +1 -1
  23. package/dist/commands/doctor.d.ts +3 -0
  24. package/dist/commands/doctor.d.ts.map +1 -0
  25. package/dist/commands/doctor.js +205 -0
  26. package/dist/commands/doctor.js.map +1 -0
  27. package/dist/commands/wallet.d.ts.map +1 -1
  28. package/dist/commands/wallet.js +299 -65
  29. package/dist/commands/wallet.js.map +1 -1
  30. package/dist/commands/watch.d.ts.map +1 -1
  31. package/dist/commands/watch.js +146 -9
  32. package/dist/commands/watch.js.map +1 -1
  33. package/dist/commands/workroom.d.ts.map +1 -1
  34. package/dist/commands/workroom.js +112 -6
  35. package/dist/commands/workroom.js.map +1 -1
  36. package/dist/config.d.ts +12 -0
  37. package/dist/config.d.ts.map +1 -1
  38. package/dist/config.js +41 -4
  39. package/dist/config.js.map +1 -1
  40. package/dist/daemon/compute-metering.d.ts +61 -0
  41. package/dist/daemon/compute-metering.d.ts.map +1 -0
  42. package/dist/daemon/compute-metering.js +299 -0
  43. package/dist/daemon/compute-metering.js.map +1 -0
  44. package/dist/daemon/compute-session.d.ts +100 -0
  45. package/dist/daemon/compute-session.d.ts.map +1 -0
  46. package/dist/daemon/compute-session.js +231 -0
  47. package/dist/daemon/compute-session.js.map +1 -0
  48. package/dist/daemon/config.d.ts +33 -1
  49. package/dist/daemon/config.d.ts.map +1 -1
  50. package/dist/daemon/config.js +69 -0
  51. package/dist/daemon/config.js.map +1 -1
  52. package/dist/daemon/credentials.d.ts +24 -0
  53. package/dist/daemon/credentials.d.ts.map +1 -0
  54. package/dist/daemon/credentials.js +80 -0
  55. package/dist/daemon/credentials.js.map +1 -0
  56. package/dist/daemon/delivery-client.d.ts +35 -0
  57. package/dist/daemon/delivery-client.d.ts.map +1 -0
  58. package/dist/daemon/delivery-client.js +231 -0
  59. package/dist/daemon/delivery-client.js.map +1 -0
  60. package/dist/daemon/file-delivery.d.ts +98 -0
  61. package/dist/daemon/file-delivery.d.ts.map +1 -0
  62. package/dist/daemon/file-delivery.js +461 -0
  63. package/dist/daemon/file-delivery.js.map +1 -0
  64. package/dist/daemon/index.d.ts +1 -0
  65. package/dist/daemon/index.d.ts.map +1 -1
  66. package/dist/daemon/index.js +793 -227
  67. package/dist/daemon/index.js.map +1 -1
  68. package/dist/daemon/notify.d.ts +35 -6
  69. package/dist/daemon/notify.d.ts.map +1 -1
  70. package/dist/daemon/notify.js +176 -48
  71. package/dist/daemon/notify.js.map +1 -1
  72. package/dist/daemon/worker-executor.d.ts +71 -0
  73. package/dist/daemon/worker-executor.d.ts.map +1 -0
  74. package/dist/daemon/worker-executor.js +382 -0
  75. package/dist/daemon/worker-executor.js.map +1 -0
  76. package/dist/drain-v4.js +2 -2
  77. package/dist/drain-v4.js.map +1 -1
  78. package/dist/endpoint-notify.d.ts +9 -1
  79. package/dist/endpoint-notify.d.ts.map +1 -1
  80. package/dist/endpoint-notify.js +116 -3
  81. package/dist/endpoint-notify.js.map +1 -1
  82. package/dist/index.js +81 -1
  83. package/dist/index.js.map +1 -1
  84. package/dist/program.d.ts.map +1 -1
  85. package/dist/program.js +6 -0
  86. package/dist/program.js.map +1 -1
  87. package/dist/repl.d.ts.map +1 -1
  88. package/dist/repl.js +69 -486
  89. package/dist/repl.js.map +1 -1
  90. package/dist/tui/App.d.ts +12 -0
  91. package/dist/tui/App.d.ts.map +1 -0
  92. package/dist/tui/App.js +154 -0
  93. package/dist/tui/App.js.map +1 -0
  94. package/dist/tui/Footer.d.ts +11 -0
  95. package/dist/tui/Footer.d.ts.map +1 -0
  96. package/dist/tui/Footer.js +13 -0
  97. package/dist/tui/Footer.js.map +1 -0
  98. package/dist/tui/Header.d.ts +14 -0
  99. package/dist/tui/Header.d.ts.map +1 -0
  100. package/dist/tui/Header.js +19 -0
  101. package/dist/tui/Header.js.map +1 -0
  102. package/dist/tui/InputLine.d.ts +11 -0
  103. package/dist/tui/InputLine.d.ts.map +1 -0
  104. package/dist/tui/InputLine.js +145 -0
  105. package/dist/tui/InputLine.js.map +1 -0
  106. package/dist/tui/Viewport.d.ts +14 -0
  107. package/dist/tui/Viewport.d.ts.map +1 -0
  108. package/dist/tui/Viewport.js +48 -0
  109. package/dist/tui/Viewport.js.map +1 -0
  110. package/dist/tui/WalletConnectPairing.d.ts +23 -0
  111. package/dist/tui/WalletConnectPairing.d.ts.map +1 -0
  112. package/dist/tui/WalletConnectPairing.js +61 -0
  113. package/dist/tui/WalletConnectPairing.js.map +1 -0
  114. package/dist/tui/components/Button.d.ts +7 -0
  115. package/dist/tui/components/Button.d.ts.map +1 -0
  116. package/dist/tui/components/Button.js +21 -0
  117. package/dist/tui/components/Button.js.map +1 -0
  118. package/dist/tui/components/CeremonyView.d.ts +13 -0
  119. package/dist/tui/components/CeremonyView.d.ts.map +1 -0
  120. package/dist/tui/components/CeremonyView.js +10 -0
  121. package/dist/tui/components/CeremonyView.js.map +1 -0
  122. package/dist/tui/components/CompletionDropdown.d.ts +7 -0
  123. package/dist/tui/components/CompletionDropdown.d.ts.map +1 -0
  124. package/dist/tui/components/CompletionDropdown.js +23 -0
  125. package/dist/tui/components/CompletionDropdown.js.map +1 -0
  126. package/dist/tui/components/ConfirmPrompt.d.ts +9 -0
  127. package/dist/tui/components/ConfirmPrompt.d.ts.map +1 -0
  128. package/dist/tui/components/ConfirmPrompt.js +10 -0
  129. package/dist/tui/components/ConfirmPrompt.js.map +1 -0
  130. package/dist/tui/components/CustomTextInput.d.ts +15 -0
  131. package/dist/tui/components/CustomTextInput.d.ts.map +1 -0
  132. package/dist/tui/components/CustomTextInput.js +99 -0
  133. package/dist/tui/components/CustomTextInput.js.map +1 -0
  134. package/dist/tui/components/InteractiveTable.d.ts +14 -0
  135. package/dist/tui/components/InteractiveTable.d.ts.map +1 -0
  136. package/dist/tui/components/InteractiveTable.js +61 -0
  137. package/dist/tui/components/InteractiveTable.js.map +1 -0
  138. package/dist/tui/components/StepSpinner.d.ts +11 -0
  139. package/dist/tui/components/StepSpinner.d.ts.map +1 -0
  140. package/dist/tui/components/StepSpinner.js +32 -0
  141. package/dist/tui/components/StepSpinner.js.map +1 -0
  142. package/dist/tui/components/Toast.d.ts +18 -0
  143. package/dist/tui/components/Toast.d.ts.map +1 -0
  144. package/dist/tui/components/Toast.js +29 -0
  145. package/dist/tui/components/Toast.js.map +1 -0
  146. package/dist/tui/index.d.ts +2 -0
  147. package/dist/tui/index.d.ts.map +1 -0
  148. package/dist/tui/index.js +55 -0
  149. package/dist/tui/index.js.map +1 -0
  150. package/dist/tui/useChat.d.ts +11 -0
  151. package/dist/tui/useChat.d.ts.map +1 -0
  152. package/dist/tui/useChat.js +91 -0
  153. package/dist/tui/useChat.js.map +1 -0
  154. package/dist/tui/useCommand.d.ts +12 -0
  155. package/dist/tui/useCommand.d.ts.map +1 -0
  156. package/dist/tui/useCommand.js +137 -0
  157. package/dist/tui/useCommand.js.map +1 -0
  158. package/dist/tui/useNotifications.d.ts +9 -0
  159. package/dist/tui/useNotifications.d.ts.map +1 -0
  160. package/dist/tui/useNotifications.js +17 -0
  161. package/dist/tui/useNotifications.js.map +1 -0
  162. package/dist/tui/useScroll.d.ts +17 -0
  163. package/dist/tui/useScroll.d.ts.map +1 -0
  164. package/dist/tui/useScroll.js +46 -0
  165. package/dist/tui/useScroll.js.map +1 -0
  166. package/dist/ui/format.d.ts.map +1 -1
  167. package/dist/ui/format.js +2 -0
  168. package/dist/ui/format.js.map +1 -1
  169. package/dist/ui/qr-render.d.ts +25 -0
  170. package/dist/ui/qr-render.d.ts.map +1 -0
  171. package/dist/ui/qr-render.js +90 -0
  172. package/dist/ui/qr-render.js.map +1 -0
  173. package/dist/ui/rpc-fallback.d.ts +11 -0
  174. package/dist/ui/rpc-fallback.d.ts.map +1 -0
  175. package/dist/ui/rpc-fallback.js +58 -0
  176. package/dist/ui/rpc-fallback.js.map +1 -0
  177. package/dist/walletconnect.d.ts +4 -0
  178. package/dist/walletconnect.d.ts.map +1 -1
  179. package/dist/walletconnect.js.map +1 -1
  180. package/package.json +10 -2
  181. package/scripts/authorize-machine-key.ts +0 -43
  182. package/scripts/drain-wallet.ts +0 -149
  183. package/scripts/execute-spend-only.ts +0 -81
  184. package/scripts/register-agent-userop.ts +0 -186
  185. package/src/abis.ts +0 -187
  186. package/src/bundler.ts +0 -235
  187. package/src/client.ts +0 -36
  188. package/src/coinbase-smart-wallet.ts +0 -51
  189. package/src/commands/accept.ts +0 -64
  190. package/src/commands/agent-handshake.ts +0 -72
  191. package/src/commands/agent.ts +0 -691
  192. package/src/commands/agreements.ts +0 -350
  193. package/src/commands/arbitrator.ts +0 -180
  194. package/src/commands/arena-handshake.ts +0 -257
  195. package/src/commands/arena.ts +0 -122
  196. package/src/commands/cancel.ts +0 -35
  197. package/src/commands/channel.ts +0 -218
  198. package/src/commands/coldstart.ts +0 -165
  199. package/src/commands/config.ts +0 -58
  200. package/src/commands/contract-interaction.ts +0 -166
  201. package/src/commands/daemon.ts +0 -978
  202. package/src/commands/deliver.ts +0 -148
  203. package/src/commands/discover.ts +0 -297
  204. package/src/commands/dispute.ts +0 -375
  205. package/src/commands/endpoint.ts +0 -620
  206. package/src/commands/feed.ts +0 -229
  207. package/src/commands/hire.ts +0 -245
  208. package/src/commands/migrate.ts +0 -177
  209. package/src/commands/negotiate.ts +0 -271
  210. package/src/commands/openshell.ts +0 -1055
  211. package/src/commands/owner.ts +0 -35
  212. package/src/commands/policy.ts +0 -263
  213. package/src/commands/relay.ts +0 -273
  214. package/src/commands/remediate.ts +0 -24
  215. package/src/commands/reputation.ts +0 -79
  216. package/src/commands/setup.ts +0 -343
  217. package/src/commands/trust.ts +0 -27
  218. package/src/commands/verify.ts +0 -91
  219. package/src/commands/wallet.ts +0 -3280
  220. package/src/commands/watch.ts +0 -23
  221. package/src/commands/watchtower.ts +0 -248
  222. package/src/commands/workroom.ts +0 -959
  223. package/src/config.ts +0 -174
  224. package/src/daemon/config.ts +0 -308
  225. package/src/daemon/hire-listener.ts +0 -226
  226. package/src/daemon/index.ts +0 -955
  227. package/src/daemon/job-lifecycle.ts +0 -215
  228. package/src/daemon/notify.ts +0 -157
  229. package/src/daemon/token-metering.ts +0 -183
  230. package/src/daemon/userops.ts +0 -119
  231. package/src/daemon/wallet-monitor.ts +0 -90
  232. package/src/drain-v4.ts +0 -159
  233. package/src/endpoint-config.ts +0 -83
  234. package/src/endpoint-notify.ts +0 -46
  235. package/src/index.ts +0 -26
  236. package/src/openshell-runtime.ts +0 -277
  237. package/src/program.ts +0 -83
  238. package/src/repl.ts +0 -680
  239. package/src/signing.ts +0 -28
  240. package/src/telegram-notify.ts +0 -88
  241. package/src/ui/banner.ts +0 -51
  242. package/src/ui/colors.ts +0 -30
  243. package/src/ui/format.ts +0 -77
  244. package/src/ui/spinner.ts +0 -56
  245. package/src/ui/tree.ts +0 -16
  246. package/src/utils/format.ts +0 -48
  247. package/src/utils/hash.ts +0 -5
  248. package/src/utils/time.ts +0 -15
  249. package/src/wallet-router.ts +0 -178
  250. package/src/walletconnect-session.ts +0 -27
  251. package/src/walletconnect.ts +0 -294
  252. package/test/time.test.js +0 -11
  253. package/tsconfig.json +0 -19
@@ -1,350 +0,0 @@
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
- import { c } from "../ui/colors";
11
- import { renderTree } from "../ui/tree";
12
- import { formatAddress } from "../ui/format";
13
-
14
- // ─── AgreementTree minimal ABI ────────────────────────────────────────────────
15
-
16
- const AGREEMENT_TREE_ABI = [
17
- "function registerSubAgreement(uint256 parentAgreementId, uint256 childAgreementId) external",
18
- "function getChildren(uint256 agreementId) external view returns (uint256[])",
19
- "function getRoot(uint256 agreementId) external view returns (uint256)",
20
- "function getPath(uint256 agreementId) external view returns (uint256[])",
21
- "function allChildrenSettled(uint256 agreementId) external view returns (bool)",
22
- "function getDepth(uint256 agreementId) external view returns (uint256)",
23
- ];
24
-
25
- export function registerAgreementsCommands(program: Command): void {
26
-
27
- // ── arc402 agreements ────────────────────────────────────────────────────────
28
- program
29
- .command("agreements")
30
- .description("List agreements for the configured wallet")
31
- .option("--as <role>", "client or provider", "client")
32
- .option("--json")
33
- .action(async (opts) => {
34
- const config = loadConfig();
35
- if (!config.serviceAgreementAddress) throw new Error("serviceAgreementAddress missing in config");
36
- const { provider, address } = await getClient(config);
37
- if (!address) throw new Error("No wallet configured");
38
- const client = new ServiceAgreementClient(config.serviceAgreementAddress, provider);
39
- const agreements = opts.as === "provider"
40
- ? await client.getProviderAgreements(address)
41
- : await client.getClientAgreements(address);
42
- if (opts.json) return console.log(JSON.stringify(agreements, (_k, value) => typeof value === "bigint" ? value.toString() : value, 2));
43
- printTable(
44
- ["ID", "COUNTERPARTY", "SERVICE", "DEADLINE", "STATUS"],
45
- agreements.map((agreement) => [
46
- agreement.id.toString(),
47
- truncateAddress(opts.as === "provider" ? agreement.client : agreement.provider),
48
- agreement.serviceType,
49
- formatDeadline(Number(agreement.deadline)),
50
- agreementStatusLabel(agreement.status),
51
- ])
52
- );
53
- });
54
-
55
- // ── arc402 agreement <id> ────────────────────────────────────────────────────
56
- program
57
- .command("agreement <id>")
58
- .description("Show agreement detail, including remediation/dispute fields")
59
- .option("--json")
60
- .action(async (id, opts) => {
61
- const config = loadConfig();
62
- if (!config.serviceAgreementAddress) throw new Error("serviceAgreementAddress missing in config");
63
- const { provider } = await getClient(config);
64
- const client = new ServiceAgreementClient(config.serviceAgreementAddress, provider);
65
- const agreement = await client.getAgreement(BigInt(id));
66
- if (opts.json) return console.log(JSON.stringify(agreement, (_k, value) => typeof value === "bigint" ? value.toString() : value, 2));
67
- console.log('\n ' + c.mark + c.white(` Agreement #${agreement.id}`));
68
- renderTree([
69
- { label: 'Client', value: formatAddress(agreement.client) },
70
- { label: 'Provider', value: formatAddress(agreement.provider) },
71
- { label: 'Status', value: agreementStatusLabel(agreement.status) },
72
- { label: 'Created', value: formatDate(Number(agreement.createdAt)) },
73
- { label: 'Deadline', value: formatDate(Number(agreement.deadline)) },
74
- { label: 'Verify end', value: Number(agreement.verifyWindowEnd) ? formatDate(Number(agreement.verifyWindowEnd)) : 'n/a' },
75
- { label: 'Hash', value: String(agreement.committedHash), last: true },
76
- ]);
77
- if ([AgreementStatus.REVISION_REQUESTED, AgreementStatus.REVISED, AgreementStatus.PARTIAL_SETTLEMENT, AgreementStatus.ESCALATED_TO_HUMAN, AgreementStatus.DISPUTED, AgreementStatus.ESCALATED_TO_ARBITRATION].includes(agreement.status)) {
78
- const remediation = await client.getRemediationCase(agreement.id);
79
- const dispute = await client.getDisputeCase(agreement.id);
80
- renderTree([
81
- { label: 'Remediation', value: `active=${remediation.active} cycles=${remediation.cycleCount}` },
82
- { label: 'Dispute', value: `outcome=${dispute.outcome}`, last: true },
83
- ]);
84
- }
85
- });
86
-
87
- // ── arc402 agreements-sub-register ──────────────────────────────────────────
88
- // Spec 19: arc402 agreements sub-register --parent <id> --child <id>
89
- program
90
- .command("agreements-sub-register")
91
- .description("Link a child agreement to its parent in the AgreementTree (Spec 19)")
92
- .requiredOption("--parent <id>", "Parent agreement ID")
93
- .requiredOption("--child <id>", "Child agreement ID to register under the parent")
94
- .option("--json", "Machine-parseable output")
95
- .action(async (opts) => {
96
- const config = loadConfig();
97
- if (!config.agreementTreeAddress) throw new Error("agreementTreeAddress missing in config");
98
- const { signer } = await getClient(config);
99
- if (!signer) { console.error("No private key configured."); process.exit(1); }
100
-
101
- const tree = new ethers.Contract(config.agreementTreeAddress, AGREEMENT_TREE_ABI, signer);
102
- const tx = await tree.registerSubAgreement(BigInt(opts.parent), BigInt(opts.child));
103
- const receipt = await tx.wait();
104
-
105
- if (opts.json) {
106
- console.log(JSON.stringify({ txHash: receipt.hash, parentId: opts.parent, childId: opts.child }));
107
- } else {
108
- console.log('\n ' + c.mark + c.white(' Sub-agreement registered'));
109
- renderTree([
110
- { label: 'Parent', value: `#${opts.parent}` },
111
- { label: 'Child', value: `#${opts.child}` },
112
- { label: 'Tx', value: receipt.hash, last: true },
113
- ]);
114
- }
115
- });
116
-
117
- // ── arc402 agreements-tree <id> ──────────────────────────────────────────────
118
- // Spec 19: arc402 agreements tree <agreementId>
119
- program
120
- .command("agreements-tree <agreementId>")
121
- .description("View full agreement tree for an agreement (Spec 19)")
122
- .option("--json", "Machine-parseable output")
123
- .action(async (agreementId, opts) => {
124
- const config = loadConfig();
125
- if (!config.agreementTreeAddress) throw new Error("agreementTreeAddress missing in config");
126
- const { provider } = await getClient(config);
127
-
128
- const tree = new ethers.Contract(config.agreementTreeAddress, AGREEMENT_TREE_ABI, provider);
129
-
130
- const id = BigInt(agreementId);
131
- const [root, path, children, depth] = await Promise.all([
132
- tree.getRoot(id),
133
- tree.getPath(id),
134
- tree.getChildren(id),
135
- tree.getDepth(id),
136
- ]);
137
-
138
- const result = {
139
- agreementId: agreementId,
140
- root: root.toString(),
141
- path: (path as bigint[]).map(String),
142
- children: (children as bigint[]).map(String),
143
- depth: Number(depth),
144
- };
145
-
146
- if (opts.json) {
147
- console.log(JSON.stringify(result));
148
- return;
149
- }
150
-
151
- console.log('\n ' + c.mark + c.white(` Agreement Tree — node #${agreementId}`));
152
- renderTree([
153
- { label: 'Root', value: `#${result.root}` },
154
- { label: 'Depth', value: String(result.depth) },
155
- { label: 'Path', value: result.path.map((p) => "#" + p).join(" → ") },
156
- { label: 'Children', value: result.children.length > 0 ? result.children.map((ch) => "#" + ch).join(", ") : "(none)", last: true },
157
- ]);
158
- });
159
-
160
- // ── arc402 agreements-tree-status <id> ───────────────────────────────────────
161
- // Spec 19: arc402 agreements tree-status <agreementId>
162
- program
163
- .command("agreements-tree-status <agreementId>")
164
- .description("Check whether all sub-agreements are settled before delivering (Spec 19)")
165
- .option("--json", "Machine-parseable output")
166
- .action(async (agreementId, opts) => {
167
- const config = loadConfig();
168
- if (!config.agreementTreeAddress) throw new Error("agreementTreeAddress missing in config");
169
- const { provider } = await getClient(config);
170
-
171
- const tree = new ethers.Contract(config.agreementTreeAddress, AGREEMENT_TREE_ABI, provider);
172
-
173
- const id = BigInt(agreementId);
174
- const [allSettled, children] = await Promise.all([
175
- tree.allChildrenSettled(id),
176
- tree.getChildren(id),
177
- ]);
178
-
179
- const result = {
180
- agreementId: agreementId,
181
- childCount: (children as bigint[]).length,
182
- allChildrenSettled: Boolean(allSettled),
183
- readyToDeliver: Boolean(allSettled),
184
- };
185
-
186
- if (opts.json) {
187
- console.log(JSON.stringify(result));
188
- return;
189
- }
190
-
191
- console.log('\n ' + c.mark + c.white(` Agreement #${agreementId} — tree status`));
192
- renderTree([
193
- { label: 'Sub-agrmts', value: String(result.childCount) },
194
- { label: 'All settled', value: result.allChildrenSettled ? 'YES' : 'NO' },
195
- { label: 'Ready', value: result.readyToDeliver ? 'YES — ready to deliver to parent' : 'NO', last: true },
196
- ]);
197
- });
198
-
199
- // ── arc402 agreements-create-tree ────────────────────────────────────────────
200
- // Spec 19: interactive multi-party agreement setup
201
- // Usage: arc402 agreements-create-tree \
202
- // --provider <addr> --task <desc> --service-type <type> --price <wei> --deadline <s> \
203
- // --sub <addr>,<task>,<type>,<wei>,<deadline> [--sub ...]
204
- program
205
- .command("agreements-create-tree")
206
- .description("Create a multi-party agreement tree: one root agreement plus sub-agreements (Spec 19)")
207
- .requiredOption("--provider <address>", "Root provider (Agent B)")
208
- .requiredOption("--task <description>", "Root task description")
209
- .requiredOption("--service-type <type>", "Root service type")
210
- .requiredOption("--price <wei>", "Root price in wei")
211
- .requiredOption("--deadline <seconds>", "Root deadline offset in seconds from now")
212
- .option(
213
- "--sub <spec>",
214
- "Sub-agreement in format <provider>,<task>,<serviceType>,<priceWei>,<deadlineSeconds>. Repeatable.",
215
- (val: string, acc: string[]) => { acc.push(val); return acc; },
216
- [] as string[],
217
- )
218
- .option("--interactive", "Prompt interactively for sub-agreements")
219
- .option("--json", "Machine-parseable output")
220
- .action(async (opts) => {
221
- const config = loadConfig();
222
- if (!config.serviceAgreementAddress) throw new Error("serviceAgreementAddress missing in config");
223
- if (!config.agreementTreeAddress) throw new Error("agreementTreeAddress missing in config");
224
- const { signer } = await getClient(config);
225
- if (!signer) { console.error("No private key configured."); process.exit(1); }
226
-
227
- type SubSpec = { provider: string; task: string; serviceType: string; price: bigint; deadline: number };
228
- const subSpecs: SubSpec[] = [];
229
-
230
- // Parse --sub flags
231
- for (const raw of opts.sub as string[]) {
232
- const parts = raw.split(",");
233
- if (parts.length < 5) {
234
- console.error(`--sub format: <provider>,<task>,<serviceType>,<priceWei>,<deadlineSeconds> got: ${raw}`);
235
- process.exit(1);
236
- }
237
- subSpecs.push({
238
- provider: parts[0],
239
- task: parts.slice(1, parts.length - 3).join(",") || parts[1],
240
- serviceType: parts[parts.length - 3],
241
- price: BigInt(parts[parts.length - 2]),
242
- deadline: Number(parts[parts.length - 1]),
243
- });
244
- }
245
-
246
- // Interactive mode: prompt for sub-agreements if no --sub flags provided
247
- if (opts.interactive || subSpecs.length === 0) {
248
- const rl = readline.createInterface({ input: process.stdin, output: process.stdout });
249
- const ask = (q: string) => new Promise<string>((res) => rl.question(q, res));
250
-
251
- console.log("\nMulti-party agreement setup — add sub-agreements (leave provider blank to finish)");
252
- while (true) {
253
- const provider = await ask(" Sub-provider address (blank to finish): ");
254
- if (!provider.trim()) break;
255
- const task = await ask(" Task description: ");
256
- const serviceType = await ask(" Service type: ");
257
- const priceRaw = await ask(" Price (wei): ");
258
- const deadlineRaw = await ask(" Deadline offset (seconds): ");
259
- subSpecs.push({
260
- provider: provider.trim(),
261
- task: task.trim(),
262
- serviceType: serviceType.trim(),
263
- price: BigInt(priceRaw.trim()),
264
- deadline: Number(deadlineRaw.trim()),
265
- });
266
- }
267
- rl.close();
268
- }
269
-
270
- const saContract = new ethers.Contract(config.serviceAgreementAddress, [
271
- "function propose(address provider, string serviceType, string description, uint256 price, address token, uint256 deadline, bytes32 deliverablesHash) external payable returns (uint256)",
272
- "event AgreementProposed(uint256 indexed id, address indexed client, address indexed provider, string serviceType, uint256 price, address token, uint256 deadline)",
273
- ], signer);
274
- const treeContract = new ethers.Contract(config.agreementTreeAddress, AGREEMENT_TREE_ABI, signer);
275
-
276
- const rootPrice = BigInt(opts.price);
277
- const rootDeadline = Number(opts.deadline);
278
- const rootHash = hashString(opts.task);
279
-
280
- if (!opts.json) console.log('\n ' + c.mark + c.white(` Proposing root agreement to ${opts.provider}...`));
281
- const rootTx = await saContract.propose(
282
- opts.provider, opts.serviceType, opts.task,
283
- rootPrice, ethers.ZeroAddress, rootDeadline, rootHash,
284
- { value: rootPrice },
285
- );
286
- const rootReceipt = await rootTx.wait();
287
- const iface = new ethers.Interface([
288
- "event AgreementProposed(uint256 indexed id, address indexed client, address indexed provider, string serviceType, uint256 price, address token, uint256 deadline)",
289
- ]);
290
- let rootAgreementId: bigint | undefined;
291
- for (const log of rootReceipt.logs) {
292
- try {
293
- const parsed = iface.parseLog(log);
294
- if (parsed?.name === "AgreementProposed") { rootAgreementId = BigInt(parsed.args.id); break; }
295
- } catch { /* skip */ }
296
- }
297
- if (rootAgreementId === undefined) throw new Error("Could not parse root AgreementProposed event");
298
-
299
- if (!opts.json) console.log(' ' + c.success + c.white(` Root agreement ID: #${rootAgreementId}`));
300
-
301
- const childAgreementIds: bigint[] = [];
302
- for (const sub of subSpecs) {
303
- if (!opts.json) console.log(' ' + c.dim(` Proposing sub-agreement to ${sub.provider}...`));
304
- const subHash = hashString(sub.task);
305
- const subTx = await saContract.propose(
306
- sub.provider, sub.serviceType, sub.task,
307
- sub.price, ethers.ZeroAddress, sub.deadline, subHash,
308
- { value: sub.price },
309
- );
310
- const subReceipt = await subTx.wait();
311
- let childId: bigint | undefined;
312
- for (const log of subReceipt.logs) {
313
- try {
314
- const parsed = iface.parseLog(log);
315
- if (parsed?.name === "AgreementProposed") { childId = BigInt(parsed.args.id); break; }
316
- } catch { /* skip */ }
317
- }
318
- if (childId === undefined) throw new Error(`Could not parse AgreementProposed for sub-agreement to ${sub.provider}`);
319
- if (!opts.json) console.log(' ' + c.success + c.white(` Sub-agreement #${childId} — registering in tree...`));
320
- await (await treeContract.registerSubAgreement(rootAgreementId, childId)).wait();
321
- childAgreementIds.push(childId);
322
- }
323
-
324
- const result = {
325
- rootAgreementId: rootAgreementId.toString(),
326
- childAgreementIds: childAgreementIds.map(String),
327
- txHash: rootReceipt.hash,
328
- };
329
-
330
- if (opts.json) {
331
- console.log(JSON.stringify(result, null, 2));
332
- } else {
333
- console.log('\n ' + c.mark + c.white(' Agreement tree created'));
334
- renderTree([
335
- { label: 'Root', value: `#${result.rootAgreementId}` },
336
- {
337
- label: 'Children',
338
- value: result.childAgreementIds.length > 0
339
- ? result.childAgreementIds.map((id) => "#" + id).join(", ")
340
- : "(none — sub-agreements can be added with agreements-sub-register)",
341
- last: true,
342
- },
343
- ]);
344
- }
345
- });
346
-
347
- // ── arc402 agreements-tree <id> (alias) already exists above ─────────────────
348
- // The spec also refers to this as "arc402 agreements tree <id>" — the flat
349
- // command agreements-tree <agreementId> above fulfils that requirement.
350
- }
@@ -1,180 +0,0 @@
1
- import { Command } from "commander";
2
- import { DisputeArbitrationClient } from "@arc402/sdk";
3
- import { loadConfig } from "../config";
4
- import { getClient, requireSigner } from "../client";
5
- import { c } from "../ui/colors";
6
- import { startSpinner } from "../ui/spinner";
7
- import { renderTree } from "../ui/tree";
8
-
9
- export function registerArbitratorCommand(program: Command): void {
10
- const arbitrator = program
11
- .command("arbitrator")
12
- .description("Arbitrator panel operations: bonds, eligibility, status");
13
-
14
- // Bond subcommands — nest under 'bond' to avoid duplicate registration
15
- const bond = arbitrator
16
- .command("bond")
17
- .description("Arbitrator bond operations: status, accept, fallback");
18
-
19
- bond
20
- .command("status <arbitratorAddress> [agreementId]")
21
- .description("Check arbitrator bond status for an agreement (or general eligibility)")
22
- .action(async (arbitratorAddress, agreementId, _opts) => {
23
- const config = loadConfig();
24
- if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
25
- const { provider } = await getClient(config);
26
- const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, provider);
27
-
28
- const eligible = await client.isEligibleArbitrator(arbitratorAddress);
29
- console.log('\n ' + c.mark + c.white(' Arbitrator status'));
30
- renderTree([
31
- { label: 'Address', value: arbitratorAddress },
32
- { label: 'Eligible', value: eligible ? 'yes' : 'no', last: !agreementId },
33
- ]);
34
-
35
- if (agreementId) {
36
- const bondState = await client.getArbitratorBondState(arbitratorAddress, BigInt(agreementId));
37
- console.log('\n ' + c.mark + c.white(` Bond state — agreement #${agreementId}`));
38
- renderTree([
39
- { label: 'Amount', value: `${bondState.bondAmount.toString()} tokens` },
40
- { label: 'Locked', value: String(bondState.locked) },
41
- { label: 'Slashed', value: String(bondState.slashed) },
42
- { label: 'Returned', value: String(bondState.returned) },
43
- { label: 'Locked at', value: new Date(Number(bondState.lockedAt) * 1000).toISOString(), last: true },
44
- ]);
45
- }
46
- });
47
-
48
- bond
49
- .command("accept <agreementId>")
50
- .description("Accept panel assignment and post bond")
51
- .option("--bond <bond>", "Bond amount in wei (for ETH agreements)", "0")
52
- .action(async (agreementId, opts) => {
53
- const config = loadConfig();
54
- if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
55
- const { signer } = await requireSigner(config);
56
- const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, signer);
57
-
58
- const bondAmount = BigInt(opts.bond);
59
- if (bondAmount === 0n) {
60
- console.warn(' ' + c.warning + c.white(' Bond amount is 0. For ERC-20 agreements, pre-approve the DisputeArbitration contract.'));
61
- }
62
-
63
- const spinner = startSpinner("Submitting transaction...");
64
- try {
65
- await client.acceptAssignment(BigInt(agreementId), bondAmount);
66
- spinner.succeed();
67
- } catch (err) {
68
- spinner.fail();
69
- throw err;
70
- }
71
- console.log(' ' + c.success + c.white(` Assignment accepted — agreement #${agreementId}`));
72
- });
73
-
74
- bond
75
- .command("fallback <agreementId>")
76
- .description("Trigger fallback to human backstop (mutual unfunded or panel incomplete)")
77
- .action(async (agreementId, _opts) => {
78
- const config = loadConfig();
79
- if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
80
- const { signer } = await requireSigner(config);
81
- const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, signer);
82
- const spinner = startSpinner("Submitting transaction...");
83
- try {
84
- await client.triggerFallback(BigInt(agreementId));
85
- spinner.succeed();
86
- } catch (err) {
87
- spinner.fail();
88
- throw err;
89
- }
90
- console.log(' ' + c.success + c.white(` Fallback triggered — agreement #${agreementId}`));
91
- });
92
-
93
- // Rate subcommands — nest under 'rate'
94
- const rate = arbitrator
95
- .command("rate")
96
- .description("Token USD rate management (owner only)");
97
-
98
- rate
99
- .command("set <tokenAddress> <usdPerToken>")
100
- .description("Set USD rate for a token. Rate in USD with 18 decimals. e.g. 2000e18 for $2000")
101
- .action(async (tokenAddress, usdPerToken, _opts) => {
102
- const config = loadConfig();
103
- if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
104
- const { signer } = await requireSigner(config);
105
- const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, signer);
106
-
107
- const rateVal = BigInt(usdPerToken);
108
- const spinner = startSpinner("Submitting transaction...");
109
- try {
110
- await client.setTokenUsdRate(tokenAddress, rateVal);
111
- spinner.succeed();
112
- } catch (err) {
113
- spinner.fail();
114
- throw err;
115
- }
116
- console.log(' ' + c.success + c.white(` Token rate set: ${tokenAddress} = ${rateVal.toString()} USD/token`));
117
- });
118
-
119
- // Admin: slash arbitrator (manual rules violation)
120
- arbitrator
121
- .command("slash <agreementId> <arbitratorAddress> <reason>")
122
- .description("Owner only: manually slash an arbitrator for rules violation")
123
- .action(async (agreementId, arbitratorAddress, reason, _opts) => {
124
- const config = loadConfig();
125
- if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
126
- const { signer } = await requireSigner(config);
127
- const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, signer);
128
- const spinner = startSpinner("Submitting transaction...");
129
- try {
130
- await client.slashArbitrator(BigInt(agreementId), arbitratorAddress, reason);
131
- spinner.succeed();
132
- } catch (err) {
133
- spinner.fail();
134
- throw err;
135
- }
136
- console.log(' ' + c.success + c.white(` Slashed ${arbitratorAddress}`));
137
- });
138
-
139
- arbitrator
140
- .command("reclaim-bond <agreementId>")
141
- .description("Reclaim an arbitrator bond after 45 days if dispute was never resolved via resolveDisputeFee")
142
- .action(async (agreementId, _opts) => {
143
- const config = loadConfig();
144
- if (!config.disputeArbitrationAddress) throw new Error("disputeArbitrationAddress missing in config");
145
- const { signer } = await requireSigner(config);
146
- const client = new DisputeArbitrationClient(config.disputeArbitrationAddress, signer);
147
- const spinner = startSpinner("Submitting transaction...");
148
- try {
149
- await client.reclaimExpiredBond(BigInt(agreementId));
150
- spinner.succeed();
151
- } catch (err) {
152
- spinner.fail();
153
- throw err;
154
- }
155
- console.log(' ' + c.success + c.white(` Bond reclaimed — agreement #${agreementId}`));
156
- });
157
-
158
- arbitrator
159
- .command("resolve-from <agreementId>")
160
- .description("Resolve a dispute from arbitration with split amounts (calls ServiceAgreement.resolveFromArbitration)")
161
- .requiredOption("--recipient <address>", "Winning recipient address")
162
- .requiredOption("--provider-amount <amount>", "Provider payout in wei/tokens")
163
- .requiredOption("--client-amount <amount>", "Client refund in wei/tokens")
164
- .action(async (agreementId, opts) => {
165
- const { ServiceAgreementClient } = await import("@arc402/sdk");
166
- const config = loadConfig();
167
- if (!config.serviceAgreementAddress) throw new Error("serviceAgreementAddress missing in config");
168
- const { signer } = await requireSigner(config);
169
- const client = new ServiceAgreementClient(config.serviceAgreementAddress, signer);
170
- const spinner = startSpinner("Submitting transaction...");
171
- try {
172
- await client.resolveFromArbitration(BigInt(agreementId), opts.recipient, BigInt(opts.providerAmount), BigInt(opts.clientAmount));
173
- spinner.succeed();
174
- } catch (err) {
175
- spinner.fail();
176
- throw err;
177
- }
178
- console.log(' ' + c.success + c.white(` Resolved from arbitration — agreement #${agreementId}`));
179
- });
180
- }