diff --git a/.cargo/config.toml b/.cargo/config.toml index 51ef227e4..3aa25973a 100644 --- a/.cargo/config.toml +++ b/.cargo/config.toml @@ -1,7 +1,7 @@ # paths = ["/path/to/override"] # path dependency overrides [alias] # command aliases -install_soroban = "install --version 21.4.1 --root ./target soroban-cli --debug" +install_stellar = "install --version 21.5.0 --root ./target stellar-cli --debug --force" # b = "build --target wasm32-unknown-unknown --release" # c = "check" # t = "test" diff --git a/.env b/.env index 6516b6c9c..efc8a0743 100644 --- a/.env +++ b/.env @@ -1,4 +1,4 @@ -SOROBAN_NETWORK_PASSPHRASE="Standalone Network ; February 2017" -SOROBAN_RPC_URL="http://localhost:8000/soroban/rpc" -SOROBAN_ACCOUNT="me" +STELLAR_NETWORK_PASSPHRASE="Standalone Network ; February 2017" +STELLAR_RPC_URL="http://localhost:8000/soroban/rpc" +STELLAR_ACCOUNT="me" FRIENDBOT_URL="http://localhost:8000/friendbot" diff --git a/src/contract/assembled_transaction.ts b/src/contract/assembled_transaction.ts index ffaa0f060..b4e909be3 100644 --- a/src/contract/assembled_transaction.ts +++ b/src/contract/assembled_transaction.ts @@ -2,6 +2,7 @@ /* eslint max-classes-per-file: 0 */ import { Account, + Address, BASE_FEE, Contract, Operation, @@ -636,9 +637,11 @@ export class AssembledTransaction { ); } - if (this.needsNonInvokerSigningBy().length) { + // filter out contracts, as these are dealt with via cross contract calls + const sigsNeeded = this.needsNonInvokerSigningBy().filter(id => !id.startsWith('C')); + if (sigsNeeded.length) { throw new AssembledTransaction.Errors.NeedsMoreSignatures( - "Transaction requires more signatures. " + + `Transaction requires signatures from ${sigsNeeded}. ` + "See `needsNonInvokerSigningBy` for details.", ); } @@ -760,9 +763,9 @@ export class AssembledTransaction { "scvVoid"), ) .map((entry) => - StrKey.encodeEd25519PublicKey( - entry.credentials().address().address().accountId().ed25519(), - ), + Address.fromScAddress( + entry.credentials().address().address(), + ).toString(), ), ), ]; diff --git a/test/e2e/initialize.sh b/test/e2e/initialize.sh index cb471901c..9bb1e6b53 100755 --- a/test/e2e/initialize.sh +++ b/test/e2e/initialize.sh @@ -14,26 +14,26 @@ dirname="$(CDPATH= cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" echo "###################### Initializing e2e tests ########################" -soroban="$dirname/../../target/bin/soroban" -if [[ -f "$soroban" ]]; then - current=$($soroban --version | head -n 1 | cut -d ' ' -f 2) +stellar=$(realpath "$dirname/../../target/bin/stellar") +if [[ -f "$stellar" ]]; then + current=$($stellar --version | head -n 1 | cut -d ' ' -f 2) desired=$(cat .cargo/config.toml | grep -oE -- "--version\s+\S+" | awk '{print $2}') if [[ "$current" != "$desired" ]]; then - echo "Current pinned soroban binary: $current. Desired: $desired. Building soroban binary." - (cd "$dirname/../.." && cargo install_soroban) + echo "Current pinned stellar binary: $current. Desired: $desired. Building stellar binary." + (cd "$dirname/../.." && cargo install_stellar) else - echo "Using soroban binary from ./target/bin" + echo "Using stellar binary from ./target/bin" fi else - echo "Building pinned soroban binary" - (cd "$dirname/../.." && cargo install_soroban) + echo "Building pinned stellar binary" + (cd "$dirname/../.." && cargo install_stellar) fi -NETWORK_STATUS=$(curl -s -X POST "$SOROBAN_RPC_URL" -H "Content-Type: application/json" -d '{ "jsonrpc": "2.0", "id": 8675309, "method": "getHealth" }' | sed -n 's/.*"status":\s*"\([^"]*\)".*/\1/p') +NETWORK_STATUS=$(curl -s -X POST "$STELLAR_RPC_URL" -H "Content-Type: application/json" -d '{ "jsonrpc": "2.0", "id": 8675309, "method": "getHealth" }' | sed -n 's/.*"status":\s*"\([^"]*\)".*/\1/p') echo Network -echo " RPC: $SOROBAN_RPC_URL" -echo " Passphrase: \"$SOROBAN_NETWORK_PASSPHRASE\"" +echo " RPC: $STELLAR_RPC_URL" +echo " Passphrase: \"$STELLAR_NETWORK_PASSPHRASE\"" echo " Status: $NETWORK_STATUS" if [[ "$NETWORK_STATUS" != "healthy" ]]; then @@ -41,23 +41,24 @@ if [[ "$NETWORK_STATUS" != "healthy" ]]; then exit 1 fi -$soroban keys generate $SOROBAN_ACCOUNT +$stellar keys generate $STELLAR_ACCOUNT -# retrieve the contracts using soroban contract init then build them if they dont already exist +# retrieve the contracts using stellar contract init then build them if they dont already exist # Define directory and WASM file paths target_dir="$dirname/test-contracts/target/wasm32-unknown-unknown/release" contracts_dir="$dirname/test-contracts" -repo_url="https://github.com/stellar/soroban-examples.git" +test_examples_repo="https://github.com/AhaLabs/soroban-test-examples.git" wasm_files=( - "soroban_other_custom_types_contract.wasm" - "soroban_atomic_swap_contract.wasm" - "soroban_token_contract.wasm" - "soroban_increment_contract.wasm" - "hello_world.wasm" + "custom_types.wasm" + "atomic_swap.wasm" + "token.wasm" + "increment.wasm" + "needs_a_signature.wasm" + "this_one_signs.wasm" ) get_remote_git_hash() { - git ls-remote "$repo_url" HEAD | cut -f1 + git ls-remote "$test_examples_repo" HEAD | cut -f1 } # Get the current git hash @@ -75,7 +76,7 @@ fi all_exist=true for wasm_file in "${wasm_files[@]}"; do if [ ! -f "$target_dir/$wasm_file" ]; then - all_exist=false + all_exist=false break fi done @@ -84,11 +85,11 @@ done if [ "$all_exist" = false ] || [ "$current_hash" != "$stored_hash" ]; then echo "WASM files are missing or contracts have been updated. Initializing and building contracts..." # Initialize contracts - $soroban contract init "$dirname/test-contracts" --with-example increment other_custom_types atomic_swap token - + npx degit --force AhaLabs/soroban-test-examples "$dirname/test-contracts" + # Change directory to test-contracts and build the contracts cd "$dirname/test-contracts" || { echo "Failed to change directory!"; exit 1; } - $soroban contract build + $stellar contract build # Save git hash to file echo "$current_hash" > "$hash_file" else diff --git a/test/e2e/src/test-contract-client-constructor.js b/test/e2e/src/test-contract-client-constructor.js index 16756156e..b03d5ec11 100644 --- a/test/e2e/src/test-contract-client-constructor.js +++ b/test/e2e/src/test-contract-client-constructor.js @@ -5,6 +5,7 @@ const { networkPassphrase, rpcUrl, generateFundedKeypair, + run, } = require("./util"); const { Address, contract } = require("../../../lib"); @@ -15,47 +16,32 @@ async function clientFromConstructor( if (!contracts[name]) { throw new Error( `Contract ${name} not found. ` + - `Pick one of: ${Object.keys(contracts).join(", ")}`, + `Pick one of: ${Object.keys(contracts).join()}`, ); } keypair = await keypair; // eslint-disable-line no-param-reassign const wallet = contract.basicNodeSigner(keypair, networkPassphrase); const { path } = contracts[name]; - const xdr = JSON.parse( - spawnSync( - "./target/bin/soroban", - ["contract", "inspect", "--wasm", path, "--output", "xdr-base64-array"], - { shell: true, encoding: "utf8" }, - ).stdout.trim(), - ); + const inspected = run( + `./target/bin/stellar contract inspect --wasm ${path} --output xdr-base64-array`, + ).stdout; + const xdr = JSON.parse(inspected); const spec = new contract.Spec(xdr); let wasmHash = contracts[name].hash; if (!wasmHash) { - wasmHash = spawnSync( - "./target/bin/soroban", - ["contract", "install", "--wasm", path], - { shell: true, encoding: "utf8" }, - ).stdout.trim(); + wasmHash = run( + `./target/bin/stellar contract install --wasm ${path}`, + ).stdout; } // TODO: do this with js-stellar-sdk, instead of shelling out to the CLI contractId = contractId ?? - spawnSync( - "./target/bin/soroban", - [ - // eslint-disable-line no-param-reassign - "contract", - "deploy", - "--source", - keypair.secret(), - "--wasm-hash", - wasmHash, - ], - { shell: true, encoding: "utf8" }, - ).stdout.trim(); + run( + `./target/bin/stellar contract deploy --source ${keypair.secret()} --wasm-hash ${wasmHash}`, + ).stdout; const client = new contract.Client(spec, { networkPassphrase, @@ -92,15 +78,15 @@ async function clientForFromTest(contractId, publicKey, keypair) { describe("Client", function () { before(async function () { const { client, keypair, contractId } = - await clientFromConstructor("helloWorld"); + await clientFromConstructor("customTypes"); const publicKey = keypair.publicKey(); const addr = Address.fromString(publicKey); this.context = { client, publicKey, addr, contractId, keypair }; }); it("can be constructed with `new Client`", async function () { - const { result } = await this.context.client.hello({ to: "tests" }); - expect(result).to.deep.equal(["Hello", "tests"]); + const { result } = await this.context.client.hello({ hello: "tests" }); + expect(result).to.equal("tests"); }); it("can be constructed with `from`", async function () { diff --git a/test/e2e/src/test-methods-as-args.js b/test/e2e/src/test-methods-as-args.js index 97cf096c8..0d7007766 100644 --- a/test/e2e/src/test-methods-as-args.js +++ b/test/e2e/src/test-methods-as-args.js @@ -8,8 +8,8 @@ function callMethod(method, args) { describe("methods-as-args", function () { it("should pass methods as arguments and have them still work", async function () { - const { client } = await clientFor("helloWorld"); - const { result } = await callMethod(client.hello, { to: "tests" }); - expect(result).to.deep.equal(["Hello", "tests"]); + const { client } = await clientFor("customTypes"); + const { result } = await callMethod(client.hello, { hello: "tests" }); + expect(result).to.equal("tests"); }); }); diff --git a/test/e2e/src/test-non-invoker-signing-by-contracts.js b/test/e2e/src/test-non-invoker-signing-by-contracts.js new file mode 100644 index 000000000..704c2e28f --- /dev/null +++ b/test/e2e/src/test-non-invoker-signing-by-contracts.js @@ -0,0 +1,32 @@ +const { expect } = require("chai"); +const { clientFor } = require("./util"); + +before(async function () { + const { contractId: signingContractId, keypair } = + await clientFor("doesSigning"); + const { client: needsSignature } = await clientFor("needsSignature", { + keypair, + }); + + const tx = await needsSignature.hello({ + to: keypair.publicKey(), + sign_with: signingContractId, + }); + this.context = { tx, doesSigningId: signingContractId }; +}); + +describe("needsNonInvokerSigningBy", function () { + it("does not assume stellar accounts", async function () { + expect(this.context.tx.needsNonInvokerSigningBy()[0]).to.equal( + this.context.doesSigningId, + ); + }); +}); + +describe("sign", function () { + it("doesn't throw error when nonInvokerSigningBy returns a contract", async function () { + expect( + async () => await this.context.tx.sign({ force: true }), + ).to.not.throw(); + }); +}); diff --git a/test/e2e/src/util.js b/test/e2e/src/util.js index 090213ef7..fedf3fa15 100644 --- a/test/e2e/src/util.js +++ b/test/e2e/src/util.js @@ -1,67 +1,58 @@ const { spawnSync } = require("node:child_process"); const { contract, Keypair } = require("../../../lib"); +const path = require("node:path"); -const basePath = `${__dirname}/../test-contracts/target/wasm32-unknown-unknown/release`; +/* + * Run a Bash command, returning stdout, stderr, and status code. + */ +function run(command) { + const [cmd, ...args] = command.split(" "); + const result = spawnSync(cmd, args, { shell: true, encoding: "utf8" }); + return { + stdout: result.stdout.trim(), + stderr: result.stderr.trim(), + status: result.status, + }; +} +module.exports.run = run; + +const stellar = "./target/bin/stellar"; +const basePath = path.resolve( + `${__dirname}/../test-contracts/target/wasm32-unknown-unknown/release`, +); const contracts = { customTypes: { - hash: spawnSync( - "./target/bin/soroban", - [ - "contract", - "install", - "--wasm", - `${basePath}/soroban_other_custom_types_contract.wasm`, - ], - { shell: true, encoding: "utf8" }, - ).stdout.trim(), - path: `${basePath}/soroban_custom_types_contract.wasm`, - }, - helloWorld: { - hash: spawnSync( - "./target/bin/soroban", - ["contract", "install", "--wasm", `${basePath}/hello_world.wasm`], - { shell: true, encoding: "utf8" }, - ).stdout.trim(), - path: `${basePath}/hello_world.wasm`, + hash: run( + `${stellar} contract install --wasm ${basePath}/custom_types.wasm`, + ).stdout, + path: `${basePath}/custom_types.wasm`, }, increment: { - hash: spawnSync( - "./target/bin/soroban", - [ - "contract", - "install", - "--wasm", - `${basePath}/soroban_increment_contract.wasm`, - ], - { shell: true, encoding: "utf8" }, - ).stdout.trim(), - path: `${basePath}/soroban_increment_contract.wasm`, + hash: run(`${stellar} contract install --wasm ${basePath}/increment.wasm`) + .stdout, + path: `${basePath}/increment.wasm`, }, swap: { - hash: spawnSync( - "./target/bin/soroban", - [ - "contract", - "install", - "--wasm", - `${basePath}/soroban_atomic_swap_contract.wasm`, - ], - { shell: true, encoding: "utf8" }, - ).stdout.trim(), - path: `${basePath}/soroban_atomic_swap_contract.wasm`, + hash: run(`${stellar} contract install --wasm ${basePath}/atomic_swap.wasm`) + .stdout, + path: `${basePath}/atomic_swap.wasm`, }, token: { - hash: spawnSync( - "./target/bin/soroban", - [ - "contract", - "install", - "--wasm", - `${basePath}/soroban_token_contract.wasm`, - ], - { shell: true, encoding: "utf8" }, - ).stdout.trim(), - path: `${basePath}/soroban_token_contract.wasm`, + hash: run(`${stellar} contract install --wasm ${basePath}/token.wasm`) + .stdout, + path: `${basePath}/token.wasm`, + }, + needsSignature: { + hash: run( + `${stellar} contract install --wasm ${basePath}/needs_a_signature.wasm`, + ).stdout, + path: `${basePath}/needs_a_signature.wasm`, + }, + doesSigning: { + hash: run( + `${stellar} contract install --wasm ${basePath}/this_one_signs.wasm`, + ).stdout, + path: `${basePath}/this_one_signs.wasm`, }, }; module.exports.contracts = contracts; @@ -97,10 +88,7 @@ module.exports.generateFundedKeypair = generateFundedKeypair; * By default, will re-deploy the contract every time. Pass in the same * `contractId` again if you want to re-use the a contract instance. */ -async function clientFor( - name, - { keypair = generateFundedKeypair(), contractId } = {}, -) { +async function clientFor(name, { keypair, contractId } = {}) { if (!contracts[name]) { throw new Error( `Contract ${name} not found. ` + @@ -108,34 +96,22 @@ async function clientFor( ); } - keypair = await keypair; // eslint-disable-line no-param-reassign - const wallet = contract.basicNodeSigner(keypair, networkPassphrase); + const internalKeypair = keypair ?? (await generateFundedKeypair()); + const wallet = contract.basicNodeSigner(internalKeypair, networkPassphrase); let wasmHash = contracts[name].hash; if (!wasmHash) { - wasmHash = spawnSync( - "./target/bin/soroban", - ["contract", "install", "--wasm", contracts[name].path], - { shell: true, encoding: "utf8" }, - ).stdout.trim(); + wasmHash = run( + `${stellar} contract install --wasm ${contracts[name].path}`, + ).stdout; } // TODO: do this with js-stellar-sdk, instead of shelling out to the CLI contractId = contractId ?? - spawnSync( - "./target/bin/soroban", - [ - // eslint-disable-line no-param-reassign - "contract", - "deploy", - "--source", - keypair.secret(), - "--wasm-hash", - wasmHash, - ], - { shell: true, encoding: "utf8" }, - ).stdout.trim(); + run( + `${stellar} contract deploy --source ${internalKeypair.secret()} --wasm-hash ${wasmHash}`, + ).stdout; const client = await contract.Client.fromWasmHash( wasmHash, @@ -144,13 +120,13 @@ async function clientFor( contractId, rpcUrl, allowHttp: true, - publicKey: keypair.publicKey(), + publicKey: internalKeypair.publicKey(), ...wallet, }, "hex", ); return { - keypair, + keypair: internalKeypair, client, contractId, }; diff --git a/test/e2e/wasms/README.md b/test/e2e/wasms/README.md index d44181b20..fa318a0b0 100644 --- a/test/e2e/wasms/README.md +++ b/test/e2e/wasms/README.md @@ -1,16 +1,13 @@ -Copy-pasted `.wasm` files (for now!) -==================================== +Wasm files for testing +====================== -These are built versions of contracts with source code that currently live in -the `test-wasms` directory within soroban-tools. Even there, they are mostly -copy-pasted contracts from `soroban-examples`. +These are copies of Wasm files from testnet or other live networks, used for testing -It would probably be best to include `soroban-examples` as a git submodule, -then directly rely on the contracts we want to test against. No more -copy-pasting! But until we can all agree on such an approach, maybe this is -good enough? +The two files here were created with: -Soon the CLI will have a `soroban init` command that allows setting up a new -Cargo workspace and may allow specifying which example contracts to put in it. -If the git submodule is controversial, then we could use that new `init` -command instead. +``` +stellar contract fetch --id CA4SAAVG73IW6JQK4F45NAR7IAT7DI5MQPRSYSQWEWX4RKUNLL77MNOF --network testnet --out-file test/e2e/wasms/stellar_asset_wrapper.wasm +stellar contract fetch --id CAIGE5STPFOY7V24NNM7KNFNELMXSHZP7ZZV3GZRADRDCUTJ7VCWTQYL --network testnet --out-file test/e2e/wasms/stellar_asset_admin.wasm +``` + +And created based on the bug report in github.com/stellar/stellar-sdk-js/issues/1030 diff --git a/test/e2e/wasms/stellar_asset_admin.wasm b/test/e2e/wasms/stellar_asset_admin.wasm new file mode 100644 index 000000000..b0216b5f0 Binary files /dev/null and b/test/e2e/wasms/stellar_asset_admin.wasm differ diff --git a/test/e2e/wasms/stellar_asset_wrapper.wasm b/test/e2e/wasms/stellar_asset_wrapper.wasm new file mode 100644 index 000000000..a55dafd51 Binary files /dev/null and b/test/e2e/wasms/stellar_asset_wrapper.wasm differ diff --git a/test/e2e/wasms/test_custom_types.wasm b/test/e2e/wasms/test_custom_types.wasm deleted file mode 100755 index b409bc3c5..000000000 Binary files a/test/e2e/wasms/test_custom_types.wasm and /dev/null differ diff --git a/test/e2e/wasms/test_hello_world.wasm b/test/e2e/wasms/test_hello_world.wasm deleted file mode 100755 index 79e09d3dc..000000000 Binary files a/test/e2e/wasms/test_hello_world.wasm and /dev/null differ diff --git a/test/e2e/wasms/test_swap.wasm b/test/e2e/wasms/test_swap.wasm deleted file mode 100755 index b609c4b94..000000000 Binary files a/test/e2e/wasms/test_swap.wasm and /dev/null differ diff --git a/test/e2e/wasms/test_token.wasm b/test/e2e/wasms/test_token.wasm deleted file mode 100755 index 6d2f77fac..000000000 Binary files a/test/e2e/wasms/test_token.wasm and /dev/null differ