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.
- package/.env.example +26 -0
- package/README.md +311 -350
- package/contracts/RiskRegistry.json +75 -1
- package/contracts/RiskRegistry.sol +29 -1
- package/dist/cli.d.ts +6 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +91 -0
- package/dist/cli.js.map +1 -0
- package/dist/demo.d.ts +2 -0
- package/dist/demo.d.ts.map +1 -0
- package/dist/demo.js +172 -0
- package/dist/demo.js.map +1 -0
- package/dist/index.js +181 -169
- package/dist/index.js.map +1 -1
- package/dist/init.d.ts +2 -0
- package/dist/init.d.ts.map +1 -0
- package/dist/init.js +66 -0
- package/dist/init.js.map +1 -0
- package/dist/lib/constants.d.ts +122 -7
- package/dist/lib/constants.d.ts.map +1 -1
- package/dist/lib/constants.js +139 -13
- package/dist/lib/constants.js.map +1 -1
- package/dist/lib/dodoApi.d.ts +14 -0
- package/dist/lib/dodoApi.d.ts.map +1 -1
- package/dist/lib/dodoApi.js +78 -22
- package/dist/lib/dodoApi.js.map +1 -1
- package/dist/lib/http.d.ts +15 -0
- package/dist/lib/http.d.ts.map +1 -0
- package/dist/lib/http.js +119 -0
- package/dist/lib/http.js.map +1 -0
- package/dist/lib/pharosClient.d.ts +4 -3
- package/dist/lib/pharosClient.d.ts.map +1 -1
- package/dist/lib/pharosClient.js +8 -5
- package/dist/lib/pharosClient.js.map +1 -1
- package/dist/lib/policy/actionPolicyEngine.d.ts +54 -0
- package/dist/lib/policy/actionPolicyEngine.d.ts.map +1 -0
- package/dist/lib/policy/actionPolicyEngine.js +213 -0
- package/dist/lib/policy/actionPolicyEngine.js.map +1 -0
- package/dist/lib/signer/index.d.ts +25 -0
- package/dist/lib/signer/index.d.ts.map +1 -0
- package/dist/lib/signer/index.js +90 -0
- package/dist/lib/signer/index.js.map +1 -0
- package/dist/lib/testDodoLive.d.ts +2 -0
- package/dist/lib/testDodoLive.d.ts.map +1 -0
- package/dist/lib/testDodoLive.js +105 -0
- package/dist/lib/testDodoLive.js.map +1 -0
- package/dist/lib/testLiveSafehands.d.ts +2 -0
- package/dist/lib/testLiveSafehands.d.ts.map +1 -0
- package/dist/lib/testLiveSafehands.js +93 -0
- package/dist/lib/testLiveSafehands.js.map +1 -0
- package/dist/lib/testRpcLive.d.ts +2 -0
- package/dist/lib/testRpcLive.d.ts.map +1 -0
- package/dist/lib/testRpcLive.js +89 -0
- package/dist/lib/testRpcLive.js.map +1 -0
- package/dist/lib/testTools.js +363 -354
- package/dist/lib/testTools.js.map +1 -1
- package/dist/lib/testX402Live.d.ts +2 -0
- package/dist/lib/testX402Live.d.ts.map +1 -0
- package/dist/lib/testX402Live.js +160 -0
- package/dist/lib/testX402Live.js.map +1 -0
- package/dist/lib/toolResponse.d.ts +26 -0
- package/dist/lib/toolResponse.d.ts.map +1 -0
- package/dist/lib/toolResponse.js +54 -0
- package/dist/lib/toolResponse.js.map +1 -0
- package/dist/lib/wallet/index.d.ts +19 -0
- package/dist/lib/wallet/index.d.ts.map +1 -0
- package/dist/lib/wallet/index.js +71 -0
- package/dist/lib/wallet/index.js.map +1 -0
- package/dist/tools/approveToken.d.ts +19 -20
- package/dist/tools/approveToken.d.ts.map +1 -1
- package/dist/tools/approveToken.js +44 -21
- package/dist/tools/approveToken.js.map +1 -1
- package/dist/tools/assessRisk.d.ts +22 -9
- package/dist/tools/assessRisk.d.ts.map +1 -1
- package/dist/tools/assessRisk.js +32 -9
- package/dist/tools/assessRisk.js.map +1 -1
- package/dist/tools/checkAllowance.d.ts +6 -6
- package/dist/tools/checkTokenSecurity.d.ts +9 -16
- package/dist/tools/checkTokenSecurity.d.ts.map +1 -1
- package/dist/tools/checkTokenSecurity.js +17 -22
- package/dist/tools/checkTokenSecurity.js.map +1 -1
- package/dist/tools/createAgentWallet.d.ts +27 -0
- package/dist/tools/createAgentWallet.d.ts.map +1 -0
- package/dist/tools/createAgentWallet.js +60 -0
- package/dist/tools/createAgentWallet.js.map +1 -0
- package/dist/tools/estimateGas.d.ts +31 -21
- package/dist/tools/estimateGas.d.ts.map +1 -1
- package/dist/tools/estimateGas.js +91 -95
- package/dist/tools/estimateGas.js.map +1 -1
- package/dist/tools/executeSwap.d.ts +13 -29
- package/dist/tools/executeSwap.d.ts.map +1 -1
- package/dist/tools/executeSwap.js +68 -46
- package/dist/tools/executeSwap.js.map +1 -1
- package/dist/tools/explainRisk.d.ts +30 -0
- package/dist/tools/explainRisk.d.ts.map +1 -0
- package/dist/tools/explainRisk.js +33 -0
- package/dist/tools/explainRisk.js.map +1 -0
- package/dist/tools/getAgentWallet.d.ts +22 -0
- package/dist/tools/getAgentWallet.d.ts.map +1 -0
- package/dist/tools/getAgentWallet.js +28 -0
- package/dist/tools/getAgentWallet.js.map +1 -0
- package/dist/tools/getAgentWalletBalance.d.ts +12 -0
- package/dist/tools/getAgentWalletBalance.d.ts.map +1 -0
- package/dist/tools/getAgentWalletBalance.js +71 -0
- package/dist/tools/getAgentWalletBalance.js.map +1 -0
- package/dist/tools/getExecutionHistory.d.ts +4 -4
- package/dist/tools/getGasPrice.d.ts +26 -8
- package/dist/tools/getGasPrice.d.ts.map +1 -1
- package/dist/tools/getGasPrice.js +43 -35
- package/dist/tools/getGasPrice.js.map +1 -1
- package/dist/tools/getPoolInfo.d.ts +47 -59
- package/dist/tools/getPoolInfo.d.ts.map +1 -1
- package/dist/tools/getPoolInfo.js +96 -57
- package/dist/tools/getPoolInfo.js.map +1 -1
- package/dist/tools/getTokenPrice.d.ts +95 -9
- package/dist/tools/getTokenPrice.d.ts.map +1 -1
- package/dist/tools/getTokenPrice.js +95 -56
- package/dist/tools/getTokenPrice.js.map +1 -1
- package/dist/tools/getWalletBalance.d.ts +40 -11
- package/dist/tools/getWalletBalance.d.ts.map +1 -1
- package/dist/tools/getWalletBalance.js +64 -47
- package/dist/tools/getWalletBalance.js.map +1 -1
- package/dist/tools/publishRiskScore.d.ts +12 -10
- package/dist/tools/publishRiskScore.d.ts.map +1 -1
- package/dist/tools/publishRiskScore.js +33 -19
- package/dist/tools/publishRiskScore.js.map +1 -1
- package/dist/tools/queryRiskRegistry.d.ts +3 -3
- package/dist/tools/safehandsPreflightCheck.d.ts +78 -0
- package/dist/tools/safehandsPreflightCheck.d.ts.map +1 -0
- package/dist/tools/safehandsPreflightCheck.js +48 -0
- package/dist/tools/safehandsPreflightCheck.js.map +1 -0
- package/dist/tools/safehandsRiskReport.d.ts +82 -0
- package/dist/tools/safehandsRiskReport.d.ts.map +1 -0
- package/dist/tools/safehandsRiskReport.js +29 -0
- package/dist/tools/safehandsRiskReport.js.map +1 -0
- package/dist/tools/safehandsSafeExecute.d.ts +21 -0
- package/dist/tools/safehandsSafeExecute.d.ts.map +1 -0
- package/dist/tools/safehandsSafeExecute.js +76 -0
- package/dist/tools/safehandsSafeExecute.js.map +1 -0
- package/dist/tools/safehandsWalletHealth.d.ts +15 -0
- package/dist/tools/safehandsWalletHealth.d.ts.map +1 -0
- package/dist/tools/safehandsWalletHealth.js +104 -0
- package/dist/tools/safehandsWalletHealth.js.map +1 -0
- package/dist/tools/safehandsX402Preflight.d.ts +27 -0
- package/dist/tools/safehandsX402Preflight.d.ts.map +1 -0
- package/dist/tools/safehandsX402Preflight.js +66 -0
- package/dist/tools/safehandsX402Preflight.js.map +1 -0
- package/dist/tools/sendPayment.d.ts +13 -35
- package/dist/tools/sendPayment.d.ts.map +1 -1
- package/dist/tools/sendPayment.js +53 -47
- package/dist/tools/sendPayment.js.map +1 -1
- package/dist/tools/simulateTransaction.d.ts +4 -4
- package/dist/tools/tokenRegistryStatus.d.ts +27 -0
- package/dist/tools/tokenRegistryStatus.d.ts.map +1 -0
- package/dist/tools/tokenRegistryStatus.js +97 -0
- package/dist/tools/tokenRegistryStatus.js.map +1 -0
- package/dist/tools/x402PayAndFetch.d.ts +40 -16
- package/dist/tools/x402PayAndFetch.d.ts.map +1 -1
- package/dist/tools/x402PayAndFetch.js +115 -47
- package/dist/tools/x402PayAndFetch.js.map +1 -1
- package/dist/x402Server.js +149 -115
- package/dist/x402Server.js.map +1 -1
- package/examples/pharos-skill-engine/SKILL.safehands.md +85 -0
- package/examples/pharos-skill-engine/assets/safehands/example-actions.json +49 -0
- package/examples/pharos-skill-engine/assets/safehands/policy-defaults.json +11 -0
- package/examples/pharos-skill-engine/references/safehands.md +345 -0
- package/examples/scenario-hack.ts +38 -0
- package/package.json +19 -5
- package/skill/SKILL.md +127 -0
- package/skill/assets/safehands/example-actions.json +49 -0
- package/skill/assets/safehands/policy-defaults.json +11 -0
- package/skill/references/safehands.md +345 -0
- package/.agents/skill/safehands/SKILL.md +0 -200
- package/.agents/skill/safehands/assets/networks.json +0 -24
- 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 @@
|
|
|
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":"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"}
|
|
@@ -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 @@
|
|
|
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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"}
|