safehands-pharos 1.2.0 → 1.2.3

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 (175) hide show
  1. package/.env.example +26 -0
  2. package/README.md +311 -350
  3. package/contracts/RiskRegistry.json +75 -1
  4. package/contracts/RiskRegistry.sol +29 -1
  5. package/dist/cli.d.ts +6 -0
  6. package/dist/cli.d.ts.map +1 -0
  7. package/dist/cli.js +91 -0
  8. package/dist/cli.js.map +1 -0
  9. package/dist/demo.d.ts +2 -0
  10. package/dist/demo.d.ts.map +1 -0
  11. package/dist/demo.js +172 -0
  12. package/dist/demo.js.map +1 -0
  13. package/dist/index.js +181 -169
  14. package/dist/index.js.map +1 -1
  15. package/dist/init.d.ts +2 -0
  16. package/dist/init.d.ts.map +1 -0
  17. package/dist/init.js +66 -0
  18. package/dist/init.js.map +1 -0
  19. package/dist/lib/constants.d.ts +122 -7
  20. package/dist/lib/constants.d.ts.map +1 -1
  21. package/dist/lib/constants.js +139 -13
  22. package/dist/lib/constants.js.map +1 -1
  23. package/dist/lib/dodoApi.d.ts +14 -0
  24. package/dist/lib/dodoApi.d.ts.map +1 -1
  25. package/dist/lib/dodoApi.js +78 -22
  26. package/dist/lib/dodoApi.js.map +1 -1
  27. package/dist/lib/http.d.ts +15 -0
  28. package/dist/lib/http.d.ts.map +1 -0
  29. package/dist/lib/http.js +119 -0
  30. package/dist/lib/http.js.map +1 -0
  31. package/dist/lib/pharosClient.d.ts +4 -3
  32. package/dist/lib/pharosClient.d.ts.map +1 -1
  33. package/dist/lib/pharosClient.js +8 -5
  34. package/dist/lib/pharosClient.js.map +1 -1
  35. package/dist/lib/policy/actionPolicyEngine.d.ts +54 -0
  36. package/dist/lib/policy/actionPolicyEngine.d.ts.map +1 -0
  37. package/dist/lib/policy/actionPolicyEngine.js +213 -0
  38. package/dist/lib/policy/actionPolicyEngine.js.map +1 -0
  39. package/dist/lib/signer/index.d.ts +25 -0
  40. package/dist/lib/signer/index.d.ts.map +1 -0
  41. package/dist/lib/signer/index.js +90 -0
  42. package/dist/lib/signer/index.js.map +1 -0
  43. package/dist/lib/testDodoLive.d.ts +2 -0
  44. package/dist/lib/testDodoLive.d.ts.map +1 -0
  45. package/dist/lib/testDodoLive.js +105 -0
  46. package/dist/lib/testDodoLive.js.map +1 -0
  47. package/dist/lib/testLiveSafehands.d.ts +2 -0
  48. package/dist/lib/testLiveSafehands.d.ts.map +1 -0
  49. package/dist/lib/testLiveSafehands.js +93 -0
  50. package/dist/lib/testLiveSafehands.js.map +1 -0
  51. package/dist/lib/testRpcLive.d.ts +2 -0
  52. package/dist/lib/testRpcLive.d.ts.map +1 -0
  53. package/dist/lib/testRpcLive.js +89 -0
  54. package/dist/lib/testRpcLive.js.map +1 -0
  55. package/dist/lib/testTools.js +363 -354
  56. package/dist/lib/testTools.js.map +1 -1
  57. package/dist/lib/testX402Live.d.ts +2 -0
  58. package/dist/lib/testX402Live.d.ts.map +1 -0
  59. package/dist/lib/testX402Live.js +160 -0
  60. package/dist/lib/testX402Live.js.map +1 -0
  61. package/dist/lib/toolResponse.d.ts +26 -0
  62. package/dist/lib/toolResponse.d.ts.map +1 -0
  63. package/dist/lib/toolResponse.js +54 -0
  64. package/dist/lib/toolResponse.js.map +1 -0
  65. package/dist/lib/wallet/index.d.ts +19 -0
  66. package/dist/lib/wallet/index.d.ts.map +1 -0
  67. package/dist/lib/wallet/index.js +71 -0
  68. package/dist/lib/wallet/index.js.map +1 -0
  69. package/dist/tools/approveToken.d.ts +19 -20
  70. package/dist/tools/approveToken.d.ts.map +1 -1
  71. package/dist/tools/approveToken.js +44 -21
  72. package/dist/tools/approveToken.js.map +1 -1
  73. package/dist/tools/assessRisk.d.ts +22 -9
  74. package/dist/tools/assessRisk.d.ts.map +1 -1
  75. package/dist/tools/assessRisk.js +32 -9
  76. package/dist/tools/assessRisk.js.map +1 -1
  77. package/dist/tools/checkAllowance.d.ts +6 -6
  78. package/dist/tools/checkTokenSecurity.d.ts +9 -16
  79. package/dist/tools/checkTokenSecurity.d.ts.map +1 -1
  80. package/dist/tools/checkTokenSecurity.js +17 -22
  81. package/dist/tools/checkTokenSecurity.js.map +1 -1
  82. package/dist/tools/createAgentWallet.d.ts +27 -0
  83. package/dist/tools/createAgentWallet.d.ts.map +1 -0
  84. package/dist/tools/createAgentWallet.js +60 -0
  85. package/dist/tools/createAgentWallet.js.map +1 -0
  86. package/dist/tools/estimateGas.d.ts +31 -21
  87. package/dist/tools/estimateGas.d.ts.map +1 -1
  88. package/dist/tools/estimateGas.js +91 -95
  89. package/dist/tools/estimateGas.js.map +1 -1
  90. package/dist/tools/executeSwap.d.ts +13 -29
  91. package/dist/tools/executeSwap.d.ts.map +1 -1
  92. package/dist/tools/executeSwap.js +68 -46
  93. package/dist/tools/executeSwap.js.map +1 -1
  94. package/dist/tools/explainRisk.d.ts +30 -0
  95. package/dist/tools/explainRisk.d.ts.map +1 -0
  96. package/dist/tools/explainRisk.js +33 -0
  97. package/dist/tools/explainRisk.js.map +1 -0
  98. package/dist/tools/getAgentWallet.d.ts +22 -0
  99. package/dist/tools/getAgentWallet.d.ts.map +1 -0
  100. package/dist/tools/getAgentWallet.js +28 -0
  101. package/dist/tools/getAgentWallet.js.map +1 -0
  102. package/dist/tools/getAgentWalletBalance.d.ts +12 -0
  103. package/dist/tools/getAgentWalletBalance.d.ts.map +1 -0
  104. package/dist/tools/getAgentWalletBalance.js +71 -0
  105. package/dist/tools/getAgentWalletBalance.js.map +1 -0
  106. package/dist/tools/getExecutionHistory.d.ts +4 -4
  107. package/dist/tools/getGasPrice.d.ts +26 -8
  108. package/dist/tools/getGasPrice.d.ts.map +1 -1
  109. package/dist/tools/getGasPrice.js +43 -35
  110. package/dist/tools/getGasPrice.js.map +1 -1
  111. package/dist/tools/getPoolInfo.d.ts +47 -59
  112. package/dist/tools/getPoolInfo.d.ts.map +1 -1
  113. package/dist/tools/getPoolInfo.js +96 -57
  114. package/dist/tools/getPoolInfo.js.map +1 -1
  115. package/dist/tools/getTokenPrice.d.ts +95 -9
  116. package/dist/tools/getTokenPrice.d.ts.map +1 -1
  117. package/dist/tools/getTokenPrice.js +95 -56
  118. package/dist/tools/getTokenPrice.js.map +1 -1
  119. package/dist/tools/getWalletBalance.d.ts +40 -11
  120. package/dist/tools/getWalletBalance.d.ts.map +1 -1
  121. package/dist/tools/getWalletBalance.js +64 -47
  122. package/dist/tools/getWalletBalance.js.map +1 -1
  123. package/dist/tools/publishRiskScore.d.ts +12 -10
  124. package/dist/tools/publishRiskScore.d.ts.map +1 -1
  125. package/dist/tools/publishRiskScore.js +33 -19
  126. package/dist/tools/publishRiskScore.js.map +1 -1
  127. package/dist/tools/queryRiskRegistry.d.ts +3 -3
  128. package/dist/tools/safehandsPreflightCheck.d.ts +78 -0
  129. package/dist/tools/safehandsPreflightCheck.d.ts.map +1 -0
  130. package/dist/tools/safehandsPreflightCheck.js +48 -0
  131. package/dist/tools/safehandsPreflightCheck.js.map +1 -0
  132. package/dist/tools/safehandsRiskReport.d.ts +82 -0
  133. package/dist/tools/safehandsRiskReport.d.ts.map +1 -0
  134. package/dist/tools/safehandsRiskReport.js +29 -0
  135. package/dist/tools/safehandsRiskReport.js.map +1 -0
  136. package/dist/tools/safehandsSafeExecute.d.ts +21 -0
  137. package/dist/tools/safehandsSafeExecute.d.ts.map +1 -0
  138. package/dist/tools/safehandsSafeExecute.js +76 -0
  139. package/dist/tools/safehandsSafeExecute.js.map +1 -0
  140. package/dist/tools/safehandsWalletHealth.d.ts +15 -0
  141. package/dist/tools/safehandsWalletHealth.d.ts.map +1 -0
  142. package/dist/tools/safehandsWalletHealth.js +104 -0
  143. package/dist/tools/safehandsWalletHealth.js.map +1 -0
  144. package/dist/tools/safehandsX402Preflight.d.ts +27 -0
  145. package/dist/tools/safehandsX402Preflight.d.ts.map +1 -0
  146. package/dist/tools/safehandsX402Preflight.js +66 -0
  147. package/dist/tools/safehandsX402Preflight.js.map +1 -0
  148. package/dist/tools/sendPayment.d.ts +13 -35
  149. package/dist/tools/sendPayment.d.ts.map +1 -1
  150. package/dist/tools/sendPayment.js +53 -47
  151. package/dist/tools/sendPayment.js.map +1 -1
  152. package/dist/tools/simulateTransaction.d.ts +4 -4
  153. package/dist/tools/tokenRegistryStatus.d.ts +27 -0
  154. package/dist/tools/tokenRegistryStatus.d.ts.map +1 -0
  155. package/dist/tools/tokenRegistryStatus.js +97 -0
  156. package/dist/tools/tokenRegistryStatus.js.map +1 -0
  157. package/dist/tools/x402PayAndFetch.d.ts +40 -16
  158. package/dist/tools/x402PayAndFetch.d.ts.map +1 -1
  159. package/dist/tools/x402PayAndFetch.js +115 -47
  160. package/dist/tools/x402PayAndFetch.js.map +1 -1
  161. package/dist/x402Server.js +149 -115
  162. package/dist/x402Server.js.map +1 -1
  163. package/examples/pharos-skill-engine/SKILL.safehands.md +85 -0
  164. package/examples/pharos-skill-engine/assets/safehands/example-actions.json +49 -0
  165. package/examples/pharos-skill-engine/assets/safehands/policy-defaults.json +11 -0
  166. package/examples/pharos-skill-engine/references/safehands.md +345 -0
  167. package/examples/scenario-hack.ts +38 -0
  168. package/package.json +19 -5
  169. package/skill/SKILL.md +127 -0
  170. package/skill/assets/safehands/example-actions.json +49 -0
  171. package/skill/assets/safehands/policy-defaults.json +11 -0
  172. package/skill/references/safehands.md +345 -0
  173. package/.agents/skill/safehands/SKILL.md +0 -200
  174. package/.agents/skill/safehands/assets/networks.json +0 -24
  175. package/.agents/skill/safehands/assets/tokens.json +0 -60
