mirror of
https://github.com/scroll-tech/scroll.git
synced 2026-01-14 08:28:02 -05:00
Co-authored-by: Péter Garamvölgyi <peter@scroll.io> Co-authored-by: HAOYUatHZ <37070449+HAOYUatHZ@users.noreply.github.com>
157 lines
4.9 KiB
Go
157 lines
4.9 KiB
Go
package utils
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/binary"
|
|
"errors"
|
|
"fmt"
|
|
"math/big"
|
|
|
|
"github.com/ethereum/go-ethereum/accounts/abi"
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
"github.com/ethereum/go-ethereum/crypto"
|
|
"github.com/ethereum/go-ethereum/ethclient"
|
|
|
|
backendabi "bridge-history-api/abi"
|
|
)
|
|
|
|
// Keccak2 compute the keccack256 of two concatenations of bytes32
|
|
func Keccak2(a common.Hash, b common.Hash) common.Hash {
|
|
return common.BytesToHash(crypto.Keccak256(append(a.Bytes()[:], b.Bytes()[:]...)))
|
|
}
|
|
|
|
func GetSafeBlockNumber(ctx context.Context, client *ethclient.Client, confirmations uint64) (uint64, error) {
|
|
number, err := client.BlockNumber(ctx)
|
|
if err != nil || number <= confirmations {
|
|
return 0, err
|
|
}
|
|
number = number - confirmations
|
|
return number, nil
|
|
}
|
|
|
|
// UnpackLog unpacks a retrieved log into the provided output structure.
|
|
// @todo: add unit test.
|
|
func UnpackLog(c *abi.ABI, out interface{}, event string, log types.Log) error {
|
|
if log.Topics[0] != c.Events[event].ID {
|
|
return fmt.Errorf("event signature mismatch")
|
|
}
|
|
if len(log.Data) > 0 {
|
|
if err := c.UnpackIntoInterface(out, event, log.Data); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
var indexed abi.Arguments
|
|
for _, arg := range c.Events[event].Inputs {
|
|
if arg.Indexed {
|
|
indexed = append(indexed, arg)
|
|
}
|
|
}
|
|
return abi.ParseTopics(out, indexed, log.Topics[1:])
|
|
}
|
|
|
|
// ComputeMessageHash compute the message hash
|
|
func ComputeMessageHash(
|
|
sender common.Address,
|
|
target common.Address,
|
|
value *big.Int,
|
|
messageNonce *big.Int,
|
|
message []byte,
|
|
) common.Hash {
|
|
data, _ := backendabi.L2ScrollMessengerABI.Pack("relayMessage", sender, target, value, messageNonce, message)
|
|
return common.BytesToHash(crypto.Keccak256(data))
|
|
}
|
|
|
|
type commitBatchArgs struct {
|
|
Version uint8
|
|
ParentBatchHeader []byte
|
|
Chunks [][]byte
|
|
SkippedL1MessageBitmap []byte
|
|
}
|
|
|
|
// GetBatchRangeFromCalldataV2 find the block range from calldata, both inclusive.
|
|
func GetBatchRangeFromCalldataV2(calldata []byte) (uint64, uint64, uint64, error) {
|
|
method := backendabi.ScrollChainV2ABI.Methods["commitBatch"]
|
|
values, err := method.Inputs.Unpack(calldata[4:])
|
|
if err != nil {
|
|
return 0, 0, 0, err
|
|
}
|
|
args := commitBatchArgs{}
|
|
err = method.Inputs.Copy(&args, values)
|
|
if err != nil {
|
|
return 0, 0, 0, err
|
|
}
|
|
|
|
var startBlock uint64
|
|
var finishBlock uint64
|
|
|
|
// decode batchIndex from ParentBatchHeader
|
|
if len(args.ParentBatchHeader) < 9 {
|
|
return 0, 0, 0, errors.New("invalid parent batch header")
|
|
}
|
|
batchIndex := binary.BigEndian.Uint64(args.ParentBatchHeader[1:9]) + 1
|
|
|
|
// decode blocks from chunk and assume that there's no empty chunk
|
|
// | 1 byte | 60 bytes | ... | 60 bytes |
|
|
// | num blocks | block 1 | ... | block n |
|
|
if len(args.Chunks) == 0 {
|
|
return 0, 0, 0, errors.New("invalid chunks")
|
|
}
|
|
chunk := args.Chunks[0]
|
|
block := chunk[1:61] // first block in chunk
|
|
startBlock = binary.BigEndian.Uint64(block[0:8])
|
|
|
|
chunk = args.Chunks[len(args.Chunks)-1]
|
|
lastBlockIndex := int(chunk[0]) - 1
|
|
block = chunk[1+lastBlockIndex*60 : 1+lastBlockIndex*60+60] // last block in chunk
|
|
finishBlock = binary.BigEndian.Uint64(block[0:8])
|
|
|
|
return batchIndex, startBlock, finishBlock, err
|
|
}
|
|
|
|
// GetBatchRangeFromCalldataV1 find the block range from calldata, both inclusive.
|
|
func GetBatchRangeFromCalldataV1(calldata []byte) ([]uint64, []uint64, []uint64, error) {
|
|
var batchIndices []uint64
|
|
var startBlocks []uint64
|
|
var finishBlocks []uint64
|
|
if bytes.Equal(calldata[0:4], common.Hex2Bytes("cb905499")) {
|
|
// commitBatches
|
|
method := backendabi.ScrollChainABI.Methods["commitBatches"]
|
|
values, err := method.Inputs.Unpack(calldata[4:])
|
|
if err != nil {
|
|
return batchIndices, startBlocks, finishBlocks, err
|
|
}
|
|
args := make([]backendabi.IScrollChainBatch, len(values))
|
|
err = method.Inputs.Copy(&args, values)
|
|
if err != nil {
|
|
return batchIndices, startBlocks, finishBlocks, err
|
|
}
|
|
|
|
for i := 0; i < len(args); i++ {
|
|
batchIndices = append(batchIndices, args[i].BatchIndex)
|
|
startBlocks = append(startBlocks, args[i].Blocks[0].BlockNumber)
|
|
finishBlocks = append(finishBlocks, args[i].Blocks[len(args[i].Blocks)-1].BlockNumber)
|
|
}
|
|
} else if bytes.Equal(calldata[0:4], common.Hex2Bytes("8c73235d")) {
|
|
// commitBatch
|
|
method := backendabi.ScrollChainABI.Methods["commitBatch"]
|
|
values, err := method.Inputs.Unpack(calldata[4:])
|
|
if err != nil {
|
|
return batchIndices, startBlocks, finishBlocks, err
|
|
}
|
|
|
|
args := backendabi.IScrollChainBatch{}
|
|
err = method.Inputs.Copy(&args, values)
|
|
if err != nil {
|
|
return batchIndices, startBlocks, finishBlocks, err
|
|
}
|
|
batchIndices = append(batchIndices, args.BatchIndex)
|
|
startBlocks = append(startBlocks, args.Blocks[0].BlockNumber)
|
|
finishBlocks = append(finishBlocks, args.Blocks[len(args.Blocks)-1].BlockNumber)
|
|
} else {
|
|
return batchIndices, startBlocks, finishBlocks, errors.New("invalid selector")
|
|
}
|
|
return batchIndices, startBlocks, finishBlocks, nil
|
|
}
|