Skip to content

Commit 17ae10c

Browse files
authored
Merge pull request #99 from hyperweb-io/starship-eth
Starship eth
2 parents b25a9d2 + 310fe5f commit 17ae10c

File tree

8 files changed

+464
-45
lines changed

8 files changed

+464
-45
lines changed

networks/ethereum/devnet/__tests__/ethers.test.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ describe('ETH Transfer Test', () => {
2929
await usdtContract.waitForDeployment();
3030
usdtAddress = await usdtContract.getAddress();
3131

32-
});
32+
}, 60000);
3333

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

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

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

6868
expect(finalSenderBalance).toEqual(initialSenderBalance - amountToSend);
6969
expect(finalReceiverBalance).toEqual(initialReceiverBalance + amountToSend);
70-
});
70+
}, 60000);
7171
});

networks/ethereum/devnet/__tests__/noethers.test.ts

+30-26
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import { ContractEncoder, AbiFunctionItem } from '../../src/utils/ContractEncode
99
// E.g., Hardhat node: http://127.0.0.1:8545
1010
// or a testnet node: https://goerli.infura.io/v3/...
1111
const RPC_URL = 'http://127.0.0.1:8545';
12+
// const RPC_URL = 'https://bsc-testnet.bnbchain.org';
1213

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

4647
const resp = await axios.post(RPC_URL, callPayload);
4748
const hexBalance = resp.data.result;
49+
if (hexBalance === undefined || hexBalance === null) {
50+
throw new Error(`eth_call returned invalid result: ${JSON.stringify(resp.data)}`);
51+
}
4852
return BigInt(hexBalance);
4953
}
5054

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

122-
it('should transfer USDT to receiver and verify balance increments by the transfer amount', async () => {
123-
const beforeReceiverBalance = await getUSDTBalance(receiverAddress);
124-
console.log('Before transfer, receiver USDT balance:', beforeReceiverBalance.toString());
125-
126-
const transferAmount = 1_000_000_000_000_000_000n; // 1 USDT
127-
128-
const dataHex = usdt.transfer(receiverAddress, transferAmount);
129-
130-
const { txHash, wait } = await transfer.sendLegacyTransactionAutoGasLimit(
131-
usdtAddress,
132-
0n,
133-
dataHex
134-
);
135-
expect(txHash).toMatch(/^0x[0-9a-fA-F]+$/);
136-
137-
const receipt = await wait();
138-
expect(receipt.status).toBe('0x1');
139-
140-
const afterReceiverBalance = await getUSDTBalance(receiverAddress);
141-
console.log('After transfer, receiver USDT balance:', afterReceiverBalance.toString());
142-
143-
const delta = afterReceiverBalance - beforeReceiverBalance;
144-
console.log('Receiver USDT balance delta:', delta.toString());
145-
expect(delta).toBe(transferAmount);
146-
});
147-
148126
it('should send ETH from sender to receiver via EIP-1559, and check balances', async () => {
149127
const beforeSenderBalance = await signerSender.getBalance();
150128
const beforeReceiverBalance = await signerReceiver.getBalance();
@@ -185,4 +163,30 @@ describe('sending Tests', () => {
185163
expect(senderDelta).toBeGreaterThanOrEqual(valueWei);
186164
}, 60000);
187165

166+
it('should transfer USDT to receiver and verify balance increments by the transfer amount', async () => {
167+
const beforeReceiverBalance = await getUSDTBalance(receiverAddress);
168+
console.log('Before transfer, receiver USDT balance:', beforeReceiverBalance.toString());
169+
170+
const transferAmount = 1_000_000_000_000_000_000n; // 1 USDT
171+
172+
const dataHex = usdt.transfer(receiverAddress, transferAmount);
173+
174+
const { txHash, wait } = await transfer.sendLegacyTransactionAutoGasLimit(
175+
usdtAddress,
176+
0n,
177+
dataHex
178+
);
179+
expect(txHash).toMatch(/^0x[0-9a-fA-F]+$/);
180+
181+
const receipt = await wait();
182+
expect(receipt.status).toBe('0x1');
183+
184+
const afterReceiverBalance = await getUSDTBalance(receiverAddress);
185+
console.log('After transfer, receiver USDT balance:', afterReceiverBalance.toString());
186+
187+
const delta = afterReceiverBalance - beforeReceiverBalance;
188+
console.log('Receiver USDT balance delta:', delta.toString());
189+
expect(delta).toBe(transferAmount);
190+
});
191+
188192
});

networks/ethereum/package.json

