mirror of https://github.com/status-im/nim-eth.git
300 lines
8.9 KiB
Nim
300 lines
8.9 KiB
Nim
# Nimbus
|
|
# Copyright (c) 2023-2024 Status Research & Development GmbH
|
|
# Licensed under either of
|
|
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE) or
|
|
# http://www.apache.org/licenses/LICENSE-2.0)
|
|
# * MIT license ([LICENSE-MIT](LICENSE-MIT) or
|
|
# http://opensource.org/licenses/MIT)
|
|
# at your option. This file may not be copied, modified, or distributed except
|
|
# according to those terms.
|
|
{.used.}
|
|
|
|
import
|
|
stew/byteutils,
|
|
unittest2,
|
|
../../eth/common/[transactions_rlp, transaction_utils]
|
|
|
|
const
|
|
recipient = address"095e7baea6a6c7c4c2dfeb977efac326af552d87"
|
|
zeroG1 = bytes48"0xc00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"
|
|
source = address"0x0000000000000000000000000000000000000001"
|
|
storageKey= default(Bytes32)
|
|
accesses = @[AccessPair(address: source, storageKeys: @[storageKey])]
|
|
blob = default(NetworkBlob)
|
|
abcdef = hexToSeqByte("abcdef")
|
|
authList = @[Authorization(
|
|
chainID: 1.ChainId,
|
|
address: source,
|
|
nonce: 2.AccountNonce,
|
|
v: 3,
|
|
r: 4.u256,
|
|
s: 5.u256
|
|
)]
|
|
|
|
proc tx0(i: int): PooledTransaction =
|
|
PooledTransaction(
|
|
tx: Transaction(
|
|
txType: TxLegacy,
|
|
nonce: i.AccountNonce,
|
|
to: Opt.some recipient,
|
|
gasLimit: 1.GasInt,
|
|
gasPrice: 2.GasInt,
|
|
payload: abcdef))
|
|
|
|
proc tx1(i: int): PooledTransaction =
|
|
PooledTransaction(
|
|
tx: Transaction(
|
|
# Legacy tx contract creation.
|
|
txType: TxLegacy,
|
|
nonce: i.AccountNonce,
|
|
gasLimit: 1.GasInt,
|
|
gasPrice: 2.GasInt,
|
|
payload: abcdef))
|
|
|
|
proc tx2(i: int): PooledTransaction =
|
|
PooledTransaction(
|
|
tx: Transaction(
|
|
# Tx with non-zero access list.
|
|
txType: TxEip2930,
|
|
chainId: 1.ChainId,
|
|
nonce: i.AccountNonce,
|
|
to: Opt.some recipient,
|
|
gasLimit: 123457.GasInt,
|
|
gasPrice: 10.GasInt,
|
|
accessList: accesses,
|
|
payload: abcdef))
|
|
|
|
proc tx3(i: int): PooledTransaction =
|
|
PooledTransaction(
|
|
tx: Transaction(
|
|
# Tx with empty access list.
|
|
txType: TxEip2930,
|
|
chainId: 1.ChainId,
|
|
nonce: i.AccountNonce,
|
|
to: Opt.some recipient,
|
|
gasLimit: 123457.GasInt,
|
|
gasPrice: 10.GasInt,
|
|
payload: abcdef))
|
|
|
|
proc tx4(i: int): PooledTransaction =
|
|
PooledTransaction(
|
|
tx: Transaction(
|
|
# Contract creation with access list.
|
|
txType: TxEip2930,
|
|
chainId: 1.ChainId,
|
|
nonce: i.AccountNonce,
|
|
gasLimit: 123457.GasInt,
|
|
gasPrice: 10.GasInt,
|
|
accessList: accesses))
|
|
|
|
proc tx5(i: int): PooledTransaction =
|
|
PooledTransaction(
|
|
tx: Transaction(
|
|
txType: TxEip1559,
|
|
chainId: 1.ChainId,
|
|
nonce: i.AccountNonce,
|
|
gasLimit: 123457.GasInt,
|
|
maxPriorityFeePerGas: 42.GasInt,
|
|
maxFeePerGas: 10.GasInt,
|
|
accessList: accesses))
|
|
|
|
proc tx6(i: int): PooledTransaction =
|
|
const
|
|
digest = hash32"010657f37554c781402a22917dee2f75def7ab966d7b770905398eba3c444014"
|
|
|
|
PooledTransaction(
|
|
tx: Transaction(
|
|
txType: TxEip4844,
|
|
chainId: 1.ChainId,
|
|
nonce: i.AccountNonce,
|
|
gasLimit: 123457.GasInt,
|
|
maxPriorityFeePerGas:42.GasInt,
|
|
maxFeePerGas: 10.GasInt,
|
|
accessList: accesses,
|
|
versionedHashes: @[digest]),
|
|
networkPayload: NetworkPayload(
|
|
blobs: @[blob],
|
|
commitments: @[zeroG1],
|
|
proofs: @[zeroG1]))
|
|
|
|
proc tx7(i: int): PooledTransaction =
|
|
const
|
|
digest = hash32"01624652859a6e98ffc1608e2af0147ca4e86e1ce27672d8d3f3c9d4ffd6ef7e"
|
|
|
|
PooledTransaction(
|
|
tx: Transaction(
|
|
txType: TxEip4844,
|
|
chainID: 1.ChainId,
|
|
nonce: i.AccountNonce,
|
|
gasLimit: 123457.GasInt,
|
|
maxPriorityFeePerGas:42.GasInt,
|
|
maxFeePerGas: 10.GasInt,
|
|
accessList: accesses,
|
|
versionedHashes: @[digest],
|
|
maxFeePerBlobGas: 10000000.u256))
|
|
|
|
proc tx8(i: int): PooledTransaction =
|
|
const
|
|
digest = hash32"01624652859a6e98ffc1608e2af0147ca4e86e1ce27672d8d3f3c9d4ffd6ef7e"
|
|
|
|
PooledTransaction(
|
|
tx: Transaction(
|
|
txType: TxEip4844,
|
|
chainID: 1.ChainId,
|
|
nonce: i.AccountNonce,
|
|
to: Opt.some(recipient),
|
|
gasLimit: 123457.GasInt,
|
|
maxPriorityFeePerGas:42.GasInt,
|
|
maxFeePerGas: 10.GasInt,
|
|
accessList: accesses,
|
|
versionedHashes: @[digest],
|
|
maxFeePerBlobGas: 10000000.u256))
|
|
|
|
proc txEip7702(i: int): PooledTransaction =
|
|
PooledTransaction(
|
|
tx: Transaction(
|
|
txType: TxEip7702,
|
|
chainId: 1.ChainId,
|
|
nonce: i.AccountNonce,
|
|
maxPriorityFeePerGas: 2.GasInt,
|
|
maxFeePerGas: 3.GasInt,
|
|
gasLimit: 4.GasInt,
|
|
to: Opt.some recipient,
|
|
value: 5.u256,
|
|
payload: abcdef,
|
|
accessList: accesses,
|
|
authorizationList: authList
|
|
)
|
|
)
|
|
|
|
template roundTrip(txFunc: untyped, i: int) =
|
|
let tx = txFunc(i)
|
|
let bytes = rlp.encode(tx)
|
|
let tx2 = rlp.decode(bytes, PooledTransaction)
|
|
let bytes2 = rlp.encode(tx2)
|
|
check bytes == bytes2
|
|
|
|
suite "Transactions":
|
|
test "Legacy Tx Call":
|
|
roundTrip(tx0, 1)
|
|
|
|
test "Legacy tx contract creation":
|
|
roundTrip(tx1, 2)
|
|
|
|
test "Tx with non-zero access list":
|
|
roundTrip(tx2, 3)
|
|
|
|
test "Tx with empty access list":
|
|
roundTrip(tx3, 4)
|
|
|
|
test "Contract creation with access list":
|
|
roundTrip(tx4, 5)
|
|
|
|
test "Dynamic Fee Tx":
|
|
roundTrip(tx5, 6)
|
|
|
|
test "NetworkBlob Tx":
|
|
roundTrip(tx6, 7)
|
|
|
|
test "Minimal Blob Tx":
|
|
roundTrip(tx7, 8)
|
|
|
|
test "Minimal Blob Tx contract creation":
|
|
roundTrip(tx8, 9)
|
|
|
|
test "EIP 7702":
|
|
roundTrip(txEip7702, 9)
|
|
|
|
test "Network payload survive encode decode":
|
|
let tx = tx6(10)
|
|
let bytes = rlp.encode(tx)
|
|
let zz = rlp.decode(bytes, PooledTransaction)
|
|
check not zz.networkPayload.isNil
|
|
check zz.networkPayload.proofs == tx.networkPayload.proofs
|
|
check zz.networkPayload.blobs == tx.networkPayload.blobs
|
|
check zz.networkPayload.commitments == tx.networkPayload.commitments
|
|
|
|
test "No Network payload still no network payload":
|
|
let tx = tx7(11)
|
|
let bytes = rlp.encode(tx)
|
|
let zz = rlp.decode(bytes, PooledTransaction)
|
|
check zz.networkPayload.isNil
|
|
|
|
test "Minimal Blob tx recipient survive encode decode":
|
|
let tx = tx8(12)
|
|
let bytes = rlp.encode(tx)
|
|
let zz = rlp.decode(bytes, PooledTransaction)
|
|
check zz.tx.to.isSome
|
|
|
|
test "Tx List 0,1,2,3,4,5,6,7,8":
|
|
let txs = @[tx0(3), tx1(3), tx2(3), tx3(3), tx4(3),
|
|
tx5(3), tx6(3), tx7(3), tx8(3)]
|
|
|
|
let bytes = rlp.encode(txs)
|
|
let zz = rlp.decode(bytes, seq[PooledTransaction])
|
|
let bytes2 = rlp.encode(zz)
|
|
check bytes2 == bytes
|
|
|
|
test "Tx List 8,7,6,5,4,3,2,1,0":
|
|
let txs = @[tx8(3), tx7(3) , tx6(3), tx5(3), tx4(3),
|
|
tx3(3), tx2(3), tx1(3), tx0(3)]
|
|
|
|
let bytes = rlp.encode(txs)
|
|
let zz = rlp.decode(bytes, seq[PooledTransaction])
|
|
let bytes2 = rlp.encode(zz)
|
|
check bytes2 == bytes
|
|
|
|
test "Tx List 0,5,8,7,6,4,3,2,1":
|
|
let txs = @[tx0(3), tx5(3), tx8(3), tx7(3), tx6(3),
|
|
tx4(3), tx3(3), tx2(3), tx1(3)]
|
|
|
|
let bytes = rlp.encode(txs)
|
|
let zz = rlp.decode(bytes, seq[PooledTransaction])
|
|
let bytes2 = rlp.encode(zz)
|
|
check bytes2 == bytes
|
|
|
|
test "EIP-155 signature":
|
|
# https://github.com/ethereum/EIPs/blob/master/EIPS/eip-155.md#example
|
|
var
|
|
tx = Transaction(
|
|
txType: TxLegacy,
|
|
chainId: ChainId(1),
|
|
nonce: 9,
|
|
gasPrice: 20000000000'u64,
|
|
gasLimit: 21000'u64,
|
|
to: Opt.some address"0x3535353535353535353535353535353535353535",
|
|
value: u256"1000000000000000000",
|
|
)
|
|
txEnc = tx.encodeForSigning(true)
|
|
txHash = tx.rlpHashForSigning(true)
|
|
key = PrivateKey.fromHex("0x4646464646464646464646464646464646464646464646464646464646464646").expect(
|
|
"working key"
|
|
)
|
|
|
|
tx.signature = tx.sign(key, true)
|
|
|
|
check:
|
|
txEnc.to0xHex == "0xec098504a817c800825208943535353535353535353535353535353535353535880de0b6b3a764000080018080"
|
|
txHash == hash32"0xdaf5a779ae972f972197303d7b574746c7ef83eadac0f2791ad23db92e4c8e53"
|
|
tx.V == 37
|
|
tx.R ==
|
|
u256"18515461264373351373200002665853028612451056578545711640558177340181847433846"
|
|
tx.S ==
|
|
u256"46948507304638947509940763649030358759909902576025900602547168820602576006531"
|
|
|
|
test "sign transaction":
|
|
let
|
|
txs = @[
|
|
tx0(3).tx, tx1(3).tx, tx2(3).tx, tx3(3).tx, tx4(3).tx,
|
|
tx5(3).tx, tx6(3).tx, tx7(3).tx, tx8(3).tx, txEip7702(3).tx]
|
|
|
|
privKey = PrivateKey.fromHex("63b508a03c3b5937ceb903af8b1b0c191012ef6eb7e9c3fb7afa94e5d214d376").expect("valid key")
|
|
sender = privKey.toPublicKey().to(Address)
|
|
|
|
for tx in txs:
|
|
var tx = tx
|
|
tx.signature = tx.sign(privKey, true)
|
|
|
|
check:
|
|
tx.recoverKey().expect("valid key").to(Address) == sender
|