-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathinvoke.ts
executable file
·115 lines (104 loc) · 3.82 KB
/
invoke.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
#!/usr/bin/env ts-node-script
import { ArgumentParser } from 'argparse';
import {
Contract,
Keypair,
rpc,
TransactionBuilder,
scValToNative,
xdr,
} from '@stellar/stellar-sdk';
async function main() {
const parser = new ArgumentParser({ description: 'Invoke a contract function' })
const subparsers = parser.add_subparsers();
parser.add_argument('--id', { dest: 'contractId', required: true, help: 'Contract ID' });
parser.add_argument('--rpc-url', { dest: 'rpcUrl', required: true, help: 'RPC URL' });
parser.add_argument('--source', { dest: 'source', required: true, help: 'Secret key' });
parser.add_argument('--network-passphrase', { dest: 'networkPassphrase', required: true, help: 'Network passphrase' });
const functionParamParser = subparsers.add_parser('function', { help: 'Function' });
functionParamParser.add_argument('--name', { dest: 'functionName', help: 'Function Name' });
functionParamParser.add_argument('--params', { dest: 'functionParams', help: 'Function Params, comma separated' })
const {
contractId,
rpcUrl,
functionParams,
networkPassphrase,
source,
functionName,
} = parser.parse_args() as Record<string, string>;
const keypair = Keypair.fromSecret(source);
const account = keypair.publicKey();
// @ts-ignore contract client only available in stellar-sdk ≥12
const { contract } = await import('@stellar/stellar-sdk');
if (contract) {
const client = await contract.Client.from({
allowHttp: true,
rpcUrl,
networkPassphrase,
contractId,
publicKey: account,
...contract.basicNodeSigner(keypair, networkPassphrase),
});
const args: Record<string, any> = {};
if (functionParams) {
functionParams.split(",").forEach((p) => {
const [name, value] = p.split(":");
args[name] = value;
});
}
// @ts-ignore client[functionName] is defined dynamically
const tx = await client[functionName](args);
const { result } = await tx.signAndSend({ force: true });
console.log(JSON.stringify(result));
return;
} else {
const server = new rpc.Server(rpcUrl, { allowHttp: true });
const sourceAccount = await server.getAccount(account);
const contract = new Contract(contractId);
// Some hacky param-parsing as csv. Generated Typescript bindings would be better.
const params: xdr.ScVal[] = functionParams
? functionParams.split(",").map((p) => xdr.ScVal.scvSymbol(p.split(':')[1])) : [];
const originalTxn = new TransactionBuilder(sourceAccount, {
fee: "100",
networkPassphrase,
})
.addOperation(contract.call(functionName, ...params))
.setTimeout(30)
.build();
const txn = await server.prepareTransaction(originalTxn);
txn.sign(keypair);
const send = await server.sendTransaction(txn);
if (send.errorResult) {
throw new Error(`Transaction failed: ${JSON.stringify(send)}`);
}
let response = await server.getTransaction(send.hash);
for (let i = 0; i < 50; i++) {
switch (response.status) {
case "NOT_FOUND": {
// retry
await new Promise(resolve => setTimeout(resolve, 100));
response = await server.getTransaction(send.hash);
break;
}
case "SUCCESS": {
if (!response.returnValue) {
throw new Error(`No invoke host fn return value provided: ${JSON.stringify(response)}`);
}
const parsed = scValToNative(response.returnValue);
console.log(JSON.stringify(parsed));
return;
}
case "FAILED": {
throw new Error(`Transaction failed: ${JSON.stringify(response)}`);
}
default:
throw new Error(`Unknown transaction status: ${response.status}`);
}
}
}
throw new Error("Transaction timed out");
}
main().catch(err => {
console.error(JSON.stringify(err));
throw err;
});