+10-3
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,10 @@
2626
"test:devnet": "npx jest --preset ts-jest devnet/__tests__/send.icjs.test.ts",
2727
"test:ethers": "npx jest --preset ts-jest devnet/__tests__/ethers.test.ts",
2828
"test:noethers": "npx jest --preset ts-jest devnet/__tests__/noethers.test.ts",
29-
"run-ganache": "bash devnet/run-ganache.sh"
29+
"run-ganache": "bash devnet/run-ganache.sh",
30+
"starship": "npx starship start --config ./starship/configs/eth-lite.yaml",
31+
"starship:stop": "npx starship stop",
32+
"starship:test": "npx jest --preset ts-jest starship/__tests__/token.test.ts"
3033
},
3134
"dependencies": {
3235
"@ethersproject/bignumber": "^5.7.0",
@@ -44,5 +47,9 @@
4447
"ethereum",
4548
"blockchain",
4649
"transaction"
47-
]
48-
}
50+
],
51+
"devDependencies": {
52+
"@starship-ci/cli": "3.5.0",
53+
"starshipjs": "^3.3.0"
54+
}
55+
}

networks/ethereum/src/signers/SignerFromPrivateKey.ts

+19-12
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import axios from 'axios';
22
import { keccak256 } from 'ethereum-cryptography/keccak';
33
import { secp256k1 } from '@noble/curves/secp256k1';
4-
import { bytesToHex, hexToBytes, equalsBytes } from 'ethereum-cryptography/utils';
4+
import { bytesToHex, hexToBytes } from 'ethereum-cryptography/utils';
55
import * as rlp from 'rlp'; // Updated import
66
import { TransactionReceipt } from '../types/transaction';
77

