Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Starship eth #99

Merged
merged 15 commits into from
Mar 17, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 4 additions & 4 deletions networks/ethereum/devnet/__tests__/ethers.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ describe('ETH Transfer Test', () => {
await usdtContract.waitForDeployment();
usdtAddress = await usdtContract.getAddress();

});
}, 60000);

it('should transfer ETH from wallet0 to wallet1 and check balances', async () => {
const initialBalanceSender = await provider.getBalance(walletSender.address);
Expand All @@ -50,10 +50,10 @@ describe('ETH Transfer Test', () => {
expect(finalBalanceSender).toBeLessThan(initialBalanceSender);
expect(finalBalanceReceiver).toBeGreaterThan(initialBalanceReceiver);
expect(finalBalanceReceiver).toEqual(initialBalanceReceiver + amountToSend);
});
}, 60000);

it('should transfer USDT from sender to receiver', async () => {
const decimals = 6;
const decimals = 6;
const amountToSend = ethers.parseUnits('100', decimals);

const initialSenderBalance = await usdtContract.balanceOf(walletSender.address);
Expand All @@ -67,5 +67,5 @@ describe('ETH Transfer Test', () => {

expect(finalSenderBalance).toEqual(initialSenderBalance - amountToSend);
expect(finalReceiverBalance).toEqual(initialReceiverBalance + amountToSend);
});
}, 60000);
});
56 changes: 30 additions & 26 deletions networks/ethereum/devnet/__tests__/noethers.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import { ContractEncoder, AbiFunctionItem } from '../../src/utils/ContractEncode
// E.g., Hardhat node: http://127.0.0.1:8545
// or a testnet node: https://goerli.infura.io/v3/...
const RPC_URL = 'http://127.0.0.1:8545';
// const RPC_URL = 'https://bsc-testnet.bnbchain.org';

// Two example private keys
const privSender = '0x' + '0'.repeat(63) + '1';
Expand Down Expand Up @@ -45,6 +46,9 @@ describe('sending Tests', () => {

const resp = await axios.post(RPC_URL, callPayload);
const hexBalance = resp.data.result;
if (hexBalance === undefined || hexBalance === null) {
throw new Error(`eth_call returned invalid result: ${JSON.stringify(resp.data)}`);
}
return BigInt(hexBalance);
}

Expand Down Expand Up @@ -119,32 +123,6 @@ describe('sending Tests', () => {
expect(senderDelta).toBeGreaterThanOrEqual(BigInt(valueWei));
}, 60000); // Increased Jest timeout to 60s for potential network delays

it('should transfer USDT to receiver and verify balance increments by the transfer amount', async () => {
const beforeReceiverBalance = await getUSDTBalance(receiverAddress);
console.log('Before transfer, receiver USDT balance:', beforeReceiverBalance.toString());

const transferAmount = 1_000_000_000_000_000_000n; // 1 USDT

const dataHex = usdt.transfer(receiverAddress, transferAmount);

const { txHash, wait } = await transfer.sendLegacyTransactionAutoGasLimit(
usdtAddress,
0n,
dataHex
);
expect(txHash).toMatch(/^0x[0-9a-fA-F]+$/);

const receipt = await wait();
expect(receipt.status).toBe('0x1');

const afterReceiverBalance = await getUSDTBalance(receiverAddress);
console.log('After transfer, receiver USDT balance:', afterReceiverBalance.toString());

const delta = afterReceiverBalance - beforeReceiverBalance;
console.log('Receiver USDT balance delta:', delta.toString());
expect(delta).toBe(transferAmount);
});

it('should send ETH from sender to receiver via EIP-1559, and check balances', async () => {
const beforeSenderBalance = await signerSender.getBalance();
const beforeReceiverBalance = await signerReceiver.getBalance();
Expand Down Expand Up @@ -185,4 +163,30 @@ describe('sending Tests', () => {
expect(senderDelta).toBeGreaterThanOrEqual(valueWei);
}, 60000);

it('should transfer USDT to receiver and verify balance increments by the transfer amount', async () => {
const beforeReceiverBalance = await getUSDTBalance(receiverAddress);
console.log('Before transfer, receiver USDT balance:', beforeReceiverBalance.toString());

const transferAmount = 1_000_000_000_000_000_000n; // 1 USDT

const dataHex = usdt.transfer(receiverAddress, transferAmount);

const { txHash, wait } = await transfer.sendLegacyTransactionAutoGasLimit(
usdtAddress,
0n,
dataHex
);
expect(txHash).toMatch(/^0x[0-9a-fA-F]+$/);

const receipt = await wait();
expect(receipt.status).toBe('0x1');

const afterReceiverBalance = await getUSDTBalance(receiverAddress);
console.log('After transfer, receiver USDT balance:', afterReceiverBalance.toString());

const delta = afterReceiverBalance - beforeReceiverBalance;
console.log('Receiver USDT balance delta:', delta.toString());
expect(delta).toBe(transferAmount);
});

});
13 changes: 10 additions & 3 deletions networks/ethereum/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,10 @@
"test:devnet": "npx jest --preset ts-jest devnet/__tests__/send.icjs.test.ts",
"test:ethers": "npx jest --preset ts-jest devnet/__tests__/ethers.test.ts",
"test:noethers": "npx jest --preset ts-jest devnet/__tests__/noethers.test.ts",
"run-ganache": "bash devnet/run-ganache.sh"
"run-ganache": "bash devnet/run-ganache.sh",
"starship": "npx starship start --config ./starship/configs/eth-lite.yaml",
"starship:stop": "npx starship stop",
"starship:test": "npx jest --preset ts-jest starship/__tests__/token.test.ts"
},
"dependencies": {
"@ethersproject/bignumber": "^5.7.0",
Expand All @@ -44,5 +47,9 @@
"ethereum",
"blockchain",
"transaction"
]
}
],
"devDependencies": {
"@starship-ci/cli": "3.5.0",
"starshipjs": "^3.3.0"
}
}
31 changes: 19 additions & 12 deletions networks/ethereum/src/signers/SignerFromPrivateKey.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import axios from 'axios';
import { keccak256 } from 'ethereum-cryptography/keccak';
import { secp256k1 } from '@noble/curves/secp256k1';
import { bytesToHex, hexToBytes, equalsBytes } from 'ethereum-cryptography/utils';
import { bytesToHex, hexToBytes } from 'ethereum-cryptography/utils';
import * as rlp from 'rlp'; // Updated import
import { TransactionReceipt } from '../types/transaction';

