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

[pallet-revive] adjust fee dry-run calculation #6393

Merged
merged 33 commits into from
Nov 12, 2024
Merged
Show file tree
Hide file tree
Changes from 14 commits
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
b8fca6a
Revive fixes
pgherveou Nov 6, 2024
dd7fe75
Add context in build.rs
pgherveou Nov 6, 2024
06481e0
Fixes
pgherveou Nov 6, 2024
e5ea069
fixes
pgherveou Nov 6, 2024
4c0cd4b
fix
pgherveou Nov 6, 2024
175cc1a
wip
pgherveou Nov 6, 2024
c60f840
Fix typo
pgherveou Nov 7, 2024
5f6eef3
err message
pgherveou Nov 7, 2024
844dcf3
Update from pgherveou running command 'prdoc --audience runtime_dev -…
actions-user Nov 7, 2024
623a1a7
Bump ah westend
pgherveou Nov 8, 2024
8844b16
Fix dry run
pgherveou Nov 8, 2024
2aca171
Extract revert message
pgherveou Nov 8, 2024
7409ccc
fix
pgherveou Nov 8, 2024
f3603e4
fix
pgherveou Nov 8, 2024
3b3d113
fix formattting
pgherveou Nov 8, 2024
b17d31c
add nice error
pgherveou Nov 8, 2024
a5a1a41
Merge branch 'pg/subxt_038' into pg/revive-fixes
pgherveou Nov 8, 2024
18c4a72
clippy
pgherveou Nov 8, 2024
79036e1
Fix hash / receipt
pgherveou Nov 9, 2024
4608073
Support logs in receipts
pgherveou Nov 9, 2024
2067833
fmt
pgherveou Nov 9, 2024
c2096c7
fix lint
pgherveou Nov 9, 2024
30b1745
Merge branch 'master' into pg/revive-fixes
pgherveou Nov 11, 2024
ccdd2c9
Update js examples to run with geth
pgherveou Nov 11, 2024
0fa4c7b
Add tests for logs and revert
pgherveou Nov 11, 2024
556a6ea
update
pgherveou Nov 11, 2024
cfa5b30
fix clippy
pgherveou Nov 11, 2024
8833aca
fix clippy
pgherveou Nov 11, 2024
c265992
fix
pgherveou Nov 11, 2024
1b119b6
fix clippy
pgherveou Nov 11, 2024
a4f8e73
use ethabi
pgherveou Nov 11, 2024
814229f
Add test for invalid transaction
pgherveou Nov 11, 2024
8dcfe61
Align subxt version
pgherveou Nov 12, 2024
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
Original file line number Diff line number Diff line change
Expand Up @@ -127,7 +127,7 @@ pub const VERSION: RuntimeVersion = RuntimeVersion {
spec_name: create_runtime_str!("westmint"),
impl_name: create_runtime_str!("westmint"),
authoring_version: 1,
spec_version: 1_016_004,
spec_version: 1_016_005,
impl_version: 0,
apis: RUNTIME_API_VERSIONS,
transaction_version: 16,
Expand Down
16 changes: 16 additions & 0 deletions prdoc/pr_6393.prdoc
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
title: '[pallet-revive] adjust fee dry-run calculation'
doc:
- audience: Runtime Dev
description: |-
- Fix bare_eth_transact so that it estimate more precisely the transaction fee
- Add some context to the build.rs to make it easier to troubleshoot errors
- Add TransactionBuilder for the RPC tests.
- Tweaked some error message, We will need to wait for the next subxt release to properly downcast some errors and
adopt MM error code (https://eips.ethereum.org/EIPS/eip-1474#error-codes)
crates:
- name: pallet-revive-eth-rpc
bump: minor
- name: pallet-revive
bump: minor
- name: pallet-revive-fixtures
bump: minor
15 changes: 10 additions & 5 deletions substrate/frame/revive/fixtures/build.rs
Original file line number Diff line number Diff line change
Expand Up @@ -106,7 +106,9 @@ fn create_cargo_toml<'a>(
);

let cargo_toml = toml::to_string_pretty(&cargo_toml)?;
fs::write(output_dir.join("Cargo.toml"), cargo_toml).map_err(Into::into)
fs::write(output_dir.join("Cargo.toml"), cargo_toml.clone())
.with_context(|| format!("Failed to write {cargo_toml:?}"))?;
Ok(())
}

