cff785fa37
* chore: eth: move & rename input/output encoding functions These are shared functions, so I'm moving them to the utils library. * fix: eth: correctly encode and simplify native input/output encoding When generating eth traces, we encode "native" message inputs/outputs to "solidity ABI" by formatting the inputs/outputs the same way we do in FEVM's "handle_native_method". However, we had quite a few bugs with the implementation: 1. We were right-aligning 64bit values in 256bit words, instead of left-aligning (as we should given that these values are big-endian). 2. The return-value encoding wasn't correctly handling lengths. This patch: 1. Fixes those bugs. 2. Deduplicates the logic (we're doing _basically_ the same thing in both cases). 3. Removes all error paths (these functions can't fail).
180 lines
4.9 KiB
Go
180 lines
4.9 KiB
Go
package full
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"testing"
|
|
|
|
"github.com/ipfs/go-cid"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/filecoin-project/go-state-types/big"
|
|
|
|
"github.com/filecoin-project/lotus/chain/types"
|
|
"github.com/filecoin-project/lotus/chain/types/ethtypes"
|
|
)
|
|
|
|
func TestEthLogFromEvent(t *testing.T) {
|
|
// basic empty
|
|
data, topics, ok := ethLogFromEvent(nil)
|
|
require.True(t, ok)
|
|
require.Nil(t, data)
|
|
require.Empty(t, topics)
|
|
require.NotNil(t, topics)
|
|
|
|
// basic topic
|
|
data, topics, ok = ethLogFromEvent([]types.EventEntry{{
|
|
Flags: 0,
|
|
Key: "t1",
|
|
Codec: cid.Raw,
|
|
Value: make([]byte, 32),
|
|
}})
|
|
require.True(t, ok)
|
|
require.Nil(t, data)
|
|
require.Len(t, topics, 1)
|
|
require.Equal(t, topics[0], ethtypes.EthHash{})
|
|
|
|
// basic topic with data
|
|
data, topics, ok = ethLogFromEvent([]types.EventEntry{{
|
|
Flags: 0,
|
|
Key: "t1",
|
|
Codec: cid.Raw,
|
|
Value: make([]byte, 32),
|
|
}, {
|
|
Flags: 0,
|
|
Key: "d",
|
|
Codec: cid.Raw,
|
|
Value: []byte{0x0},
|
|
}})
|
|
require.True(t, ok)
|
|
require.Equal(t, data, []byte{0x0})
|
|
require.Len(t, topics, 1)
|
|
require.Equal(t, topics[0], ethtypes.EthHash{})
|
|
|
|
// skip topic
|
|
_, _, ok = ethLogFromEvent([]types.EventEntry{{
|
|
Flags: 0,
|
|
Key: "t2",
|
|
Codec: cid.Raw,
|
|
Value: make([]byte, 32),
|
|
}})
|
|
require.False(t, ok)
|
|
|
|
// duplicate topic
|
|
_, _, ok = ethLogFromEvent([]types.EventEntry{{
|
|
Flags: 0,
|
|
Key: "t1",
|
|
Codec: cid.Raw,
|
|
Value: make([]byte, 32),
|
|
}, {
|
|
Flags: 0,
|
|
Key: "t1",
|
|
Codec: cid.Raw,
|
|
Value: make([]byte, 32),
|
|
}})
|
|
require.False(t, ok)
|
|
|
|
// duplicate data
|
|
_, _, ok = ethLogFromEvent([]types.EventEntry{{
|
|
Flags: 0,
|
|
Key: "d",
|
|
Codec: cid.Raw,
|
|
Value: make([]byte, 32),
|
|
}, {
|
|
Flags: 0,
|
|
Key: "d",
|
|
Codec: cid.Raw,
|
|
Value: make([]byte, 32),
|
|
}})
|
|
require.False(t, ok)
|
|
|
|
// unknown key is fine
|
|
data, topics, ok = ethLogFromEvent([]types.EventEntry{{
|
|
Flags: 0,
|
|
Key: "t5",
|
|
Codec: cid.Raw,
|
|
Value: make([]byte, 32),
|
|
}, {
|
|
Flags: 0,
|
|
Key: "t1",
|
|
Codec: cid.Raw,
|
|
Value: make([]byte, 32),
|
|
}})
|
|
require.True(t, ok)
|
|
require.Nil(t, data)
|
|
require.Len(t, topics, 1)
|
|
require.Equal(t, topics[0], ethtypes.EthHash{})
|
|
}
|
|
|
|
func TestReward(t *testing.T) {
|
|
baseFee := big.NewInt(100)
|
|
testcases := []struct {
|
|
maxFeePerGas, maxPriorityFeePerGas big.Int
|
|
answer big.Int
|
|
}{
|
|
{maxFeePerGas: big.NewInt(600), maxPriorityFeePerGas: big.NewInt(200), answer: big.NewInt(200)},
|
|
{maxFeePerGas: big.NewInt(600), maxPriorityFeePerGas: big.NewInt(300), answer: big.NewInt(300)},
|
|
{maxFeePerGas: big.NewInt(600), maxPriorityFeePerGas: big.NewInt(500), answer: big.NewInt(500)},
|
|
{maxFeePerGas: big.NewInt(600), maxPriorityFeePerGas: big.NewInt(600), answer: big.NewInt(500)},
|
|
{maxFeePerGas: big.NewInt(600), maxPriorityFeePerGas: big.NewInt(1000), answer: big.NewInt(500)},
|
|
{maxFeePerGas: big.NewInt(50), maxPriorityFeePerGas: big.NewInt(200), answer: big.NewInt(0)},
|
|
}
|
|
for _, tc := range testcases {
|
|
msg := &types.Message{GasFeeCap: tc.maxFeePerGas, GasPremium: tc.maxPriorityFeePerGas}
|
|
reward := msg.EffectiveGasPremium(baseFee)
|
|
require.Equal(t, 0, reward.Int.Cmp(tc.answer.Int), reward, tc.answer)
|
|
}
|
|
}
|
|
|
|
func TestRewardPercentiles(t *testing.T) {
|
|
testcases := []struct {
|
|
percentiles []float64
|
|
txGasRewards gasRewardSorter
|
|
answer []int64
|
|
}{
|
|
{
|
|
percentiles: []float64{25, 50, 75},
|
|
txGasRewards: []gasRewardTuple{},
|
|
answer: []int64{MinGasPremium, MinGasPremium, MinGasPremium},
|
|
},
|
|
{
|
|
percentiles: []float64{25, 50, 75, 100},
|
|
txGasRewards: []gasRewardTuple{
|
|
{gasUsed: int64(0), premium: big.NewInt(300)},
|
|
{gasUsed: int64(100), premium: big.NewInt(200)},
|
|
{gasUsed: int64(350), premium: big.NewInt(100)},
|
|
{gasUsed: int64(500), premium: big.NewInt(600)},
|
|
{gasUsed: int64(300), premium: big.NewInt(700)},
|
|
},
|
|
answer: []int64{200, 700, 700, 700},
|
|
},
|
|
}
|
|
for _, tc := range testcases {
|
|
rewards, totalGasUsed := calculateRewardsAndGasUsed(tc.percentiles, tc.txGasRewards)
|
|
var gasUsed int64
|
|
for _, tx := range tc.txGasRewards {
|
|
gasUsed += tx.gasUsed
|
|
}
|
|
ans := []ethtypes.EthBigInt{}
|
|
for _, bi := range tc.answer {
|
|
ans = append(ans, ethtypes.EthBigInt(big.NewInt(bi)))
|
|
}
|
|
require.Equal(t, totalGasUsed, gasUsed)
|
|
require.Equal(t, len(ans), len(tc.percentiles))
|
|
require.Equal(t, ans, rewards)
|
|
}
|
|
}
|
|
|
|
func TestABIEncoding(t *testing.T) {
|
|
// Generated from https://abi.hashex.org/
|
|
const expected = "000000000000000000000000000000000000000000000000000000000000001600000000000000000000000000000000000000000000000000000000000000510000000000000000000000000000000000000000000000000000000000000060000000000000000000000000000000000000000000000000000000000000001b1111111111111111111020200301000000044444444444444444010000000000"
|
|
const data = "111111111111111111102020030100000004444444444444444401"
|
|
|
|
expectedBytes, err := hex.DecodeString(expected)
|
|
require.NoError(t, err)
|
|
|
|
dataBytes, err := hex.DecodeString(data)
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, expectedBytes, encodeAsABIHelper(22, 81, dataBytes))
|
|
}
|