@@ -177,24 +177,28 @@ export class SignerFromPrivateKey {
177177
wait: () => Promise<TransactionReceipt>;
178178
}> {
179179
const fromAddr = this.getAddress();
180-
console.log('from address in sendLegacyTransaction:', fromAddr);
180+
// console.log('from address in sendLegacyTransaction:', fromAddr);
181181
const nonce = await this.getNonce();
182-
console.log('Nonce:', nonce);
182+
// console.log('Nonce:', nonce);
183183
const chainId = await this.getChainId();
184184

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

191192
// RLP for signing (chainId in item #7, then 0,0 placeholders)
192193
const txForSign = [
193194
hexToBytes(nonceHex),
194195
hexToBytes(gasPriceHex),
195196
hexToBytes(gasLimitHex),
196197
hexToBytes(to),
197-
hexToBytes(valueHex),
198+
199+
// hexToBytes(valueHex),
200+
valueBytes,
201+
198202
hexToBytes(dataHex),
199203
hexToBytes(this.toHexPadded(chainId)),
200204
new Uint8Array([]),
@@ -215,13 +219,16 @@ export class SignerFromPrivateKey {
215219
hexToBytes(gasPriceHex),
216220
hexToBytes(gasLimitHex),
217221
hexToBytes(to),
218-
hexToBytes(valueHex),
222+
223+
// hexToBytes(valueHex),
224+
valueBytes,
225+
219226
hexToBytes(dataHex),
220227
hexToBytes(vHex),
221228
r,
222229
s,
223230
];
224-
console.log('txSigned:', txSigned);
231+
// console.log('txSigned:', txSigned);
225232

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

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

323-
// For typed transactions, v = recovery
330+
// 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.
324331
const v = recovery;
325-
const vHex = this.toHexPadded(v);
332+
const vBytes = v === 0 ? new Uint8Array([]) : hexToBytes(this.toHexPadded(v));
326333

327334
// RLP( [chainId, nonce, maxPriorityFeePerGas, maxFeePerGas, gasLimit, to, value, data, accessList, v, r, s] )
328335
const txSigned = [
@@ -335,7 +342,7 @@ export class SignerFromPrivateKey {
335342
hexToBytes(valueHex),
336343
hexToBytes(data),
337344
accessList,
338-
hexToBytes(vHex),
345+
vBytes,
339346
r,
340347
s
341348
];
@@ -384,9 +391,9 @@ export class SignerFromPrivateKey {
384391
// Estimate gas limit from the node
385392
const estimatedGasLimit = await this.estimateGas(to, valueWei, dataHex);
386393
const gasLimit = BigInt(Math.ceil(Number(estimatedGasLimit) * 1.5)); // 1.5x estimated
387-
console.log('Gas Limit:', gasLimit.toString())
394+
// console.log('Gas Limit:', gasLimit.toString())
388395

389-
console.log('Value:', valueWei.toString());
396+
// console.log('Value:', valueWei.toString());
390397
return this.sendLegacyTransaction(to, valueWei, dataHex, gasPrice, gasLimit);
391398
}
392399

@@ -425,7 +432,7 @@ export class SignerFromPrivateKey {
425432

426433
const txParams: any = {
427434
from: fromAddr,
428-
value: this.toHexPadded(valueWei),
435+
value: valueWei === 0n ? '0x0' : '0x' + valueWei.toString(16),
429436
data: data
430437
};
431438

networks/ethereum/starship/README.md

+153
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,153 @@
1+
## TLDR
2+
3+
Deploy
4+
5+
```sh
6+
yarn starship
7+
```
8+
9+
Run Tests
10+
11+
```sh
12+
yarn starship:test
13+
```
14+
15+
Teardown
16+
17+
```sh
18+
yarn starship:stop
19+
```
20+
21+
Port fording manually if needed
22+
```sh
23+
kubectl port-forward pods/ethereum-1337-0 8545:8545
24+
```
25+
26+
Get chain id from node
27+
```sh
28+
curl -X POST -H "Content-Type: application/json" \
29+
--data '{"jsonrpc":"2.0","method":"eth_chainId","params":[],"id":1}' \
30+
http://localhost:8545/
31+
```
32+
33+
Get balance:
34+
```
35+
curl -X POST \
36+
-H "Content-Type: application/json" \
37+
--data '{"jsonrpc":"2.0","method":"eth_getBalance","params":["0x7e5f4552091a69125d5dfcb7b8c2659029395bdf", "latest"],"id":1}' \
38+
http://localhost:8545
39+
```
40+
41+
## 1. Installation
42+
43+
Inorder to get started with starship, one needs to install the following
44+
45+
- `kubectl`: https://kubernetes.io/docs/tasks/tools/
46+
- `kind`: https://kind.sigs.k8s.io/docs/user/quick-start/#installation
47+
- `helm`: https://helm.sh/docs/intro/install/
48+
49+
Note: To make the process easy we have a simple command that will try and install dependencies
50+
so that you dont have to.
51+
52+
```bash
53+
yarn starship setup
54+
```
55+
56+
This command will
57+
58+
- check (and install) if your system has all the dependencies needed to run the e2e tests wtih Starship
59+
- fetch the helm charts for Starship
60+
61+
## 2. Connect to a kubernetes cluster
62+
63+
Inorder to set up the infrastructure, for Starship, we need access to a kubernetes cluster.
64+
One can either perform connect to a
65+
66+
- remote cluster in a managed kubernetes service
67+
- use kubernetes desktop to spin up a cluster
68+
- use kind to create a local cluster on local machine
69+
70+
To make this easier we have a handy command which will create a local kind cluster and give you access
71+
to a kubernetes cluster locally.
72+
73+
NOTE: Resources constraint on local machine will affect the performance of Starship spinup time
74+
75+
```bash
76+
yarn starship setup-kind
77+
```
78+
79+
Run the following command to check connection to a k8s cluster
80+
81+
```bash
82+
kubectl get pods
83+
```
84+
85+
## 3. Start Starship
86+
87+
Now with the dependencies and a kubernetes cluster in handy, we can proceed with creating the mini-cosmos ecosystem
88+
89+
Run
90+
91+
```bash
92+
yarn starship deploy
93+
```
94+
95+
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
96+
97+
Note: Spinup will take some time, while you wait for the system, can check the progress in another tab with `kubectl get pods`
98+
99+
## 4. Run the tests
100+
101+
We have everything we need, our desired infrastructure is now running as intended, now we can run
102+
our end-to-end tests.
103+
104+
Run
105+
106+
```bash
107+
npm run starship:test
108+
```
109+
110+
## 5. Stop the infra
111+
112+
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).
113+
114+
Once the state of the mini-cosmos is corrupted, you can stop the deployments with
115+
116+
```bash
117+
npm run starship clean
118+
```
119+
120+
Which will
121+
122+
- Stop port-forwarding the traffic to your local
123+
- Delete all the helm charts deployed
124+
125+
## 6. Cleanup kind (optional)
126+
127+
If you are using kind for your kubernetes cluster, you can delete it with
128+
129+
```bash
130+
yarn starship clean-kind
131+
```
132+
133+
## Related
134+
135+
Checkout these related projects:
136+
137+
- [@cosmology/telescope](https://github.com/hyperweb-io/telescope) Your Frontend Companion for Building with TypeScript with Cosmos SDK Modules.
138+
- [@cosmwasm/ts-codegen](https://github.com/CosmWasm/ts-codegen) Convert your CosmWasm smart contracts into dev-friendly TypeScript classes.
139+
- [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.
140+
- [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.
141+
- [create-interchain-app](https://github.com/hyperweb-io/create-interchain-app) Set up a modern Cosmos app by running one command.
142+
- [interchain-ui](https://github.com/hyperweb-io/interchain-ui) The Interchain Design System, empowering developers with a flexible, easy-to-use UI kit.
143+
- [starship](https://github.com/hyperweb-io/starship) Unified Testing and Development for the Interchain.
144+
145+
## Credits
146+
147+
🛠 Built by Hyperweb (formerly Cosmology) — if you like our tools, please checkout and contribute to [our github ⚛️](https://github.com/hyperweb-io)
148+
149+
## Disclaimer
150+
151+
AS DESCRIBED IN THE LICENSES, THE SOFTWARE IS PROVIDED “AS IS”, AT YOUR OWN RISK, AND WITHOUT WARRANTIES OF ANY KIND.
152+
153+
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.

0 commit comments

Comments
 (0)