@@ -0,0 +1,90 @@
1
+ // ─── SignerProvider ─────────────────────────────────────────────────────
2
+ // Abstraction layer for signing. Tools never read process.env.PRIVATE_KEY
3
+ // directly — they request a signer through this provider.
4
+ // ────────────────────────────────────────────────────────────────────────
5
+ import { privateKeyToAccount } from "viem/accounts";
6
+ import { walletStore, deobfuscateKey } from "../wallet/index.js";
7
+ export function isSignerFailure(r) {
8
+ return "error" in r;
9
+ }
10
+ function normalizePrivateKey(pk) {
11
+ return (pk.startsWith("0x") ? pk : `0x${pk}`);
12
+ }
13
+ function accountFromEnvKey(envName, mode) {
14
+ const pk = process.env[envName];
15
+ if (!pk)
16
+ return null;
17
+ try {
18
+ const account = privateKeyToAccount(normalizePrivateKey(pk));
19
+ return { account, address: account.address, mode };
20
+ }
21
+ catch {
22
+ return {
23
+ error: {
24
+ code: "INVALID_PRIVATE_KEY",
25
+ message: `${envName} is not a valid private key.`,
26
+ },
27
+ };
28
+ }
29
+ }
30
+ async function accountFromManagedWallet(agentId) {
31
+ const walletMode = process.env.WALLET_MODE || "none";
32
+ if (walletMode !== "managed-testnet" || !agentId)
33
+ return null;
34
+ if (process.env.WALLET_STORE_PATH && !process.env.WALLET_ENCRYPTION_KEY) {
35
+ return {
36
+ error: {
37
+ code: "WALLET_ENCRYPTION_KEY_REQUIRED",
38
+ message: "WALLET_ENCRYPTION_KEY is required when WALLET_STORE_PATH is used for persistent managed-testnet wallets.",
39
+ },
40
+ };
41
+ }
42
+ const stored = await walletStore.get(agentId);
43
+ if (!stored)
44
+ return null;
45
+ const encryptionKey = process.env.WALLET_ENCRYPTION_KEY || "safehands-testnet-default-key";
46
+ try {
47
+ const privateKey = deobfuscateKey(stored.encryptedKey, encryptionKey);
48
+ const account = privateKeyToAccount(normalizePrivateKey(privateKey));
49
+ return { account, address: account.address, mode: "managed-testnet" };
50
+ }
51
+ catch {
52
+ return {
53
+ error: {
54
+ code: "NO_SIGNER_AVAILABLE",
55
+ message: "Managed wallet key could not be decrypted/deobfuscated for signing.",
56
+ },
57
+ };
58
+ }
59
+ }
60
+ /**
61
+ * Get a signer for write/payment operations.
62
+ * Priority for x402: managed wallet > X402_SIGNER_PRIVATE_KEY > PRIVATE_KEY fallback.
63
+ * Priority for writes: managed wallet > PRIVATE_KEY fallback.
64
+ */
65
+ export async function getSigner(agentId, options = {}) {
66
+ const purpose = options.purpose || "write";
67
+ const managed = await accountFromManagedWallet(agentId);
68
+ if (managed)
69
+ return managed;
70
+ if (purpose === "x402") {
71
+ const x402Env = accountFromEnvKey("X402_SIGNER_PRIVATE_KEY", "x402-env");
72
+ if (x402Env)
73
+ return x402Env;
74
+ }
75
+ const walletMode = process.env.WALLET_MODE || "none";
76
+ if (walletMode === "env" || process.env.PRIVATE_KEY) {
77
+ const env = accountFromEnvKey("PRIVATE_KEY", "env");
78
+ if (env)
79
+ return env;
80
+ }
81
+ return {
82
+ error: {
83
+ code: "NO_SIGNER_AVAILABLE",
84
+ message: purpose === "x402"
85
+ ? "No x402 signer available. Use WALLET_MODE=managed-testnet with agentId, X402_SIGNER_PRIVATE_KEY, or PRIVATE_KEY as testnet developer fallback."
86
+ : "No signer available. Use WALLET_MODE=managed-testnet with agentId, or WALLET_MODE=env with PRIVATE_KEY for testnet developer mode.",
87
+ },
88
+ };
89
+ }
90
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/lib/signer/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=testDodoLive.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testDodoLive.d.ts","sourceRoot":"","sources":["../../src/lib/testDodoLive.ts"],"names":[],"mappings":""}
@@ -0,0 +1,105 @@
1
+ // ─── Live DODO/FaroSwap Verification ───────────────────────────────────
2
+ // Tests DODO API reachability if DODO_API_KEY is present.
3
+ // Skips cleanly if API key is missing.
4
+ // ────────────────────────────────────────────────────────────────────────
5
+ import { DODO_API_BASE, DODO_API_KEY } from "./constants.js";
6
+ import { getDodoRoute } from "./dodoApi.js";
7
+ const results = [];
8
+ async function main() {
9
+ console.log("═══════════════════════════════════════════════════════════");
10
+ console.log(" SafeHands — DODO/FaroSwap Live Verification");
11
+ console.log("═══════════════════════════════════════════════════════════");
12
+ console.log(` DODO API Base: ${DODO_API_BASE}`);
13
+ const maskedKey = DODO_API_KEY ? `pub_****${DODO_API_KEY.slice(-4)}` : "NOT SET";
14
+ console.log(` API Key: ${maskedKey}`);
15
+ console.log("");
16
+ if (!DODO_API_KEY) {
17
+ console.log(" ⏭️ SKIPPED_MISSING_DODO_API_KEY — set DODO_API_KEY to enable live DODO checks.");
18
+ results.push({
19
+ name: "dodo_api_route_check",
20
+ status: "SKIPPED_MISSING_DODO_API_KEY",
21
+ detail: "DODO_API_KEY env var not set. This is expected for CI/demo environments.",
22
+ });
23
+ }
24
+ else {
25
+ // Try a read-only route quote
26
+ try {
27
+ const quote = await getDodoRoute({
28
+ fromToken: "PHRS",
29
+ toToken: "USDC",
30
+ amountHuman: "0.000001",
31
+ walletAddress: "0x0000000000000000000000000000000000000001",
32
+ });
33
+ if (quote.routeAvailable && parseFloat(quote.amountOut) > 0) {
34
+ results.push({
35
+ name: "dodo_api_route_check",
36
+ status: "PASS",
37
+ detail: `Route found: 0.000001 PHRS → ${quote.amountOut} USDC`,
38
+ });
39
+ }
40
+ else {
41
+ results.push({
42
+ name: "dodo_api_route_check",
43
+ status: "NO_ROUTE_AVAILABLE",
44
+ detail: `API reachable. No route available (may be no liquidity).`,
45
+ });
46
+ }
47
+ }
48
+ catch (err) {
49
+ const msg = err.message || String(err);
50
+ if (msg.includes("DODO_API_AUTH_REQUIRED")) {
51
+ results.push({
52
+ name: "dodo_api_route_check",
53
+ status: "DODO_API_AUTH_REQUIRED",
54
+ detail: msg,
55
+ });
56
+ }
57
+ else if (msg.includes("DODO_API_UNAVAILABLE") || msg.includes("DODO_API_RATE_LIMITED") || msg.includes("fetch failed") || msg.includes("network timeout")) {
58
+ results.push({
59
+ name: "dodo_api_route_check",
60
+ status: "DODO_API_UNAVAILABLE",
61
+ detail: `Network/timeout/rate-limited: ${msg}`,
62
+ });
63
+ }
64
+ else {
65
+ results.push({
66
+ name: "dodo_api_route_check",
67
+ status: "FAIL",
68
+ detail: msg,
69
+ });
70
+ }
71
+ }
72
+ }
73
+ // Router address verification status
74
+ results.push({
75
+ name: "dodo_approve_address_status",
76
+ status: "PASS",
77
+ detail: "0x4Cf317b8918FbE8A890c01eDAb7d548555Ac2cE9 — PROJECT_CONFIGURED (not verified by official FaroSwap/DODO docs)",
78
+ });
79
+ results.push({
80
+ name: "dodo_route_proxy_status",
81
+ status: "PASS",
82
+ detail: "0x819829e5CF6e19F9fED92F6b4CC1edF45a2cC4A2 — PROJECT_CONFIGURED (not verified by official FaroSwap/DODO docs)",
83
+ });
84
+ // Print results
85
+ console.log(`\n${"#".padStart(2)} ${"Status".padEnd(32)} ${"Check".padEnd(34)} Detail`);
86
+ console.log("─".repeat(110));
87
+ for (const [i, r] of results.entries()) {
88
+ const icon = r.status === "PASS" ? "✅" : r.status.startsWith("SKIPPED") ? "⏭️" : "❌";
89
+ console.log(`${String(i + 1).padStart(2)} ${icon} ${r.status.padEnd(29)} ${r.name.padEnd(34)} ${r.detail.slice(0, 80)}`);
90
+ }
91
+ const failed = results.filter(r => r.status === "FAIL");
92
+ console.log("─".repeat(110));
93
+ console.log(`${results.filter(r => r.status === "PASS").length}/${results.length} checks passed, ${results.filter(r => r.status.startsWith("SKIPPED")).length} skipped.`);
94
+ if (failed.length > 0) {
95
+ console.error("\nFailed:");
96
+ for (const f of failed)
97
+ console.error(` - ${f.name}: ${f.detail}`);
98
+ process.exit(1);
99
+ }
100
+ }
101
+ main().catch((err) => {
102
+ console.error("❌ Unexpected error:", err);
103
+ process.exit(1);
104
+ });
105
+ //# sourceMappingURL=testDodoLive.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testDodoLive.js","sourceRoot":"","sources":["../../src/lib/testDodoLive.ts"],"names":[],"mappings":"AAAA,0EAA0E;AAC1E,0DAA0D;AAC1D,uCAAuC;AACvC,2EAA2E;AAE3E,OAAO,EAAE,aAAa,EAAE,YAAY,EAA6D,MAAM,gBAAgB,CAAC;AACxH,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAQ5C,MAAM,OAAO,GAAsB,EAAE,CAAC;AAEtC,KAAK,UAAU,IAAI;IACjB,OAAO,CAAC,GAAG,CAAC,6DAA6D,CAAC,CAAC;IAC3E,OAAO,CAAC,GAAG,CAAC,+CAA+C,CAAC,CAAC;IAC7D,OAAO,CAAC,GAAG,CAAC,6DAA6D,CAAC,CAAC;IAC3E,OAAO,CAAC,GAAG,CAAC,oBAAoB,aAAa,EAAE,CAAC,CAAC;IACjD,MAAM,SAAS,GAAG,YAAY,CAAC,CAAC,CAAC,WAAW,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;IACjF,OAAO,CAAC,GAAG,CAAC,oBAAoB,SAAS,EAAE,CAAC,CAAC;IAC7C,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAEhB,IAAI,CAAC,YAAY,EAAE,CAAC;QAClB,OAAO,CAAC,GAAG,CAAC,mFAAmF,CAAC,CAAC;QACjG,OAAO,CAAC,IAAI,CAAC;YACX,IAAI,EAAE,sBAAsB;YAC5B,MAAM,EAAE,8BAA8B;YACtC,MAAM,EAAE,0EAA0E;SACnF,CAAC,CAAC;IACL,CAAC;SAAM,CAAC;QACN,8BAA8B;QAC9B,IAAI,CAAC;YACH,MAAM,KAAK,GAAG,MAAM,YAAY,CAAC;gBAC/B,SAAS,EAAE,MAAM;gBACjB,OAAO,EAAE,MAAM;gBACf,WAAW,EAAE,UAAU;gBACvB,aAAa,EAAE,4CAA4C;aAC5D,CAAC,CAAC;YAEH,IAAI,KAAK,CAAC,cAAc,IAAI,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC;gBAC5D,OAAO,CAAC,IAAI,CAAC;oBACX,IAAI,EAAE,sBAAsB;oBAC5B,MAAM,EAAE,MAAM;oBACd,MAAM,EAAE,gCAAgC,KAAK,CAAC,SAAS,OAAO;iBAC/D,CAAC,CAAC;YACL,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,IAAI,CAAC;oBACX,IAAI,EAAE,sBAAsB;oBAC5B,MAAM,EAAE,oBAAoB;oBAC5B,MAAM,EAAE,0DAA0D;iBACnE,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAAC,OAAO,GAAQ,EAAE,CAAC;YAClB,MAAM,GAAG,GAAG,GAAG,CAAC,OAAO,IAAI,MAAM,CAAC,GAAG,CAAC,CAAC;YACvC,IAAI,GAAG,CAAC,QAAQ,CAAC,wBAAwB,CAAC,EAAE,CAAC;gBAC3C,OAAO,CAAC,IAAI,CAAC;oBACX,IAAI,EAAE,sBAAsB;oBAC5B,MAAM,EAAE,wBAAwB;oBAChC,MAAM,EAAE,GAAG;iBACZ,CAAC,CAAC;YACL,CAAC;iBAAM,IAAI,GAAG,CAAC,QAAQ,CAAC,sBAAsB,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,uBAAuB,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,iBAAiB,CAAC,EAAE,CAAC;gBAC5J,OAAO,CAAC,IAAI,CAAC;oBACX,IAAI,EAAE,sBAAsB;oBAC5B,MAAM,EAAE,sBAAsB;oBAC9B,MAAM,EAAE,iCAAiC,GAAG,EAAE;iBAC/C,CAAC,CAAC;YACL,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,IAAI,CAAC;oBACX,IAAI,EAAE,sBAAsB;oBAC5B,MAAM,EAAE,MAAM;oBACd,MAAM,EAAE,GAAG;iBACZ,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;IAED,qCAAqC;IACrC,OAAO,CAAC,IAAI,CAAC;QACX,IAAI,EAAE,6BAA6B;QACnC,MAAM,EAAE,MAAM;QACd,MAAM,EAAE,+GAA+G;KACxH,CAAC,CAAC;IACH,OAAO,CAAC,IAAI,CAAC;QACX,IAAI,EAAE,yBAAyB;QAC/B,MAAM,EAAE,MAAM;QACd,MAAM,EAAE,+GAA+G;KACxH,CAAC,CAAC;IAEH,gBAAgB;IAChB,OAAO,CAAC,GAAG,CAAC,KAAK,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC;IACxF,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;IAC7B,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC;QACvC,MAAM,IAAI,GAAG,CAAC,CAAC,MAAM,KAAK,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC;QACrF,OAAO,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,IAAI,IAAI,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,EAAE,CAAC,CAAC;IAC3H,CAAC;IAED,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,MAAM,CAAC,CAAC;IACxD,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;IAC7B,OAAO,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,MAAM,CAAC,CAAC,MAAM,IAAI,OAAO,CAAC,MAAM,mBAAmB,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,WAAW,CAAC,CAAC;IAE1K,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACtB,OAAO,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QAC3B,KAAK,MAAM,CAAC,IAAI,MAAM;YAAE,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,KAAK,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC;QACpE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC;AAED,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;IACnB,OAAO,CAAC,KAAK,CAAC,qBAAqB,EAAE,GAAG,CAAC,CAAC;IAC1C,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=testLiveSafehands.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testLiveSafehands.d.ts","sourceRoot":"","sources":["../../src/lib/testLiveSafehands.ts"],"names":[],"mappings":""}
@@ -0,0 +1,93 @@
1
+ // ─── Live SafeHands CLI Verification ───────────────────────────────────
2
+ // Runs real CLI safety checks without broadcasting transactions.
3
+ // All checks are read-only or dry-run preflight checks.
4
+ // ────────────────────────────────────────────────────────────────────────
5
+ import { spawnSync } from "node:child_process";
6
+ import { join } from "node:path";
7
+ import { existsSync } from "node:fs";
8
+ const results = [];
9
+ function runCli(args) {
10
+ const entry = join(process.cwd(), "dist", "index.js");
11
+ if (!existsSync(entry)) {
12
+ console.error("❌ dist/index.js not found. Run: npm run build");
13
+ process.exit(1);
14
+ }
15
+ const result = spawnSync(process.execPath, [entry, ...args], {
16
+ cwd: process.cwd(),
17
+ env: { ...process.env, WALLET_MODE: "none", WRITE_TOOLS_ENABLED: "false", PRIVATE_KEY: "" },
18
+ encoding: "utf8",
19
+ timeout: 30_000,
20
+ });
21
+ return { exitCode: result.status ?? 1, stdout: result.stdout || "", stderr: result.stderr || "" };
22
+ }
23
+ function parseJson(stdout) {
24
+ try {
25
+ return JSON.parse(stdout.trim());
26
+ }
27
+ catch {
28
+ return null;
29
+ }
30
+ }
31
+ function check(name, args, expected, validator) {
32
+ const { stdout } = runCli(args);
33
+ const json = parseJson(stdout);
34
+ const passed = json !== null && validator(json);
35
+ const actual = json !== null
36
+ ? (json.success ? `success: ${JSON.stringify(json.data).slice(0, 120)}` : `error: ${json.error?.code}`)
37
+ : `invalid JSON: ${stdout.slice(0, 100)}`;
38
+ results.push({ name, status: passed ? "PASS" : "FAIL", expected, actual });
39
+ }
40
+ console.log("═══════════════════════════════════════════════════════════");
41
+ console.log(" SafeHands — Live CLI Verification (read-only)");
42
+ console.log("═══════════════════════════════════════════════════════════");
43
+ console.log("");
44
+ // 1. Wallet health — no wallet configured
45
+ check("wallet_health_no_wallet", ["skill", "safehands_wallet_health", "--input-json", "{}"], "valid JSON with status NOT_READY or DEGRADED", (json) => json.success && ["NOT_READY", "DEGRADED"].includes(json.data?.status));
46
+ // 2. Token registry — Pharos Skill Engine USDC (DOCS_VERIFIED_FROM_PHAROS_SKILL_ENGINE)
47
+ check("token_registry_skill_engine_usdc", ["skill", "token_registry_status", "--input-json", JSON.stringify({ tokenAddress: "0xE0BE08c77f415F577A1B3A9aD7a1Df1479564ec8" })], "SKILL_ENGINE_CANONICAL_TOKEN with DOCS_VERIFIED_FROM_PHAROS_SKILL_ENGINE", (json) => json.success && json.data?.status === "SKILL_ENGINE_CANONICAL_TOKEN" && json.data?.verificationStatus === "DOCS_VERIFIED_FROM_PHAROS_SKILL_ENGINE");
48
+ // 3. Token registry — Circle USDC (ALTERNATE_SOURCE_TOKEN)
49
+ check("token_registry_circle_usdc", ["skill", "token_registry_status", "--input-json", JSON.stringify({ tokenAddress: "0xcfC8330f4BCAB529c625D12781b1C19466A9Fc8B" })], "ALTERNATE_SOURCE_TOKEN with CIRCLE_REFERENCED_USDC", (json) => json.success && json.data?.status === "ALTERNATE_SOURCE_TOKEN" && json.data?.verificationStatus === "CIRCLE_REFERENCED_USDC");
50
+ // 4. Token registry — USDT (DOCS_VERIFIED)
51
+ check("token_registry_usdt_docs_verified", ["skill", "token_registry_status", "--input-json", JSON.stringify({ tokenAddress: "0xE7E84B8B4f39C507499c40B4ac199B050e2882d5" })], "CANONICAL_TESTNET_TOKEN with DOCS_VERIFIED", (json) => json.success && json.data?.status === "CANONICAL_TESTNET_TOKEN" && json.data?.verificationStatus === "DOCS_VERIFIED");
52
+ // 5. Preflight — unlimited approval → BLOCK
53
+ check("preflight_block_unlimited_approval", ["skill", "safehands_preflight_check", "--input-json", JSON.stringify({
54
+ actionType: "approve_token",
55
+ chainId: 688689,
56
+ tokenAddress: "0xE0BE08c77f415F577A1B3A9aD7a1Df1479564ec8",
57
+ targetAddress: "0x0000000000000000000000000000000000000001",
58
+ amount: "unlimited",
59
+ approvalAmount: "max",
60
+ })], "decision=BLOCK", (json) => json.success && json.data?.decision === "BLOCK");
61
+ // 6. Preflight — mainnet action → BLOCK
62
+ check("preflight_block_mainnet", ["skill", "safehands_preflight_check", "--input-json", JSON.stringify({
63
+ actionType: "send_payment",
64
+ chainId: 1,
65
+ isMainnet: true,
66
+ targetAddress: "0x0000000000000000000000000000000000000001",
67
+ amount: "0.001",
68
+ })], "decision=BLOCK", (json) => json.success && json.data?.decision === "BLOCK");
69
+ // 7. Preflight — safe testnet action → ALLOW or WARN (not BLOCK)
70
+ check("preflight_allow_testnet", ["skill", "safehands_preflight_check", "--input-json", JSON.stringify({
71
+ actionType: "send_payment",
72
+ chainId: 688689,
73
+ isMainnet: false,
74
+ amount: "0.001",
75
+ recipient: "0x000000000000000000000000000000000000dEaD",
76
+ })], "decision=ALLOW or WARN (not BLOCK)", (json) => json.success && json.data?.decision !== "BLOCK");
77
+ // Print results
78
+ console.log(`\n${"#".padStart(2)} ${"Status".padEnd(8)} ${"Check".padEnd(42)} Expected`);
79
+ console.log("─".repeat(100));
80
+ for (const [i, r] of results.entries()) {
81
+ const icon = r.status === "PASS" ? "✅" : "❌";
82
+ console.log(`${String(i + 1).padStart(2)} ${icon} ${r.status.padEnd(5)} ${r.name.padEnd(42)} ${r.expected}`);
83
+ }
84
+ const failed = results.filter(r => r.status === "FAIL");
85
+ console.log("─".repeat(100));
86
+ console.log(`${results.length - failed.length}/${results.length} live CLI checks passed.`);
87
+ if (failed.length > 0) {
88
+ console.error("\nFailed checks:");
89
+ for (const f of failed)
90
+ console.error(` - ${f.name}: got ${f.actual}`);
91
+ process.exit(1);
92
+ }
93
+ //# sourceMappingURL=testLiveSafehands.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testLiveSafehands.js","sourceRoot":"","sources":["../../src/lib/testLiveSafehands.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=testRpcLive.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testRpcLive.d.ts","sourceRoot":"","sources":["../../src/lib/testRpcLive.ts"],"names":[],"mappings":""}
@@ -0,0 +1,89 @@
1
+ // ─── Live RPC Verification ─────────────────────────────────────────────
2
+ // Real read-only test against Pharos Atlantic Testnet.
3
+ // Does NOT require a private key. Wallet balance check requires WALLET_ADDRESS.
4
+ // ────────────────────────────────────────────────────────────────────────
5
+ import { publicClient } from "./pharosClient.js";
6
+ import { CHAIN_ID, RPC_URL, PHAROS_ENVIRONMENT } from "./constants.js";
7
+ import { formatEther, isAddress } from "viem";
8
+ async function main() {
9
+ const result = {
10
+ rpcReachable: false,
11
+ chainId: null,
12
+ chainIdMatch: false,
13
+ latestBlock: null,
14
+ walletBalance: null,
15
+ walletBalanceCheck: "SKIPPED_NO_WALLET_ADDRESS",
16
+ status: "FAIL",
17
+ rpcUrl: RPC_URL,
18
+ environment: PHAROS_ENVIRONMENT,
19
+ expectedChainId: CHAIN_ID,
20
+ errors: [],
21
+ };
22
+ console.log("═══════════════════════════════════════════════════════════");
23
+ console.log(" SafeHands — Live RPC Verification (read-only)");
24
+ console.log("═══════════════════════════════════════════════════════════");
25
+ console.log(` RPC URL: ${RPC_URL}`);
26
+ console.log(` Environment: ${PHAROS_ENVIRONMENT}`);
27
+ console.log(` Expected ID: ${CHAIN_ID}`);
28
+ console.log("");
29
+ // 1. Chain ID
30
+ try {
31
+ const chainId = await publicClient.getChainId();
32
+ result.rpcReachable = true;
33
+ result.chainId = chainId;
34
+ result.chainIdMatch = chainId === CHAIN_ID;
35
+ console.log(` RPC reachable: ✅ yes`);
36
+ console.log(` Chain ID: ${chainId} ${result.chainIdMatch ? "✅ match" : "❌ MISMATCH"}`);
37
+ }
38
+ catch (err) {
39
+ result.errors.push(`RPC connect: ${err.message || String(err)}`);
40
+ console.log(` RPC reachable: ❌ no`);
41
+ console.log(` Error: ${err.message || String(err)}`);
42
+ result.status = "SKIPPED_NETWORK";
43
+ console.log(`\n Status: ${result.status}`);
44
+ console.log(JSON.stringify(result, (_k, v) => typeof v === "bigint" ? v.toString() : v, 2));
45
+ process.exit(0);
46
+ }
47
+ // 2. Latest block
48
+ try {
49
+ const blockNumber = await publicClient.getBlockNumber();
50
+ result.latestBlock = blockNumber;
51
+ console.log(` Latest block: ${blockNumber}`);
52
+ }
53
+ catch (err) {
54
+ result.errors.push(`Block number: ${err.message || String(err)}`);
55
+ console.log(` Latest block: ❌ failed (${err.message})`);
56
+ }
57
+ // 3. Wallet balance
58
+ const walletAddress = process.env.WALLET_ADDRESS;
59
+ if (walletAddress && isAddress(walletAddress)) {
60
+ try {
61
+ const balance = await publicClient.getBalance({ address: walletAddress });
62
+ result.walletBalance = formatEther(balance);
63
+ result.walletBalanceCheck = "PASS";
64
+ console.log(` Wallet: ${walletAddress}`);
65
+ console.log(` Balance: ${result.walletBalance} PHRS ✅`);
66
+ }
67
+ catch (err) {
68
+ result.walletBalanceCheck = "FAIL";
69
+ result.errors.push(`Balance: ${err.message || String(err)}`);
70
+ console.log(` Balance check: ❌ failed (${err.message})`);
71
+ }
72
+ }
73
+ else {
74
+ console.log(` Wallet check: SKIPPED_NO_WALLET_ADDRESS`);
75
+ }
76
+ // Final status
77
+ result.status = result.rpcReachable && result.chainIdMatch && result.latestBlock !== null ? "PASS" : "FAIL";
78
+ console.log("");
79
+ console.log(` Status: ${result.status}`);
80
+ console.log("═══════════════════════════════════════════════════════════");
81
+ console.log(JSON.stringify(result, (_k, v) => typeof v === "bigint" ? v.toString() : v, 2));
82
+ if (result.status === "FAIL")
83
+ process.exit(1);
84
+ }
85
+ main().catch((err) => {
86
+ console.error("❌ Unexpected error:", err);
87
+ process.exit(1);
88
+ });
89
+ //# sourceMappingURL=testRpcLive.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testRpcLive.js","sourceRoot":"","sources":["../../src/lib/testRpcLive.ts"],"names":[],"mappings":"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"}