Expand Down Expand Up @@ -177,24 +177,28 @@ export class SignerFromPrivateKey {
wait: () => Promise<TransactionReceipt>;
}> {
const fromAddr = this.getAddress();
console.log('from address in sendLegacyTransaction:', fromAddr);
// console.log('from address in sendLegacyTransaction:', fromAddr);
const nonce = await this.getNonce();
console.log('Nonce:', nonce);
// console.log('Nonce:', nonce);
const chainId = await this.getChainId();

// Convert inputs to padded hex strings
const nonceHex = this.toHexPadded(nonce);
const gasPriceHex = this.toHexPadded(gasPrice);
const gasLimitHex = this.toHexPadded(gasLimit);
const valueHex = this.toHexPadded(valueWei);
const valueBytes = valueWei === 0n ? new Uint8Array([]) : hexToBytes('0x' + valueWei.toString(16));

// RLP for signing (chainId in item #7, then 0,0 placeholders)
const txForSign = [
hexToBytes(nonceHex),
hexToBytes(gasPriceHex),
hexToBytes(gasLimitHex),
hexToBytes(to),
hexToBytes(valueHex),

// hexToBytes(valueHex),
valueBytes,

hexToBytes(dataHex),
hexToBytes(this.toHexPadded(chainId)),
new Uint8Array([]),
Expand All @@ -215,13 +219,16 @@ export class SignerFromPrivateKey {
hexToBytes(gasPriceHex),
hexToBytes(gasLimitHex),
hexToBytes(to),
hexToBytes(valueHex),

// hexToBytes(valueHex),
valueBytes,

hexToBytes(dataHex),
hexToBytes(vHex),
r,
s,
];
console.log('txSigned:', txSigned);
// console.log('txSigned:', txSigned);

const serializedTx = rlp.encode(txSigned);
const rawTxHex = '0x' + bytesToHex(serializedTx);
Expand Down Expand Up @@ -320,9 +327,9 @@ export class SignerFromPrivateKey {

const { r, s, recovery } = this.signWithRecovery(msgHash);

// For typed transactions, v = recovery
// For typed transactions, v is simply the recovery value. To ensure canonical RLP encoding (no leading zero bytes), encode 0 as an empty byte array.
const v = recovery;
const vHex = this.toHexPadded(v);
const vBytes = v === 0 ? new Uint8Array([]) : hexToBytes(this.toHexPadded(v));

// RLP( [chainId, nonce, maxPriorityFeePerGas, maxFeePerGas, gasLimit, to, value, data, accessList, v, r, s] )
const txSigned = [
Expand All @@ -335,7 +342,7 @@ export class SignerFromPrivateKey {
hexToBytes(valueHex),
hexToBytes(data),
accessList,
hexToBytes(vHex),
vBytes,
r,
s
];
Expand Down Expand Up @@ -384,9 +391,9 @@ export class SignerFromPrivateKey {
// Estimate gas limit from the node
const estimatedGasLimit = await this.estimateGas(to, valueWei, dataHex);
const gasLimit = BigInt(Math.ceil(Number(estimatedGasLimit) * 1.5)); // 1.5x estimated
console.log('Gas Limit:', gasLimit.toString())
// console.log('Gas Limit:', gasLimit.toString())

console.log('Value:', valueWei.toString());
// console.log('Value:', valueWei.toString());
return this.sendLegacyTransaction(to, valueWei, dataHex, gasPrice, gasLimit);
}

Expand Down Expand Up @@ -425,7 +432,7 @@ export class SignerFromPrivateKey {

const txParams: any = {
from: fromAddr,
value: this.toHexPadded(valueWei),
value: valueWei === 0n ? '0x0' : '0x' + valueWei.toString(16),
data: data
};

Expand Down
153 changes: 153 additions & 0 deletions networks/ethereum/starship/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,153 @@
## TLDR

Deploy

```sh
yarn starship
```

Run Tests

```sh
yarn starship:test
```

Teardown

```sh
yarn starship:stop
```

Port fording manually if needed
```sh
kubectl port-forward pods/ethereum-1337-0 8545:8545
```

Get chain id from node
```sh
curl -X POST -H "Content-Type: application/json" \
--data '{"jsonrpc":"2.0","method":"eth_chainId","params":[],"id":1}' \
http://localhost:8545/
```

Get balance:
```
curl -X POST \
-H "Content-Type: application/json" \
--data '{"jsonrpc":"2.0","method":"eth_getBalance","params":["0x7e5f4552091a69125d5dfcb7b8c2659029395bdf", "latest"],"id":1}' \
http://localhost:8545
```

## 1. Installation

Inorder to get started with starship, one needs to install the following

- `kubectl`: https://kubernetes.io/docs/tasks/tools/
- `kind`: https://kind.sigs.k8s.io/docs/user/quick-start/#installation
- `helm`: https://helm.sh/docs/intro/install/

Note: To make the process easy we have a simple command that will try and install dependencies
so that you dont have to.

```bash
yarn starship setup
```

This command will

- check (and install) if your system has all the dependencies needed to run the e2e tests wtih Starship
- fetch the helm charts for Starship

## 2. Connect to a kubernetes cluster

Inorder to set up the infrastructure, for Starship, we need access to a kubernetes cluster.
One can either perform connect to a

- remote cluster in a managed kubernetes service
- use kubernetes desktop to spin up a cluster
- use kind to create a local cluster on local machine

To make this easier we have a handy command which will create a local kind cluster and give you access
to a kubernetes cluster locally.

NOTE: Resources constraint on local machine will affect the performance of Starship spinup time

```bash
yarn starship setup-kind
```

Run the following command to check connection to a k8s cluster

```bash
kubectl get pods
```

## 3. Start Starship

Now with the dependencies and a kubernetes cluster in handy, we can proceed with creating the mini-cosmos ecosystem

Run

```bash
yarn starship deploy
```

We use the config file `configs/config.yaml` as the genesis file to define the topology of the e2e test infra. Change it as required

Note: Spinup will take some time, while you wait for the system, can check the progress in another tab with `kubectl get pods`

## 4. Run the tests

We have everything we need, our desired infrastructure is now running as intended, now we can run
our end-to-end tests.

Run

```bash
npm run starship:test
```

## 5. Stop the infra

The tests should be ideompotent, so the tests can be run multiple times (which is recommeded), since the time to spinup is still high (around 5 to 10 mins).

Once the state of the mini-cosmos is corrupted, you can stop the deployments with

```bash
npm run starship clean
```

Which will

- Stop port-forwarding the traffic to your local
- Delete all the helm charts deployed

## 6. Cleanup kind (optional)

If you are using kind for your kubernetes cluster, you can delete it with

```bash
yarn starship clean-kind
```

## Related

Checkout these related projects:

- [@cosmology/telescope](https://github.com/hyperweb-io/telescope) Your Frontend Companion for Building with TypeScript with Cosmos SDK Modules.
- [@cosmwasm/ts-codegen](https://github.com/CosmWasm/ts-codegen) Convert your CosmWasm smart contracts into dev-friendly TypeScript classes.
- [chain-registry](https://github.com/hyperweb-io/chain-registry) Everything from token symbols, logos, and IBC denominations for all assets you want to support in your application.
- [interchain-kit](https://github.com/hyperweb-io/interchain-kit) Experience the convenience of connecting with a variety of web3 wallets through a single, streamlined interface.
- [create-interchain-app](https://github.com/hyperweb-io/create-interchain-app) Set up a modern Cosmos app by running one command.
- [interchain-ui](https://github.com/hyperweb-io/interchain-ui) The Interchain Design System, empowering developers with a flexible, easy-to-use UI kit.
- [starship](https://github.com/hyperweb-io/starship) Unified Testing and Development for the Interchain.

## Credits

🛠 Built by Hyperweb (formerly Cosmology) — if you like our tools, please checkout and contribute to [our github ⚛️](https://github.com/hyperweb-io)

## Disclaimer

AS DESCRIBED IN THE LICENSES, THE SOFTWARE IS PROVIDED “AS IS”, AT YOUR OWN RISK, AND WITHOUT WARRANTIES OF ANY KIND.

No developer or entity involved in creating this software will be liable for any claims or damages whatsoever associated with your use, inability to use, or your interaction with other users of the code, including any direct, indirect, incidental, special, exemplary, punitive or consequential damages, or loss of profits, cryptocurrencies, tokens, or anything else of value.
Loading
Loading