eth/catalyst: evict old payloads, type PayloadID (#24236)
* eth/catalyst: evict old payloads, type PayloadID * eth/catalyst: added tracing info to engine api * eth/catalyst: add test for create payload timestamps * catalyst: better logs * eth/catalyst: computePayloadId return style * catalyst: add queue for payloads * eth/catalyst: nitpicks Co-authored-by: Marius van der Wijden <m.vanderwijden@live.de> Co-authored-by: Péter Szilágyi <peterke@gmail.com>
This commit is contained in:
		
				
					committed by
					
						 GitHub
						GitHub
					
				
			
			
				
	
			
			
			
						parent
						
							03aaea11d1
						
					
				
				
					commit
					514ae7cfa3
				
			| @@ -26,7 +26,6 @@ import ( | |||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/ethereum/go-ethereum/common" | 	"github.com/ethereum/go-ethereum/common" | ||||||
| 	"github.com/ethereum/go-ethereum/common/hexutil" |  | ||||||
| 	"github.com/ethereum/go-ethereum/consensus" | 	"github.com/ethereum/go-ethereum/consensus" | ||||||
| 	"github.com/ethereum/go-ethereum/consensus/beacon" | 	"github.com/ethereum/go-ethereum/consensus/beacon" | ||||||
| 	"github.com/ethereum/go-ethereum/consensus/misc" | 	"github.com/ethereum/go-ethereum/consensus/misc" | ||||||
| @@ -50,7 +49,6 @@ var ( | |||||||
| 	GenericServerError = rpc.CustomError{Code: -32000, ValidationError: "Server error"} | 	GenericServerError = rpc.CustomError{Code: -32000, ValidationError: "Server error"} | ||||||
| 	UnknownPayload     = rpc.CustomError{Code: -32001, ValidationError: "Unknown payload"} | 	UnknownPayload     = rpc.CustomError{Code: -32001, ValidationError: "Unknown payload"} | ||||||
| 	InvalidTB          = rpc.CustomError{Code: -32002, ValidationError: "Invalid terminal block"} | 	InvalidTB          = rpc.CustomError{Code: -32002, ValidationError: "Invalid terminal block"} | ||||||
| 	InvalidPayloadID   = rpc.CustomError{Code: 1, ValidationError: "invalid payload id"} |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Register adds catalyst APIs to the full node. | // Register adds catalyst APIs to the full node. | ||||||
| @@ -86,7 +84,7 @@ type ConsensusAPI struct { | |||||||
| 	eth            *eth.Ethereum | 	eth            *eth.Ethereum | ||||||
| 	les            *les.LightEthereum | 	les            *les.LightEthereum | ||||||
| 	engine         consensus.Engine // engine is the post-merge consensus engine, only for block creation | 	engine         consensus.Engine // engine is the post-merge consensus engine, only for block creation | ||||||
| 	preparedBlocks map[uint64]*ExecutableDataV1 | 	preparedBlocks *payloadQueue    // preparedBlocks caches payloads (*ExecutableDataV1) by payload ID (PayloadID) | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewConsensusAPI(eth *eth.Ethereum, les *les.LightEthereum) *ConsensusAPI { | func NewConsensusAPI(eth *eth.Ethereum, les *les.LightEthereum) *ConsensusAPI { | ||||||
| @@ -110,12 +108,13 @@ func NewConsensusAPI(eth *eth.Ethereum, les *les.LightEthereum) *ConsensusAPI { | |||||||
| 			engine = beacon.New(eth.Engine()) | 			engine = beacon.New(eth.Engine()) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return &ConsensusAPI{ | 	return &ConsensusAPI{ | ||||||
| 		light:          eth == nil, | 		light:          eth == nil, | ||||||
| 		eth:            eth, | 		eth:            eth, | ||||||
| 		les:            les, | 		les:            les, | ||||||
| 		engine:         engine, | 		engine:         engine, | ||||||
| 		preparedBlocks: make(map[uint64]*ExecutableDataV1), | 		preparedBlocks: newPayloadQueue(), | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -175,20 +174,17 @@ func (api *ConsensusAPI) makeEnv(parent *types.Block, header *types.Header) (*bl | |||||||
| 	return env, nil | 	return env, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (api *ConsensusAPI) GetPayloadV1(payloadID hexutil.Bytes) (*ExecutableDataV1, error) { | func (api *ConsensusAPI) GetPayloadV1(payloadID PayloadID) (*ExecutableDataV1, error) { | ||||||
| 	hash := []byte(payloadID) | 	log.Trace("Engine API request received", "method", "GetPayload", "id", payloadID) | ||||||
| 	if len(hash) < 8 { | 	data := api.preparedBlocks.get(payloadID) | ||||||
| 		return nil, &InvalidPayloadID | 	if data == nil { | ||||||
| 	} |  | ||||||
| 	id := binary.BigEndian.Uint64(hash[:8]) |  | ||||||
| 	data, ok := api.preparedBlocks[id] |  | ||||||
| 	if !ok { |  | ||||||
| 		return nil, &UnknownPayload | 		return nil, &UnknownPayload | ||||||
| 	} | 	} | ||||||
| 	return data, nil | 	return data, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (api *ConsensusAPI) ForkchoiceUpdatedV1(heads ForkchoiceStateV1, PayloadAttributes *PayloadAttributesV1) (ForkChoiceResponse, error) { | func (api *ConsensusAPI) ForkchoiceUpdatedV1(heads ForkchoiceStateV1, PayloadAttributes *PayloadAttributesV1) (ForkChoiceResponse, error) { | ||||||
|  | 	log.Trace("Engine API request received", "method", "ForkChoiceUpdated", "head", heads.HeadBlockHash, "finalized", heads.FinalizedBlockHash, "safe", heads.SafeBlockHash) | ||||||
| 	if heads.HeadBlockHash == (common.Hash{}) { | 	if heads.HeadBlockHash == (common.Hash{}) { | ||||||
| 		return ForkChoiceResponse{Status: SUCCESS.Status, PayloadID: nil}, nil | 		return ForkChoiceResponse{Status: SUCCESS.Status, PayloadID: nil}, nil | ||||||
| 	} | 	} | ||||||
| @@ -216,25 +212,24 @@ func (api *ConsensusAPI) ForkchoiceUpdatedV1(heads ForkchoiceStateV1, PayloadAtt | |||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return INVALID, err | 			return INVALID, err | ||||||
| 		} | 		} | ||||||
| 		hash := computePayloadId(heads.HeadBlockHash, PayloadAttributes) | 		id := computePayloadId(heads.HeadBlockHash, PayloadAttributes) | ||||||
| 		id := binary.BigEndian.Uint64(hash) | 		api.preparedBlocks.put(id, data) | ||||||
| 		api.preparedBlocks[id] = data | 		log.Info("Created payload", "payloadID", id) | ||||||
| 		log.Info("Created payload", "payloadid", id) | 		return ForkChoiceResponse{Status: SUCCESS.Status, PayloadID: &id}, nil | ||||||
| 		// TODO (MariusVanDerWijden) do something with the payloadID? |  | ||||||
| 		hex := hexutil.Bytes(hash) |  | ||||||
| 		return ForkChoiceResponse{Status: SUCCESS.Status, PayloadID: &hex}, nil |  | ||||||
| 	} | 	} | ||||||
| 	return ForkChoiceResponse{Status: SUCCESS.Status, PayloadID: nil}, nil | 	return ForkChoiceResponse{Status: SUCCESS.Status, PayloadID: nil}, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func computePayloadId(headBlockHash common.Hash, params *PayloadAttributesV1) []byte { | func computePayloadId(headBlockHash common.Hash, params *PayloadAttributesV1) PayloadID { | ||||||
| 	// Hash | 	// Hash | ||||||
| 	hasher := sha256.New() | 	hasher := sha256.New() | ||||||
| 	hasher.Write(headBlockHash[:]) | 	hasher.Write(headBlockHash[:]) | ||||||
| 	binary.Write(hasher, binary.BigEndian, params.Timestamp) | 	binary.Write(hasher, binary.BigEndian, params.Timestamp) | ||||||
| 	hasher.Write(params.Random[:]) | 	hasher.Write(params.Random[:]) | ||||||
| 	hasher.Write(params.SuggestedFeeRecipient[:]) | 	hasher.Write(params.SuggestedFeeRecipient[:]) | ||||||
| 	return hasher.Sum([]byte{})[:8] | 	var out PayloadID | ||||||
|  | 	copy(out[:], hasher.Sum(nil)[:8]) | ||||||
|  | 	return out | ||||||
| } | } | ||||||
|  |  | ||||||
| func (api *ConsensusAPI) invalid() ExecutePayloadResponse { | func (api *ConsensusAPI) invalid() ExecutePayloadResponse { | ||||||
| @@ -244,8 +239,9 @@ func (api *ConsensusAPI) invalid() ExecutePayloadResponse { | |||||||
| 	return ExecutePayloadResponse{Status: INVALID.Status, LatestValidHash: api.eth.BlockChain().CurrentHeader().Hash()} | 	return ExecutePayloadResponse{Status: INVALID.Status, LatestValidHash: api.eth.BlockChain().CurrentHeader().Hash()} | ||||||
| } | } | ||||||
|  |  | ||||||
| // ExecutePayload creates an Eth1 block, inserts it in the chain, and returns the status of the chain. | // ExecutePayloadV1 creates an Eth1 block, inserts it in the chain, and returns the status of the chain. | ||||||
| func (api *ConsensusAPI) ExecutePayloadV1(params ExecutableDataV1) (ExecutePayloadResponse, error) { | func (api *ConsensusAPI) ExecutePayloadV1(params ExecutableDataV1) (ExecutePayloadResponse, error) { | ||||||
|  | 	log.Trace("Engine API request received", "method", "ExecutePayload", params.BlockHash, "number", params.Number) | ||||||
| 	block, err := ExecutableDataToBlock(params) | 	block, err := ExecutableDataToBlock(params) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return api.invalid(), err | 		return api.invalid(), err | ||||||
| @@ -276,6 +272,7 @@ func (api *ConsensusAPI) ExecutePayloadV1(params ExecutableDataV1) (ExecutePaylo | |||||||
| 	if td.Cmp(ttd) < 0 { | 	if td.Cmp(ttd) < 0 { | ||||||
| 		return api.invalid(), fmt.Errorf("can not execute payload on top of block with low td got: %v threshold %v", td, ttd) | 		return api.invalid(), fmt.Errorf("can not execute payload on top of block with low td got: %v threshold %v", td, ttd) | ||||||
| 	} | 	} | ||||||
|  | 	log.Trace("Inserting block without head", "hash", block.Hash(), "number", block.Number) | ||||||
| 	if err := api.eth.BlockChain().InsertBlockWithoutSetHead(block); err != nil { | 	if err := api.eth.BlockChain().InsertBlockWithoutSetHead(block); err != nil { | ||||||
| 		return api.invalid(), err | 		return api.invalid(), err | ||||||
| 	} | 	} | ||||||
| @@ -301,8 +298,8 @@ func (api *ConsensusAPI) assembleBlock(parentHash common.Hash, params *PayloadAt | |||||||
| 		return nil, fmt.Errorf("cannot assemble block with unknown parent %s", parentHash) | 		return nil, fmt.Errorf("cannot assemble block with unknown parent %s", parentHash) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if params.Timestamp < parent.Time() { | 	if params.Timestamp <= parent.Time() { | ||||||
| 		return nil, fmt.Errorf("child timestamp lower than parent's: %d < %d", params.Timestamp, parent.Time()) | 		return nil, fmt.Errorf("invalid timestamp: child's %d <= parent's %d", params.Timestamp, parent.Time()) | ||||||
| 	} | 	} | ||||||
| 	if now := uint64(time.Now().Unix()); params.Timestamp > now+1 { | 	if now := uint64(time.Now().Unix()); params.Timestamp > now+1 { | ||||||
| 		diff := time.Duration(params.Timestamp-now) * time.Second | 		diff := time.Duration(params.Timestamp-now) * time.Second | ||||||
|   | |||||||
| @@ -17,12 +17,12 @@ | |||||||
| package catalyst | package catalyst | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"fmt" | ||||||
| 	"math/big" | 	"math/big" | ||||||
| 	"testing" | 	"testing" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/ethereum/go-ethereum/common" | 	"github.com/ethereum/go-ethereum/common" | ||||||
| 	"github.com/ethereum/go-ethereum/common/hexutil" |  | ||||||
| 	"github.com/ethereum/go-ethereum/consensus/ethash" | 	"github.com/ethereum/go-ethereum/consensus/ethash" | ||||||
| 	"github.com/ethereum/go-ethereum/core" | 	"github.com/ethereum/go-ethereum/core" | ||||||
| 	"github.com/ethereum/go-ethereum/core/rawdb" | 	"github.com/ethereum/go-ethereum/core/rawdb" | ||||||
| @@ -158,13 +158,21 @@ func TestEth2PrepareAndGetPayload(t *testing.T) { | |||||||
| 		t.Fatalf("error preparing payload, err=%v", err) | 		t.Fatalf("error preparing payload, err=%v", err) | ||||||
| 	} | 	} | ||||||
| 	payloadID := computePayloadId(fcState.HeadBlockHash, &blockParams) | 	payloadID := computePayloadId(fcState.HeadBlockHash, &blockParams) | ||||||
| 	execData, err := api.GetPayloadV1(hexutil.Bytes(payloadID)) | 	execData, err := api.GetPayloadV1(payloadID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Fatalf("error getting payload, err=%v", err) | 		t.Fatalf("error getting payload, err=%v", err) | ||||||
| 	} | 	} | ||||||
| 	if len(execData.Transactions) != blocks[9].Transactions().Len() { | 	if len(execData.Transactions) != blocks[9].Transactions().Len() { | ||||||
| 		t.Fatalf("invalid number of transactions %d != 1", len(execData.Transactions)) | 		t.Fatalf("invalid number of transactions %d != 1", len(execData.Transactions)) | ||||||
| 	} | 	} | ||||||
|  | 	// Test invalid payloadID | ||||||
|  | 	var invPayload PayloadID | ||||||
|  | 	copy(invPayload[:], payloadID[:]) | ||||||
|  | 	invPayload[0] = ^invPayload[0] | ||||||
|  | 	_, err = api.GetPayloadV1(invPayload) | ||||||
|  | 	if err == nil { | ||||||
|  | 		t.Fatal("expected error retrieving invalid payload") | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func checkLogEvents(t *testing.T, logsCh <-chan []*types.Log, rmLogsCh <-chan core.RemovedLogsEvent, wantNew, wantRemoved int) { | func checkLogEvents(t *testing.T, logsCh <-chan []*types.Log, rmLogsCh <-chan core.RemovedLogsEvent, wantNew, wantRemoved int) { | ||||||
| @@ -185,6 +193,48 @@ func checkLogEvents(t *testing.T, logsCh <-chan []*types.Log, rmLogsCh <-chan co | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestInvalidPayloadTimestamp(t *testing.T) { | ||||||
|  | 	genesis, preMergeBlocks := generatePreMergeChain(10) | ||||||
|  | 	n, ethservice := startEthService(t, genesis, preMergeBlocks) | ||||||
|  | 	ethservice.Merger().ReachTTD() | ||||||
|  | 	defer n.Close() | ||||||
|  | 	var ( | ||||||
|  | 		api    = NewConsensusAPI(ethservice, nil) | ||||||
|  | 		parent = ethservice.BlockChain().CurrentBlock() | ||||||
|  | 	) | ||||||
|  | 	tests := []struct { | ||||||
|  | 		time      uint64 | ||||||
|  | 		shouldErr bool | ||||||
|  | 	}{ | ||||||
|  | 		{0, true}, | ||||||
|  | 		{parent.Time(), true}, | ||||||
|  | 		{parent.Time() - 1, true}, | ||||||
|  | 		{parent.Time() + 1, false}, | ||||||
|  | 		{uint64(time.Now().Unix()) + uint64(time.Minute), false}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for i, test := range tests { | ||||||
|  | 		t.Run(fmt.Sprintf("Timestamp test: %v", i), func(t *testing.T) { | ||||||
|  | 			params := PayloadAttributesV1{ | ||||||
|  | 				Timestamp:             test.time, | ||||||
|  | 				Random:                crypto.Keccak256Hash([]byte{byte(123)}), | ||||||
|  | 				SuggestedFeeRecipient: parent.Coinbase(), | ||||||
|  | 			} | ||||||
|  | 			fcState := ForkchoiceStateV1{ | ||||||
|  | 				HeadBlockHash:      parent.Hash(), | ||||||
|  | 				SafeBlockHash:      common.Hash{}, | ||||||
|  | 				FinalizedBlockHash: common.Hash{}, | ||||||
|  | 			} | ||||||
|  | 			_, err := api.ForkchoiceUpdatedV1(fcState, ¶ms) | ||||||
|  | 			if test.shouldErr && err == nil { | ||||||
|  | 				t.Fatalf("expected error preparing payload with invalid timestamp, err=%v", err) | ||||||
|  | 			} else if !test.shouldErr && err != nil { | ||||||
|  | 				t.Fatalf("error preparing payload with valid timestamp, err=%v", err) | ||||||
|  | 			} | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| func TestEth2NewBlock(t *testing.T) { | func TestEth2NewBlock(t *testing.T) { | ||||||
| 	genesis, preMergeBlocks := generatePreMergeChain(10) | 	genesis, preMergeBlocks := generatePreMergeChain(10) | ||||||
| 	n, ethservice := startEthService(t, genesis, preMergeBlocks) | 	n, ethservice := startEthService(t, genesis, preMergeBlocks) | ||||||
| @@ -391,7 +441,7 @@ func TestFullAPI(t *testing.T) { | |||||||
| 			t.Fatalf("error preparing payload, invalid status: %v", resp.Status) | 			t.Fatalf("error preparing payload, invalid status: %v", resp.Status) | ||||||
| 		} | 		} | ||||||
| 		payloadID := computePayloadId(parent.Hash(), ¶ms) | 		payloadID := computePayloadId(parent.Hash(), ¶ms) | ||||||
| 		payload, err := api.GetPayloadV1(hexutil.Bytes(payloadID)) | 		payload, err := api.GetPayloadV1(payloadID) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			t.Fatalf("can't get payload: %v", err) | 			t.Fatalf("can't get payload: %v", err) | ||||||
| 		} | 		} | ||||||
| @@ -414,6 +464,5 @@ func TestFullAPI(t *testing.T) { | |||||||
| 			t.Fatalf("Chain head should be updated") | 			t.Fatalf("Chain head should be updated") | ||||||
| 		} | 		} | ||||||
| 		parent = ethservice.BlockChain().CurrentBlock() | 		parent = ethservice.BlockChain().CurrentBlock() | ||||||
|  |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -17,6 +17,7 @@ | |||||||
| package catalyst | package catalyst | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"fmt" | ||||||
| 	"math/big" | 	"math/big" | ||||||
|  |  | ||||||
| 	"github.com/ethereum/go-ethereum/common" | 	"github.com/ethereum/go-ethereum/common" | ||||||
| @@ -69,17 +70,6 @@ type executableDataMarshaling struct { | |||||||
| 	Transactions  []hexutil.Bytes | 	Transactions  []hexutil.Bytes | ||||||
| } | } | ||||||
|  |  | ||||||
| //go:generate go run github.com/fjl/gencodec -type PayloadResponse -field-override payloadResponseMarshaling -out gen_payload.go |  | ||||||
|  |  | ||||||
| type PayloadResponse struct { |  | ||||||
| 	PayloadID uint64 `json:"payloadId"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // JSON type overrides for payloadResponse. |  | ||||||
| type payloadResponseMarshaling struct { |  | ||||||
| 	PayloadID hexutil.Uint64 |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type NewBlockResponse struct { | type NewBlockResponse struct { | ||||||
| 	Valid bool `json:"valid"` | 	Valid bool `json:"valid"` | ||||||
| } | } | ||||||
| @@ -102,9 +92,28 @@ type ConsensusValidatedParams struct { | |||||||
| 	Status    string      `json:"status"` | 	Status    string      `json:"status"` | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // PayloadID is an identifier of the payload build process | ||||||
|  | type PayloadID [8]byte | ||||||
|  |  | ||||||
|  | func (b PayloadID) String() string { | ||||||
|  | 	return hexutil.Encode(b[:]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b PayloadID) MarshalText() ([]byte, error) { | ||||||
|  | 	return hexutil.Bytes(b[:]).MarshalText() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *PayloadID) UnmarshalText(input []byte) error { | ||||||
|  | 	err := hexutil.UnmarshalFixedText("PayloadID", input, b[:]) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("invalid payload id %q: %w", input, err) | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
| type ForkChoiceResponse struct { | type ForkChoiceResponse struct { | ||||||
| 	Status    string         `json:"status"` | 	Status    string     `json:"status"` | ||||||
| 	PayloadID *hexutil.Bytes `json:"payloadId"` | 	PayloadID *PayloadID `json:"payloadId"` | ||||||
| } | } | ||||||
|  |  | ||||||
| type ForkchoiceStateV1 struct { | type ForkchoiceStateV1 struct { | ||||||
|   | |||||||
| @@ -1,36 +0,0 @@ | |||||||
| // Code generated by github.com/fjl/gencodec. DO NOT EDIT. |  | ||||||
|  |  | ||||||
| package catalyst |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"encoding/json" |  | ||||||
|  |  | ||||||
| 	"github.com/ethereum/go-ethereum/common/hexutil" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var _ = (*payloadResponseMarshaling)(nil) |  | ||||||
|  |  | ||||||
| // MarshalJSON marshals as JSON. |  | ||||||
| func (p PayloadResponse) MarshalJSON() ([]byte, error) { |  | ||||||
| 	type PayloadResponse struct { |  | ||||||
| 		PayloadID hexutil.Uint64 `json:"payloadId"` |  | ||||||
| 	} |  | ||||||
| 	var enc PayloadResponse |  | ||||||
| 	enc.PayloadID = hexutil.Uint64(p.PayloadID) |  | ||||||
| 	return json.Marshal(&enc) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // UnmarshalJSON unmarshals from JSON. |  | ||||||
| func (p *PayloadResponse) UnmarshalJSON(input []byte) error { |  | ||||||
| 	type PayloadResponse struct { |  | ||||||
| 		PayloadID *hexutil.Uint64 `json:"payloadId"` |  | ||||||
| 	} |  | ||||||
| 	var dec PayloadResponse |  | ||||||
| 	if err := json.Unmarshal(input, &dec); err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	if dec.PayloadID != nil { |  | ||||||
| 		p.PayloadID = uint64(*dec.PayloadID) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
							
								
								
									
										74
									
								
								eth/catalyst/queue.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								eth/catalyst/queue.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,74 @@ | |||||||
|  | // Copyright 2022 The go-ethereum Authors | ||||||
|  | // This file is part of the go-ethereum library. | ||||||
|  | // | ||||||
|  | // The go-ethereum library is free software: you can redistribute it and/or modify | ||||||
|  | // it under the terms of the GNU Lesser General Public License as published by | ||||||
|  | // the Free Software Foundation, either version 3 of the License, or | ||||||
|  | // (at your option) any later version. | ||||||
|  | // | ||||||
|  | // The go-ethereum library is distributed in the hope that it will be useful, | ||||||
|  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
|  | // GNU Lesser General Public License for more details. | ||||||
|  | // | ||||||
|  | // You should have received a copy of the GNU Lesser General Public License | ||||||
|  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  |  | ||||||
|  | package catalyst | ||||||
|  |  | ||||||
|  | import "sync" | ||||||
|  |  | ||||||
|  | // maxTrackedPayloads is the maximum number of prepared payloads the execution | ||||||
|  | // engine tracks before evicting old ones. Ideally we should only ever track the | ||||||
|  | // latest one; but have a slight wiggle room for non-ideal conditions. | ||||||
|  | const maxTrackedPayloads = 10 | ||||||
|  |  | ||||||
|  | // payloadQueueItem represents an id->payload tuple to store until it's retrieved | ||||||
|  | // or evicted. | ||||||
|  | type payloadQueueItem struct { | ||||||
|  | 	id      PayloadID | ||||||
|  | 	payload *ExecutableDataV1 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // payloadQueue tracks the latest handful of constructed payloads to be retrieved | ||||||
|  | // by the beacon chain if block production is requested. | ||||||
|  | type payloadQueue struct { | ||||||
|  | 	payloads []*payloadQueueItem | ||||||
|  | 	lock     sync.RWMutex | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // newPayloadQueue creates a pre-initialized queue with a fixed number of slots | ||||||
|  | // all containing empty items. | ||||||
|  | func newPayloadQueue() *payloadQueue { | ||||||
|  | 	return &payloadQueue{ | ||||||
|  | 		payloads: make([]*payloadQueueItem, maxTrackedPayloads), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // put inserts a new payload into the queue at the given id. | ||||||
|  | func (q *payloadQueue) put(id PayloadID, data *ExecutableDataV1) { | ||||||
|  | 	q.lock.Lock() | ||||||
|  | 	defer q.lock.Unlock() | ||||||
|  |  | ||||||
|  | 	copy(q.payloads[1:], q.payloads) | ||||||
|  | 	q.payloads[0] = &payloadQueueItem{ | ||||||
|  | 		id:      id, | ||||||
|  | 		payload: data, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // get retrieves a previously stored payload item or nil if it does not exist. | ||||||
|  | func (q *payloadQueue) get(id PayloadID) *ExecutableDataV1 { | ||||||
|  | 	q.lock.RLock() | ||||||
|  | 	defer q.lock.RUnlock() | ||||||
|  |  | ||||||
|  | 	for _, item := range q.payloads { | ||||||
|  | 		if item == nil { | ||||||
|  | 			return nil // no more items | ||||||
|  | 		} | ||||||
|  | 		if item.id == id { | ||||||
|  | 			return item.payload | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user