fn invoke_build(target: &Path, current_dir: &Path) -> Result<()> {
Expand Down Expand Up @@ -154,10 +156,11 @@ fn post_process(input_path: &Path, output_path: &Path) -> Result<()> {
let mut config = polkavm_linker::Config::default();
config.set_strip(strip);
config.set_optimize(optimize);
let orig = fs::read(input_path).with_context(|| format!("Failed to read {:?}", input_path))?;
let orig = fs::read(input_path).with_context(|| format!("Failed to read {input_path:?}"))?;
let linked = polkavm_linker::program_from_elf(config, orig.as_ref())
.map_err(|err| anyhow::format_err!("Failed to link polkavm program: {}", err))?;
fs::write(output_path, linked).map_err(Into::into)
fs::write(output_path, linked).with_context(|| format!("Failed to write {output_path:?}"))?;
Ok(())
}

/// Write the compiled contracts to the given output directory.
Expand Down Expand Up @@ -209,9 +212,11 @@ pub fn main() -> Result<()> {
let symlink_dir: PathBuf = symlink_dir.into();
let symlink_dir: PathBuf = symlink_dir.join("target").join("pallet-revive-fixtures");
if symlink_dir.is_symlink() {
fs::remove_file(&symlink_dir)?
fs::remove_file(&symlink_dir)
.with_context(|| format!("Failed to remove_file {symlink_dir:?}"))?;
}
std::os::unix::fs::symlink(&out_dir, &symlink_dir)?;
std::os::unix::fs::symlink(&out_dir, &symlink_dir)
.with_context(|| format!("Failed to symlink {out_dir:?} -> {symlink_dir:?}"))?;
}

Ok(())
Expand Down
8 changes: 7 additions & 1 deletion substrate/frame/revive/fixtures/contracts/rpc_demo.rs
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@
#![no_std]
#![no_main]

use common::input;
use common::{input, u64_output};
use uapi::{HostFn, HostFnImpl as api};

#[no_mangle]
Expand All @@ -31,6 +31,12 @@ pub extern "C" fn deploy() {
#[no_mangle]
#[polkavm_derive::polkavm_export]
pub extern "C" fn call() {
// Not payable
let value = u64_output!(api::value_transferred,);
if value > 0 {
panic!();
}

input!(128, data: [u8],);
api::deposit_event(&[], data);
}
34 changes: 3 additions & 31 deletions substrate/frame/revive/rpc/examples/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ zombienet spawn --provider native westend_local_network.toml
This command starts the Ethereum JSON-RPC server, which runs on `localhost:8545` by default:

```bash
RUST_LOG="info,eth-rpc=debug" cargo run -p pallet-revive-eth-rpc --features dev
RUST_LOG="info,eth-rpc=debug" cargo run -p pallet-revive-eth-rpc -- --dev
```

## Rust examples
Expand Down Expand Up @@ -65,34 +65,6 @@ bun src/script.ts

### Configure MetaMask

You can use the following instructions to setup [MetaMask] with the local chain.
See the doc [here](https://contracts.polkadot.io/work-with-a-local-node#metemask-configuration) for more
information on how to configure MetaMask.

> **Note**: When you interact with MetaMask and restart the chain, you need to clear the activity tab (Settings >
Advanced > Clear activity tab data), and in some cases lock/unlock MetaMask to reset the nonce.
See [this guide][reset-account] for more info on how to reset the account activity.

#### Add a new network

To interact with the local chain, add a new network in [MetaMask].
See [this guide][add-network] for more info on how to add a custom network.

Make sure the node and the RPC server are started, and use the following settings to configure the network
(MetaMask > Networks > Add a network manually):

- Network name: KitchenSink
- RPC URL: <http://localhost:8545>
- Chain ID: 420420420
- Currency Symbol: `DEV`

#### Import Dev account

You will need to import the following account, endowed with some balance at genesis, to interact with the chain.
See [this guide][import-account] for more info on how to import an account.

- Account: `0xf24FF3a9CF04c71Dbc94D0b566f7A27B94566cac`
- Private Key: `5fb92d6e98884f76de468fa3f6278f8807c48bebc13595d45af5bdc4da702133`

[MetaMask]: https://metamask.io
[add-network]: https://support.metamask.io/networks-and-sidechains/managing-networks/how-to-add-a-custom-network-rpc/#adding-a-network-manually
[import-account]: https://support.metamask.io/managing-my-wallet/accounts-and-addresses/how-to-import-an-account/
[reset-account]: https://support.metamask.io/managing-my-wallet/resetting-deleting-and-restoring/how-to-clear-your-account-activity-reset-account
Binary file added substrate/frame/revive/rpc/examples/bun.lockb
Binary file not shown.
12 changes: 12 additions & 0 deletions substrate/frame/revive/rpc/examples/js/contracts/Revert.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

contract RevertExample {
constructor() {
}

function do_revert() public {
revert("revert message");
pgherveou marked this conversation as resolved.
Show resolved Hide resolved
}

}
2 changes: 1 addition & 1 deletion substrate/frame/revive/rpc/examples/js/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,6 @@
<button id="callButton">Call Contract</button>

<button id="deployAndCallButton">Deploy and Call Contract</button>
<script type="module" src="src/main.ts"></script>
<script type="module" src="src/web.ts"></script>
</body>
</html>
54 changes: 54 additions & 0 deletions substrate/frame/revive/rpc/examples/js/src/script-revert.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
//! Run with bun run script.ts

import { readFileSync } from "fs";
import { compile } from "@parity/revive";
import { Contract, ContractFactory, JsonRpcProvider, TransactionResponse } from "ethers";
import { InterfaceAbi } from "ethers";

const provider = new JsonRpcProvider("http://localhost:8545");
const signer = await provider.getSigner();
console.log(
`Signer address: ${await signer.getAddress()}, Nonce: ${await signer
.getNonce()}`,
);

// deploy
async function deploy(bytecode: string, abi: InterfaceAbi) {
console.log(`Deploying Contract...`);

const contractFactory = new ContractFactory(
abi,
bytecode,
signer,
);

console.log("Deploying contract");
const contract = await contractFactory.deploy();
await contract.waitForDeployment();
const address = await contract.getAddress();
console.log(`Contract deployed: ${address}`);
return address;
}

async function call(address: string, abi: InterfaceAbi) {
console.log(`Calling Contract at ${address}...`);
const contract = new Contract(address, abi, signer);
const tx = await contract.do_revert() as TransactionResponse;
console.log("Call transaction hash:", tx.hash);
tx.wait();
}

try {
const res = await compile({
["revert.sol"]: { content: readFileSync("./contracts/Revert.sol", "utf8")}
});

const out = res.contracts["revert.sol"]["RevertExample"];
const {abi, evm: {bytecode: {object: bytecode}}} = out

const address = await deploy(bytecode, abi);
await call(address, abi);
} catch(err) {
console.error(err);
}

14 changes: 12 additions & 2 deletions substrate/frame/revive/rpc/examples/js/src/script.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
//! Run with bun run script.ts

import { readFileSync } from "fs";
import { Contract, ContractFactory, JsonRpcProvider } from "ethers";
import { Contract, ContractFactory, JsonRpcProvider, parseEther, TransactionResponse } from "ethers";

const provider = new JsonRpcProvider("http://localhost:8545");
const signer = await provider.getSigner();
Expand Down Expand Up @@ -41,9 +41,19 @@ async function call(address: string) {

const abi = ["function call(bytes data)"];
const contract = new Contract(address, abi, signer);
const tx = await contract.call(str_to_bytes("world"));

const value = parseEther("10");
const tx = await contract.call(str_to_bytes("world"), { value }) as TransactionResponse;
console.log("Call transaction hash:", tx.hash);
tx.wait();
}

try {
const address = await deploy();
await call(address);

// check contract balance
console.log(`Address balance: ${await provider.getBalance(address)}`);
} catch(err) {
console.error(err);
}
1 change: 1 addition & 0 deletions substrate/frame/revive/rpc/examples/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
{ "dependencies": { "@parity/revive": "^0.0.5" } }
18 changes: 11 additions & 7 deletions substrate/frame/revive/rpc/examples/rust/deploy.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ use pallet_revive::{
evm::{Account, BlockTag, Bytes, ReceiptInfo, U256},
};
use pallet_revive_eth_rpc::{
example::{send_transaction, wait_for_receipt},
example::{wait_for_receipt, TransactionBuilder},
EthRpcClient,
};

Expand All @@ -41,9 +41,11 @@ async fn main() -> anyhow::Result<()> {
println!("\n\n=== Deploying contract ===\n\n");

let nonce = client.get_transaction_count(account.address(), BlockTag::Latest.into()).await?;
let hash =
send_transaction(&account, &client, 5_000_000_000_000u128.into(), input.into(), None)
.await?;
let hash = TransactionBuilder::new()
.value(5_000_000_000_000u128.into())
.input(input)
.send(&client)
.await?;

println!("Deploy Tx hash: {hash:?}");
let ReceiptInfo { block_number, gas_used, contract_address, .. } =
Expand All @@ -60,9 +62,11 @@ async fn main() -> anyhow::Result<()> {
println!("- Contract balance: {balance:?}");

println!("\n\n=== Calling contract ===\n\n");
let hash =
send_transaction(&account, &client, U256::zero(), Bytes::default(), Some(contract_address))
.await?;
let hash = TransactionBuilder::new()
.value(U256::from(1_000_000u32))
.to(contract_address)
.send(&client)
.await?;

println!("Contract call tx hash: {hash:?}");
let ReceiptInfo { block_number, gas_used, to, .. } = wait_for_receipt(&client, hash).await?;
Expand Down
20 changes: 13 additions & 7 deletions substrate/frame/revive/rpc/examples/rust/transfer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,23 +15,24 @@
// See the License for the specific language governing permissions and
// limitations under the License.
use jsonrpsee::http_client::HttpClientBuilder;
use pallet_revive::evm::{Account, BlockTag, Bytes, ReceiptInfo};
use pallet_revive::evm::{Account, BlockTag, ReceiptInfo};
use pallet_revive_eth_rpc::{
example::{send_transaction, wait_for_receipt},
example::{wait_for_receipt, TransactionBuilder},
EthRpcClient,
};

#[tokio::main]
async fn main() -> anyhow::Result<()> {
let alith = Account::default();
let client = HttpClientBuilder::default().build("http://localhost:8545")?;

let alith = Account::default();
let alith_address = alith.address();
let ethan = Account::from(subxt_signer::eth::dev::ethan());
let value = 1_000_000_000_000_000_000_000u128.into();

let print_balance = || async {
let balance = client.get_balance(alith.address(), BlockTag::Latest.into()).await?;
println!("Alith {:?} balance: {balance:?}", alith.address());
let balance = client.get_balance(alith_address, BlockTag::Latest.into()).await?;
println!("Alith {:?} balance: {balance:?}", alith_address);
let balance = client.get_balance(ethan.address(), BlockTag::Latest.into()).await?;
println!("ethan {:?} balance: {balance:?}", ethan.address());
anyhow::Result::<()>::Ok(())
Expand All @@ -40,8 +41,13 @@ async fn main() -> anyhow::Result<()> {
print_balance().await?;
println!("\n\n=== Transferring ===\n\n");

let hash =
send_transaction(&alith, &client, value, Bytes::default(), Some(ethan.address())).await?;
let hash = TransactionBuilder::new()
.signer(alith)
.value(value)
.to(ethan.address())
.mutate(|tx| tx.chain_id = Some(42u32.into()))
.send(&client)
.await?;
println!("Transaction hash: {hash:?}");

let ReceiptInfo { block_number, gas_used, status, .. } =
Expand Down
Binary file modified substrate/frame/revive/rpc/revive_chain.metadata
Binary file not shown.
Loading
Loading