From f4427158b40e62bbff85edbf7e03e87659244af2 Mon Sep 17 00:00:00 2001 From: Jay Jie Date: Thu, 24 Oct 2024 22:09:32 -0700 Subject: [PATCH 01/10] feat: add client implementation --- execution.go | 726 +++++++++++++++++++++++++++++++++++++++++++++++++++ go.mod | 102 +++++++- go.sum | 510 ++++++++++++++++++++++++++++++++++++ 3 files changed, 1337 insertions(+), 1 deletion(-) create mode 100644 execution.go create mode 100644 go.sum diff --git a/execution.go b/execution.go new file mode 100644 index 0000000..cb0df80 --- /dev/null +++ b/execution.go @@ -0,0 +1,726 @@ +package execution + +import ( + "bytes" + "context" + "errors" + "fmt" + "time" + + execution "github.com/LastL2/go-execution" + abci "github.com/cometbft/cometbft/abci/types" + cmbytes "github.com/cometbft/cometbft/libs/bytes" + cmproto "github.com/cometbft/cometbft/proto/tendermint/types" + "github.com/cometbft/cometbft/proxy" + cmtypes "github.com/cometbft/cometbft/types" + "github.com/rollkit/rollkit/mempool" + "github.com/rollkit/rollkit/state" + "github.com/rollkit/rollkit/store" + "github.com/rollkit/rollkit/third_party/log" + "github.com/rollkit/rollkit/types" + abciconv "github.com/rollkit/rollkit/types/abci" +) + +// ErrEmptyValSetGenerated is returned when applying the validator changes would result in empty set. +var ErrEmptyValSetGenerated = errors.New("applying the validator changes would result in empty set") + +// ErrAddingValidatorToBased is returned when trying to add a validator to an empty validator set. +var ErrAddingValidatorToBased = errors.New("cannot add validators to empty validator set") + +type ABCIExecutionClient struct { + // abci specific + proxyApp proxy.AppConnConsensus + eventBus *cmtypes.EventBus + genesis *cmtypes.GenesisDoc + maxBytes uint64 + proposerAddress []byte + chainID string + + // rollkit specific + mempool mempool.Mempool + mempoolReaper *mempool.CListMempoolReaper + logger log.Logger + metrics *state.Metrics + state *types.State + store store.Store +} + +func NewABCIExecutionClient(proposerAddress []byte, chainID string, mempool mempool.Mempool, mempoolReaper *mempool.CListMempoolReaper, proxyApp proxy.AppConnConsensus, eventBus *cmtypes.EventBus, maxBytes uint64, logger log.Logger, metrics *state.Metrics, store store.Store, genesis *cmtypes.GenesisDoc, state *types.State) *ABCIExecutionClient { + return &ABCIExecutionClient{ + proxyApp: proxyApp, + eventBus: eventBus, + genesis: genesis, + maxBytes: maxBytes, + proposerAddress: proposerAddress, + chainID: chainID, + mempool: mempool, + mempoolReaper: mempoolReaper, + logger: logger, + metrics: metrics, + store: store, + state: state, + } +} + +var _ execution.Execute = (*ABCIExecutionClient)(nil) + +// InitChain initializes the blockchain with genesis information. +func (c *ABCIExecutionClient) InitChain( + genesisTime time.Time, + initialHeight uint64, + chainID string, +) (types.Hash, uint64, error) { + genesis := &cmtypes.GenesisDoc{ + GenesisTime: genesisTime, + ChainID: chainID, + ConsensusParams: c.genesis.ConsensusParams, + Validators: c.genesis.Validators, + AppState: c.genesis.AppState, + InitialHeight: int64(initialHeight), + } + + response, err := c.initChain(genesis) + if err != nil { + return types.Hash{}, 0, err + } + + stateRoot := types.Hash(response.AppHash) + maxBytes := response.ConsensusParams.Block.MaxBytes + + return stateRoot, uint64(maxBytes), nil +} + +// initChain calls InitChainSync using consensus connection to app. +func (c *ABCIExecutionClient) initChain(genesis *cmtypes.GenesisDoc) (*abci.ResponseInitChain, error) { + params := genesis.ConsensusParams + + validators := make([]*cmtypes.Validator, len(genesis.Validators)) + for i, v := range genesis.Validators { + validators[i] = cmtypes.NewValidator(v.PubKey, v.Power) + } + + return c.proxyApp.InitChain(context.Background(), &abci.RequestInitChain{ + Time: genesis.GenesisTime, + ChainId: genesis.ChainID, + ConsensusParams: &cmproto.ConsensusParams{ + Block: &cmproto.BlockParams{ + MaxBytes: params.Block.MaxBytes, + MaxGas: params.Block.MaxGas, + }, + Evidence: &cmproto.EvidenceParams{ + MaxAgeNumBlocks: params.Evidence.MaxAgeNumBlocks, + MaxAgeDuration: params.Evidence.MaxAgeDuration, + MaxBytes: params.Evidence.MaxBytes, + }, + Validator: &cmproto.ValidatorParams{ + PubKeyTypes: params.Validator.PubKeyTypes, + }, + Version: &cmproto.VersionParams{ + App: params.Version.App, + }, + Abci: &cmproto.ABCIParams{ + VoteExtensionsEnableHeight: params.ABCI.VoteExtensionsEnableHeight, + }, + }, + Validators: cmtypes.TM2PB.ValidatorUpdates(cmtypes.NewValidatorSet(validators)), + AppStateBytes: genesis.AppState, + InitialHeight: genesis.InitialHeight, + }) +} + +// GetTxs retrieves all available transactions from the mempool. +func (c *ABCIExecutionClient) GetTxs() ([]types.Tx, error) { + state, err := c.store.GetState(context.Background()) + if err != nil { + return nil, fmt.Errorf("failed to get current state: %w", err) + } + + maxBytes := state.ConsensusParams.Block.MaxBytes + if maxBytes == -1 { + maxBytes = int64(cmtypes.MaxBlockSizeBytes) + } + if maxBytes > int64(c.maxBytes) { + c.logger.Debug("limiting maxBytes to", "maxBytes", c.maxBytes) + maxBytes = int64(c.maxBytes) + } + + cmTxs := c.mempool.ReapMaxTxs(int(maxBytes)) + + rollkitTxs := make([]types.Tx, len(cmTxs)) + for i, tx := range cmTxs { + rollkitTxs[i] = types.Tx(tx) + } + + return rollkitTxs, nil +} + +// ExecuteTxs executes a set of transactions to produce a new block. +func (c *ABCIExecutionClient) ExecuteTxs( + txs []types.Tx, + blockHeight uint64, + timestamp time.Time, + prevStateRoot types.Hash, +) (types.Hash, uint64, error) { + ctx := context.Background() + + state, err := c.store.GetState(ctx) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to get current state: %w", err) + } + + cmTxs := fromRollkitTxs(txs) + + var lastSignature *types.Signature + var lastHeaderHash types.Hash + var lastExtendedCommit abci.ExtendedCommitInfo + + if blockHeight == uint64(c.genesis.InitialHeight) { + lastSignature = &types.Signature{} + lastHeaderHash = types.Hash{} + lastExtendedCommit = abci.ExtendedCommitInfo{} + } else { + lastSignature, err = c.store.GetSignature(ctx, blockHeight-1) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("error while loading last commit: %w", err) + } + + lastHeader, _, err := c.store.GetBlockData(ctx, blockHeight-1) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("error while loading last block: %w", err) + } + lastHeaderHash = lastHeader.Hash() + + extCommit, err := c.store.GetExtendedCommit(ctx, blockHeight-1) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to load extended commit for height %d: %w", blockHeight-1, err) + } + if extCommit != nil { + lastExtendedCommit = *extCommit + } + } + + header, data, err := c.CreateBlock( + blockHeight, + lastSignature, + lastExtendedCommit, + lastHeaderHash, + state, + cmTxs, + timestamp, + ) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to create block: %w", err) + } + + isValid, err := c.ProcessProposal(header, data, state) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to process proposal: %w", err) + } + if !isValid { + return types.Hash{}, 0, fmt.Errorf("proposal was not valid") + } + + newState, resp, err := c.ApplyBlock(ctx, state, header, data) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to apply block: %w", err) + } + + appHash, _, err := c.Commit(ctx, newState, header, data, resp) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to commit: %w", err) + } + + return types.Hash(appHash), uint64(newState.ConsensusParams.Block.MaxBytes), nil +} + +// SetFinal marks a block at the given height as final. +func (c *ABCIExecutionClient) SetFinal(blockHeight uint64) error { + ctx := context.Background() + + header, data, err := c.store.GetBlockData(ctx, blockHeight) + if err != nil { + return fmt.Errorf("failed to get block data for height %d: %w", blockHeight, err) + } + + state, err := c.store.GetState(ctx) + if err != nil { + return fmt.Errorf("failed to get current state: %w", err) + } + + resp, err := c.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ + Hash: header.Hash(), + Height: int64(blockHeight), + Time: header.Time(), + Txs: data.Txs.ToSliceOfBytes(), + ProposerAddress: header.ProposerAddress, + NextValidatorsHash: state.Validators.Hash(), + }) + if err != nil { + return fmt.Errorf("failed to finalize block at height %d: %w", blockHeight, err) + } + + state.AppHash = resp.AppHash + if err := c.store.UpdateState(ctx, state); err != nil { + return fmt.Errorf("failed to update state after finalizing block %d: %w", blockHeight, err) + } + + c.logger.Info("Block finalized", "height", blockHeight, "hash", header.Hash()) + + return nil +} + +// CreateBlock reaps transactions from mempool and builds a block. +func (c *ABCIExecutionClient) CreateBlock(height uint64, lastSignature *types.Signature, lastExtendedCommit abci.ExtendedCommitInfo, lastHeaderHash types.Hash, state types.State, txs cmtypes.Txs, timestamp time.Time) (*types.SignedHeader, *types.Data, error) { + maxBytes := state.ConsensusParams.Block.MaxBytes + emptyMaxBytes := maxBytes == -1 + if emptyMaxBytes { + maxBytes = int64(cmtypes.MaxBlockSizeBytes) + } + if maxBytes > int64(c.maxBytes) { //nolint:gosec + c.logger.Debug("limiting maxBytes to", "e.maxBytes=%d", c.maxBytes) + maxBytes = int64(c.maxBytes) //nolint:gosec + } + + header := &types.SignedHeader{ + Header: types.Header{ + Version: types.Version{ + Block: state.Version.Consensus.Block, + App: state.Version.Consensus.App, + }, + BaseHeader: types.BaseHeader{ + ChainID: c.chainID, + Height: height, + Time: uint64(timestamp.UnixNano()), //nolint:gosec + }, + DataHash: make(types.Hash, 32), + ConsensusHash: make(types.Hash, 32), + AppHash: state.AppHash, + LastResultsHash: state.LastResultsHash, + ProposerAddress: c.proposerAddress, + }, + Signature: *lastSignature, + } + data := &types.Data{ + Txs: toRollkitTxs(txs), + // IntermediateStateRoots: types.IntermediateStateRoots{RawRootsList: nil}, + // Note: Temporarily remove Evidence #896 + // Evidence: types.EvidenceData{Evidence: nil}, + } + + rpp, err := c.proxyApp.PrepareProposal( + context.TODO(), + &abci.RequestPrepareProposal{ + MaxTxBytes: maxBytes, + Txs: txs.ToSliceOfBytes(), + LocalLastCommit: lastExtendedCommit, + Misbehavior: []abci.Misbehavior{}, + Height: int64(header.Height()), //nolint:gosec + Time: header.Time(), //TODO: replace with sequencer timestamp + NextValidatorsHash: state.Validators.Hash(), + ProposerAddress: c.proposerAddress, + }, + ) + if err != nil { + // The App MUST ensure that only valid (and hence 'processable') transactions + // enter the mempool. Hence, at this point, we can't have any non-processable + // transaction causing an error. + // + // Also, the App can simply skip any transaction that could cause any kind of trouble. + // Either way, we cannot recover in a meaningful way, unless we skip proposing + // this block, repair what caused the error and try again. Hence, we return an + // error for now (the production code calling this function is expected to panic). + return nil, nil, err + } + + txl := cmtypes.ToTxs(rpp.Txs) + if err := txl.Validate(maxBytes); err != nil { + return nil, nil, err + } + + data.Txs = toRollkitTxs(txl) + // Note: This is hash of an ABCI type commit equivalent of the last signature in the signed header. + header.LastCommitHash = lastSignature.GetCommitHash(&header.Header, c.proposerAddress) + header.LastHeaderHash = lastHeaderHash + + return header, data, nil +} + +// ProcessProposal calls the corresponding ABCI method on the app. +func (c *ABCIExecutionClient) ProcessProposal( + header *types.SignedHeader, + data *types.Data, + state types.State, +) (bool, error) { + resp, err := c.proxyApp.ProcessProposal(context.TODO(), &abci.RequestProcessProposal{ + Hash: header.Hash(), + Height: int64(header.Height()), //nolint:gosec + Time: header.Time(), + Txs: data.Txs.ToSliceOfBytes(), + ProposedLastCommit: abci.CommitInfo{ + Round: 0, + Votes: []abci.VoteInfo{ + { + Validator: abci.Validator{ + Address: header.Validators.GetProposer().Address, + Power: header.Validators.GetProposer().VotingPower, + }, + BlockIdFlag: cmproto.BlockIDFlagCommit, + }, + }, + }, + Misbehavior: []abci.Misbehavior{}, + ProposerAddress: c.proposerAddress, + NextValidatorsHash: state.Validators.Hash(), + }) + if err != nil { + return false, err + } + if resp.IsStatusUnknown() { + panic(fmt.Sprintf("ProcessProposal responded with status %s", resp.Status.String())) + } + + return resp.IsAccepted(), nil +} + +// ApplyBlock validates and executes the block. +func (c *ABCIExecutionClient) ApplyBlock(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data) (types.State, *abci.ResponseFinalizeBlock, error) { + isAppValid, err := c.ProcessProposal(header, data, state) + if err != nil { + return types.State{}, nil, err + } + if !isAppValid { + return types.State{}, nil, fmt.Errorf("proposal processing resulted in an invalid application state") + } + + err = c.Validate(state, header, data) + if err != nil { + return types.State{}, nil, err + } + // This makes calls to the AppClient + resp, err := c.execute(ctx, state, header, data) + if err != nil { + return types.State{}, nil, err + } + abciValUpdates := resp.ValidatorUpdates + + validatorUpdates, err := cmtypes.PB2TM.ValidatorUpdates(abciValUpdates) + if err != nil { + return state, nil, err + } + + if resp.ConsensusParamUpdates != nil { + c.metrics.ConsensusParamUpdates.Add(1) + } + + state, err = c.updateState(state, header, data, resp, validatorUpdates) + if err != nil { + return types.State{}, nil, err + } + + if state.ConsensusParams.Block.MaxBytes <= 0 { + c.logger.Error("maxBytes<=0", "state.ConsensusParams.Block", state.ConsensusParams.Block, "header", header) + } + + return state, resp, nil +} + +// ExtendVote calls the ExtendVote ABCI method on the proxy app. +func (c *ABCIExecutionClient) ExtendVote(ctx context.Context, header *types.SignedHeader, data *types.Data) ([]byte, error) { + resp, err := c.proxyApp.ExtendVote(ctx, &abci.RequestExtendVote{ + Hash: header.Hash(), + Height: int64(header.Height()), //nolint:gosec + Time: header.Time(), + Txs: data.Txs.ToSliceOfBytes(), + ProposedLastCommit: abci.CommitInfo{ + Votes: []abci.VoteInfo{{ + Validator: abci.Validator{ + Address: header.Validators.GetProposer().Address, + Power: header.Validators.GetProposer().VotingPower, + }, + BlockIdFlag: cmproto.BlockIDFlagCommit, + }}, + }, + Misbehavior: nil, + NextValidatorsHash: header.ValidatorHash, + ProposerAddress: header.ProposerAddress, + }) + if err != nil { + return nil, err + } + return resp.VoteExtension, nil +} + +// Commit commits the block +func (c *ABCIExecutionClient) Commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { + appHash, retainHeight, err := c.commit(ctx, state, header, data, resp) + if err != nil { + return []byte{}, 0, err + } + + state.AppHash = appHash + + c.publishEvents(resp, header, data, state) + + return appHash, retainHeight, nil +} + +// updateConsensusParams updates the consensus parameters based on the provided updates. +func (c *ABCIExecutionClient) updateConsensusParams(height uint64, params cmtypes.ConsensusParams, consensusParamUpdates *cmproto.ConsensusParams) (cmproto.ConsensusParams, uint64, error) { + nextParams := params.Update(consensusParamUpdates) + if err := types.ConsensusParamsValidateBasic(nextParams); err != nil { + return cmproto.ConsensusParams{}, 0, fmt.Errorf("validating new consensus params: %w", err) + } + if err := nextParams.ValidateUpdate(consensusParamUpdates, int64(height)); err != nil { //nolint:gosec + return cmproto.ConsensusParams{}, 0, fmt.Errorf("updating consensus params: %w", err) + } + return nextParams.ToProto(), nextParams.Version.App, nil +} + +func (c *ABCIExecutionClient) updateState(state types.State, header *types.SignedHeader, data *types.Data, finalizeBlockResponse *abci.ResponseFinalizeBlock, validatorUpdates []*cmtypes.Validator) (types.State, error) { + height := header.Height() + if finalizeBlockResponse.ConsensusParamUpdates != nil { + nextParamsProto, appVersion, err := c.updateConsensusParams(height, types.ConsensusParamsFromProto(state.ConsensusParams), finalizeBlockResponse.ConsensusParamUpdates) + if err != nil { + return types.State{}, err + } + // Change results from this height but only applies to the next height. + state.LastHeightConsensusParamsChanged = height + 1 + state.Version.Consensus.App = appVersion + state.ConsensusParams = nextParamsProto + } + + nValSet := state.NextValidators.Copy() + lastHeightValSetChanged := state.LastHeightValidatorsChanged + + if len(nValSet.Validators) > 0 { + err := nValSet.UpdateWithChangeSet(validatorUpdates) + if err != nil { + if err.Error() != ErrEmptyValSetGenerated.Error() { + return state, err + } + nValSet = &cmtypes.ValidatorSet{ + Validators: make([]*cmtypes.Validator, 0), + Proposer: nil, + } + } + // Change results from this height but only applies to the next next height. + lastHeightValSetChanged = int64(header.Header.Height() + 1 + 1) //nolint:gosec + + if len(nValSet.Validators) > 0 { + nValSet.IncrementProposerPriority(1) + } + } + + s := types.State{ + Version: state.Version, + ChainID: state.ChainID, + InitialHeight: state.InitialHeight, + LastBlockHeight: height, + LastBlockTime: header.Time(), + LastBlockID: cmtypes.BlockID{ + Hash: cmbytes.HexBytes(header.Hash()), + // for now, we don't care about part set headers + }, + ConsensusParams: state.ConsensusParams, + LastHeightConsensusParamsChanged: state.LastHeightConsensusParamsChanged, + AppHash: finalizeBlockResponse.AppHash, + Validators: state.NextValidators.Copy(), + NextValidators: nValSet, + LastHeightValidatorsChanged: lastHeightValSetChanged, + LastValidators: state.Validators.Copy(), + } + copy(s.LastResultsHash[:], cmtypes.NewResults(finalizeBlockResponse.TxResults).Hash()) + + return s, nil +} + +func (e *ABCIExecutionClient) commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { + e.mempool.Lock() + defer e.mempool.Unlock() + + err := e.mempool.FlushAppConn() + if err != nil { + return nil, 0, err + } + + commitResp, err := e.proxyApp.Commit(ctx) + if err != nil { + return nil, 0, err + } + + maxBytes := state.ConsensusParams.Block.MaxBytes + maxGas := state.ConsensusParams.Block.MaxGas + cTxs := fromRollkitTxs(data.Txs) + e.mempoolReaper.UpdateCommitedTxs(cTxs) + err = e.mempool.Update(header.Height(), cTxs, resp.TxResults, mempool.PreCheckMaxBytes(maxBytes), mempool.PostCheckMaxGas(maxGas)) + if err != nil { + return nil, 0, err + } + + return resp.AppHash, uint64(commitResp.RetainHeight), err //nolint:gosec +} + +// Validate validates the state and the block for the executor +func (e *ABCIExecutionClient) Validate(state types.State, header *types.SignedHeader, data *types.Data) error { + if err := header.ValidateBasic(); err != nil { + return err + } + if err := data.ValidateBasic(); err != nil { + return err + } + if err := types.Validate(header, data); err != nil { + return err + } + if header.Version.App != state.Version.Consensus.App || + header.Version.Block != state.Version.Consensus.Block { + return errors.New("block version mismatch") + } + if state.LastBlockHeight <= 0 && header.Height() != state.InitialHeight { + return errors.New("initial block height mismatch") + } + if state.LastBlockHeight > 0 && header.Height() != state.LastBlockHeight+1 { + return errors.New("block height mismatch") + } + if !bytes.Equal(header.AppHash[:], state.AppHash[:]) { + return errors.New("AppHash mismatch") + } + + if !bytes.Equal(header.LastResultsHash[:], state.LastResultsHash[:]) { + return errors.New("LastResultsHash mismatch") + } + + return nil +} + +func (e *ABCIExecutionClient) execute(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data) (*abci.ResponseFinalizeBlock, error) { + // Only execute if the node hasn't already shut down + select { + case <-ctx.Done(): + return nil, ctx.Err() + default: + } + abciHeader, err := abciconv.ToABCIHeaderPB(&header.Header) + if err != nil { + return nil, err + } + abciHeader.ChainID = e.chainID + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return nil, err + } + + startTime := time.Now().UnixNano() + finalizeBlockResponse, err := e.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ + Hash: header.Hash(), + NextValidatorsHash: state.Validators.Hash(), + ProposerAddress: abciHeader.ProposerAddress, + Height: abciHeader.Height, + Time: abciHeader.Time, + DecidedLastCommit: abci.CommitInfo{ + Round: 0, + Votes: nil, + }, + Misbehavior: abciBlock.Evidence.Evidence.ToABCI(), + Txs: abciBlock.Txs.ToSliceOfBytes(), + }) + endTime := time.Now().UnixNano() + e.metrics.BlockProcessingTime.Observe(float64(endTime-startTime) / 1000000) + if err != nil { + e.logger.Error("error in proxyAppConn.FinalizeBlock", "err", err) + return nil, err + } + + e.logger.Info( + "finalized block", + "height", abciBlock.Height, + "num_txs_res", len(finalizeBlockResponse.TxResults), + "num_val_updates", len(finalizeBlockResponse.ValidatorUpdates), + "block_app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash), + ) + + // Assert that the application correctly returned tx results for each of the transactions provided in the block + if len(abciBlock.Data.Txs) != len(finalizeBlockResponse.TxResults) { + return nil, fmt.Errorf("expected tx results length to match size of transactions in block. Expected %d, got %d", len(data.Txs), len(finalizeBlockResponse.TxResults)) + } + + e.logger.Info("executed block", "height", abciHeader.Height, "app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash)) + + return finalizeBlockResponse, nil +} + +func (e *ABCIExecutionClient) publishEvents(resp *abci.ResponseFinalizeBlock, header *types.SignedHeader, data *types.Data, state types.State) { + if e.eventBus == nil { + return + } + + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return + } + + if err := e.eventBus.PublishEventNewBlock(cmtypes.EventDataNewBlock{ + Block: abciBlock, + BlockID: cmtypes.BlockID{ + Hash: cmbytes.HexBytes(header.Hash()), + // for now, we don't care about part set headers + }, + ResultFinalizeBlock: *resp, + }); err != nil { + e.logger.Error("failed publishing new block", "err", err) + } + + if err := e.eventBus.PublishEventNewBlockHeader(cmtypes.EventDataNewBlockHeader{ + Header: abciBlock.Header, + }); err != nil { + e.logger.Error("failed publishing new block header", "err", err) + } + + if err := e.eventBus.PublishEventNewBlockEvents(cmtypes.EventDataNewBlockEvents{ + Height: abciBlock.Height, + Events: resp.Events, + NumTxs: int64(len(abciBlock.Txs)), + }); err != nil { + e.logger.Error("failed publishing new block events", "err", err) + } + + if len(abciBlock.Evidence.Evidence) != 0 { + for _, ev := range abciBlock.Evidence.Evidence { + if err := e.eventBus.PublishEventNewEvidence(cmtypes.EventDataNewEvidence{ + Evidence: ev, + Height: int64(header.Header.Height()), //nolint:gosec + }); err != nil { + e.logger.Error("failed publishing new evidence", "err", err) + } + } + } + + for i, tx := range abciBlock.Data.Txs { + err := e.eventBus.PublishEventTx(cmtypes.EventDataTx{ + TxResult: abci.TxResult{ + Height: abciBlock.Height, + Index: uint32(i), //nolint:gosec + Tx: tx, + Result: *(resp.TxResults[i]), + }, + }) + if err != nil { + e.logger.Error("failed publishing event TX", "err", err) + } + } +} + +func toRollkitTxs(txs cmtypes.Txs) types.Txs { + rollkitTxs := make(types.Txs, len(txs)) + for i := range txs { + rollkitTxs[i] = []byte(txs[i]) + } + return rollkitTxs +} + +func fromRollkitTxs(rollkitTxs types.Txs) cmtypes.Txs { + txs := make(cmtypes.Txs, len(rollkitTxs)) + for i := range rollkitTxs { + txs[i] = []byte(rollkitTxs[i]) + } + return txs +} diff --git a/go.mod b/go.mod index 77d492d..2cbb42e 100644 --- a/go.mod +++ b/go.mod @@ -1,3 +1,103 @@ module github.com/rollkit/template-da-repo -go 1.21.0 +go 1.22.7 + +toolchain go1.22.8 + +replace github.com/LastL2/go-execution => github.com/LastL2/go-execution v0.0.0-20241025044830-6028e95ddb3a + +require ( + github.com/LastL2/go-execution v0.0.0-00010101000000-000000000000 + github.com/cometbft/cometbft v0.38.13 + github.com/rollkit/rollkit v0.13.8-0.20241021170619-fc65faf646a7 +) + +require ( + github.com/DataDog/zstd v1.4.5 // indirect + github.com/beorn7/perks v1.0.1 // indirect + github.com/celestiaorg/go-header v0.6.2 // indirect + github.com/cespare/xxhash/v2 v2.3.0 // indirect + github.com/cockroachdb/errors v1.11.3 // indirect + github.com/cockroachdb/fifo v0.0.0-20240606204812-0bbfbd93a7ce // indirect + github.com/cockroachdb/logtags v0.0.0-20230118201751-21c54148d20b // indirect + github.com/cockroachdb/pebble v1.1.1 // indirect + github.com/cockroachdb/redact v1.1.5 // indirect + github.com/cockroachdb/tokenbucket v0.0.0-20230807174530-cc333fc44b06 // indirect + github.com/cometbft/cometbft-db v0.14.1 // indirect + github.com/cosmos/gogoproto v1.7.0 // indirect + github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect + github.com/decred/dcrd/dcrec/secp256k1/v4 v4.3.0 // indirect + github.com/dgraph-io/badger/v4 v4.2.1-0.20231013074411-fb1b00959581 // indirect + github.com/dgraph-io/ristretto v0.1.1 // indirect + github.com/dustin/go-humanize v1.0.1 // indirect + github.com/getsentry/sentry-go v0.27.0 // indirect + github.com/go-kit/kit v0.13.0 // indirect + github.com/go-kit/log v0.2.1 // indirect + github.com/go-logfmt/logfmt v0.6.0 // indirect + github.com/gogo/protobuf v1.3.2 // indirect + github.com/golang/glog v1.2.2 // indirect + github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect + github.com/golang/protobuf v1.5.4 // indirect + github.com/golang/snappy v0.0.4 // indirect + github.com/google/btree v1.1.3 // indirect + github.com/google/flatbuffers v1.12.1 // indirect + github.com/google/go-cmp v0.6.0 // indirect + github.com/google/uuid v1.6.0 // indirect + github.com/hashicorp/golang-lru/v2 v2.0.7 // indirect + github.com/ipfs/go-cid v0.4.1 // indirect + github.com/ipfs/go-datastore v0.6.0 // indirect + github.com/ipfs/go-ds-badger4 v0.1.5 // indirect + github.com/ipfs/go-log/v2 v2.5.1 // indirect + github.com/jbenet/goprocess v0.1.4 // indirect + github.com/jmhodges/levigo v1.0.0 // indirect + github.com/klauspost/compress v1.17.9 // indirect + github.com/klauspost/cpuid/v2 v2.2.8 // indirect + github.com/kr/pretty v0.3.1 // indirect + github.com/kr/text v0.2.0 // indirect + github.com/libp2p/go-buffer-pool v0.1.0 // indirect + github.com/libp2p/go-libp2p v0.36.2 // indirect + github.com/libp2p/go-libp2p-pubsub v0.12.0 // indirect + github.com/libp2p/go-msgio v0.3.0 // indirect + github.com/linxGnu/grocksdb v1.8.14 // indirect + github.com/mattn/go-isatty v0.0.20 // indirect + github.com/minio/sha256-simd v1.0.1 // indirect + github.com/mr-tron/base58 v1.2.0 // indirect + github.com/multiformats/go-base32 v0.1.0 // indirect + github.com/multiformats/go-base36 v0.2.0 // indirect + github.com/multiformats/go-multiaddr v0.13.0 // indirect + github.com/multiformats/go-multiaddr-fmt v0.1.0 // indirect + github.com/multiformats/go-multibase v0.2.0 // indirect + github.com/multiformats/go-multicodec v0.9.0 // indirect + github.com/multiformats/go-multihash v0.2.3 // indirect + github.com/multiformats/go-multistream v0.5.0 // indirect + github.com/multiformats/go-varint v0.0.7 // indirect + github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect + github.com/oasisprotocol/curve25519-voi v0.0.0-20220708102147-0a8a51822cae // indirect + github.com/petermattis/goid v0.0.0-20240813172612-4fcff4a6cae7 // indirect + github.com/pkg/errors v0.9.1 // indirect + github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect + github.com/prometheus/client_golang v1.20.4 // indirect + github.com/prometheus/client_model v0.6.1 // indirect + github.com/prometheus/common v0.59.1 // indirect + github.com/prometheus/procfs v0.15.1 // indirect + github.com/rogpeppe/go-internal v1.11.0 // indirect + github.com/rollkit/go-sequencing v0.2.1-0.20241010053131-3134457dc4e5 // indirect + github.com/sasha-s/go-deadlock v0.3.5 // indirect + github.com/spaolacci/murmur3 v1.1.0 // indirect + github.com/stretchr/testify v1.9.0 // indirect + github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7 // indirect + go.etcd.io/bbolt v1.4.0-alpha.0.0.20240404170359-43604f3112c5 // indirect + go.opencensus.io v0.24.0 // indirect + go.uber.org/multierr v1.11.0 // indirect + go.uber.org/zap v1.27.0 // indirect + golang.org/x/crypto v0.27.0 // indirect + golang.org/x/exp v0.0.0-20240808152545-0cdaa3abc0fa // indirect + golang.org/x/net v0.29.0 // indirect + golang.org/x/sys v0.25.0 // indirect + golang.org/x/text v0.18.0 // indirect + google.golang.org/genproto/googleapis/rpc v0.0.0-20240827150818-7e3bb234dfed // indirect + google.golang.org/grpc v1.67.1 // indirect + google.golang.org/protobuf v1.35.1 // indirect + gopkg.in/yaml.v3 v3.0.1 // indirect + lukechampine.com/blake3 v1.3.0 // indirect +) diff --git a/go.sum b/go.sum new file mode 100644 index 0000000..02dbdbf --- /dev/null +++ b/go.sum @@ -0,0 +1,510 @@ +cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= +github.com/DataDog/zstd v1.4.5 h1:EndNeuB0l9syBZhut0wns3gV1hL8zX8LIu6ZiVHWLIQ= +github.com/DataDog/zstd v1.4.5/go.mod h1:1jcaCB/ufaK+sKp1NBhlGmpz41jOoPQ35bpF36t7BBo= +github.com/LastL2/go-execution v0.0.0-20241025044830-6028e95ddb3a h1:l3EnFurDODyP+abpbtfb6v/F7TQmkMIYKRVu6xygO9c= +github.com/LastL2/go-execution v0.0.0-20241025044830-6028e95ddb3a/go.mod h1:PM6JVuwMV/RZa3SYq6vznGpzcfKFy6KYaDauAPQ13qc= +github.com/VividCortex/gohistogram v1.0.0 h1:6+hBz+qvs0JOrrNhhmR7lFxo5sINxBCGXrdtl/UvroE= +github.com/VividCortex/gohistogram v1.0.0/go.mod h1:Pf5mBqqDxYaXu3hDrrU+w6nw50o/4+TcAqDqk/vUH7g= +github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= +github.com/benbjohnson/clock v1.3.5 h1:VvXlSJBzZpA/zum6Sj74hxwYI2DIxRWuNIoXAzHZz5o= +github.com/benbjohnson/clock v1.3.5/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= +github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= +github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= +github.com/btcsuite/btcd/btcutil v1.1.6 h1:zFL2+c3Lb9gEgqKNzowKUPQNb8jV7v5Oaodi/AYFd6c= +github.com/btcsuite/btcd/btcutil v1.1.6/go.mod h1:9dFymx8HpuLqBnsPELrImQeTQfKBQqzqGbbV3jK55aE= +github.com/celestiaorg/go-header v0.6.2 h1:qgWyJQg+/x6k4QAfN1rPt2HXHZjQOmCqD0ct4dFBIZY= +github.com/celestiaorg/go-header v0.6.2/go.mod h1:Az4S4NxMOJ1eAzOaF8u5AZt5UzsSzg92uqpdXS3yOZE= +github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= +github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= +github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= +github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= +github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= +github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= +github.com/cockroachdb/datadriven v1.0.3-0.20230413201302-be42291fc80f h1:otljaYPt5hWxV3MUfO5dFPFiOXg9CyG5/kCfayTqsJ4= +github.com/cockroachdb/datadriven v1.0.3-0.20230413201302-be42291fc80f/go.mod h1:a9RdTaap04u637JoCzcUoIcDmvwSUtcUFtT/C3kJlTU= +github.com/cockroachdb/errors v1.11.3 h1:5bA+k2Y6r+oz/6Z/RFlNeVCesGARKuC6YymtcDrbC/I= +github.com/cockroachdb/errors v1.11.3/go.mod h1:m4UIW4CDjx+R5cybPsNrRbreomiFqt8o1h1wUVazSd8= +github.com/cockroachdb/fifo v0.0.0-20240606204812-0bbfbd93a7ce h1:giXvy4KSc/6g/esnpM7Geqxka4WSqI1SZc7sMJFd3y4= +github.com/cockroachdb/fifo v0.0.0-20240606204812-0bbfbd93a7ce/go.mod h1:9/y3cnZ5GKakj/H4y9r9GTjCvAFta7KLgSHPJJYc52M= +github.com/cockroachdb/logtags v0.0.0-20230118201751-21c54148d20b h1:r6VH0faHjZeQy818SGhaone5OnYfxFR/+AzdY3sf5aE= +github.com/cockroachdb/logtags v0.0.0-20230118201751-21c54148d20b/go.mod h1:Vz9DsVWQQhf3vs21MhPMZpMGSht7O/2vFW2xusFUVOs= +github.com/cockroachdb/pebble v1.1.1 h1:XnKU22oiCLy2Xn8vp1re67cXg4SAasg/WDt1NtcRFaw= +github.com/cockroachdb/pebble v1.1.1/go.mod h1:4exszw1r40423ZsmkG/09AFEG83I0uDgfujJdbL6kYU= +github.com/cockroachdb/redact v1.1.5 h1:u1PMllDkdFfPWaNGMyLD1+so+aq3uUItthCFqzwPJ30= +github.com/cockroachdb/redact v1.1.5/go.mod h1:BVNblN9mBWFyMyqK1k3AAiSxhvhfK2oOZZ2lK+dpvRg= +github.com/cockroachdb/tokenbucket v0.0.0-20230807174530-cc333fc44b06 h1:zuQyyAKVxetITBuuhv3BI9cMrmStnpT18zmgmTxunpo= +github.com/cockroachdb/tokenbucket v0.0.0-20230807174530-cc333fc44b06/go.mod h1:7nc4anLGjupUW/PeY5qiNYsdNXj7zopG+eqsS7To5IQ= +github.com/cometbft/cometbft v0.38.13 h1:k0ssyC8W0FfFvGpSHpojZ4JrXAANlJEIM41hjRq6OoU= +github.com/cometbft/cometbft v0.38.13/go.mod h1:NMmNysQM+T5uxTXVLMgzcvq35OVOyC7GK2/bnHqldjE= +github.com/cometbft/cometbft-db v0.14.1 h1:SxoamPghqICBAIcGpleHbmoPqy+crij/++eZz3DlerQ= +github.com/cometbft/cometbft-db v0.14.1/go.mod h1:KHP1YghilyGV/xjD5DP3+2hyigWx0WTp9X+0Gnx0RxQ= +github.com/containerd/cgroups v1.1.0 h1:v8rEWFl6EoqHB+swVNjVoCJE8o3jX7e8nqBGPLaDFBM= +github.com/containerd/cgroups v1.1.0/go.mod h1:6ppBcbh/NOOUU+dMKrykgaBnK9lCIBxHqJDGwsa1mIw= +github.com/coreos/go-systemd/v22 v22.5.0 h1:RrqgGjYQKalulkV8NGVIfkXQf6YYmOyiJKk8iXXhfZs= +github.com/coreos/go-systemd/v22 v22.5.0/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= +github.com/cosmos/gogoproto v1.7.0 h1:79USr0oyXAbxg3rspGh/m4SWNyoz/GLaAh0QlCe2fro= +github.com/cosmos/gogoproto v1.7.0/go.mod h1:yWChEv5IUEYURQasfyBW5ffkMHR/90hiHgbNgrtp4j0= +github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= +github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM= +github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davidlazar/go-crypto v0.0.0-20200604182044-b73af7476f6c h1:pFUpOrbxDR6AkioZ1ySsx5yxlDQZ8stG2b88gTPxgJU= +github.com/davidlazar/go-crypto v0.0.0-20200604182044-b73af7476f6c/go.mod h1:6UhI8N9EjYm1c2odKpFpAYeR8dsBeM7PtzQhRgxRr9U= +github.com/decred/dcrd/crypto/blake256 v1.0.1 h1:7PltbUIQB7u/FfZ39+DGa/ShuMyJ5ilcvdfma9wOH6Y= +github.com/decred/dcrd/crypto/blake256 v1.0.1/go.mod h1:2OfgNZ5wDpcsFmHmCK5gZTPcCXqlm2ArzUIkw9czNJo= +github.com/decred/dcrd/dcrec/secp256k1/v4 v4.3.0 h1:rpfIENRNNilwHwZeG5+P150SMrnNEcHYvcCuK6dPZSg= +github.com/decred/dcrd/dcrec/secp256k1/v4 v4.3.0/go.mod h1:v57UDF4pDQJcEfFUCRop3lJL149eHGSe9Jvczhzjo/0= +github.com/dgraph-io/badger/v4 v4.2.1-0.20231013074411-fb1b00959581 h1:yy45brf1ktmnkTCZlHynP1gRlVwZ9g19oz5D9wG81v4= +github.com/dgraph-io/badger/v4 v4.2.1-0.20231013074411-fb1b00959581/go.mod h1:T/uWAYxrXdaXw64ihI++9RMbKTCpKd/yE9+saARew7k= +github.com/dgraph-io/ristretto v0.1.1 h1:6CWw5tJNgpegArSHpNHJKldNeq03FQCwYvfMVWajOK8= +github.com/dgraph-io/ristretto v0.1.1/go.mod h1:S1GPSBCYCIhmVNfcth17y2zZtQT6wzkzgwUve0VDWWA= +github.com/dgryski/go-farm v0.0.0-20190423205320-6a90982ecee2/go.mod h1:SqUrOPUnsFjfmXRMNPybcSiG0BgUW2AuFH8PAnS2iTw= +github.com/dgryski/go-farm v0.0.0-20200201041132-a6ae2369ad13 h1:fAjc9m62+UWV/WAFKLNi6ZS0675eEUC9y3AlwSbQu1Y= +github.com/dgryski/go-farm v0.0.0-20200201041132-a6ae2369ad13/go.mod h1:SqUrOPUnsFjfmXRMNPybcSiG0BgUW2AuFH8PAnS2iTw= +github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4= +github.com/docker/go-units v0.5.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk= +github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= +github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY= +github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= +github.com/elastic/gosigar v0.14.3 h1:xwkKwPia+hSfg9GqrCUKYdId102m9qTJIIr7egmK/uo= +github.com/elastic/gosigar v0.14.3/go.mod h1:iXRIGg2tLnu7LBdpqzyQfGDEidKCfWcCMS0WKyPWoMs= +github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= +github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= +github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= +github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= +github.com/flynn/noise v1.1.0 h1:KjPQoQCEFdZDiP03phOvGi11+SVVhBG2wOWAorLsstg= +github.com/flynn/noise v1.1.0/go.mod h1:xbMo+0i6+IGbYdJhF31t2eR1BIU0CYc12+BNAKwUTag= +github.com/fortytw2/leaktest v1.3.0 h1:u8491cBMTQ8ft8aeV+adlcytMZylmA5nnwwkRZjI8vw= +github.com/fortytw2/leaktest v1.3.0/go.mod h1:jDsjWgpAGjm2CA7WthBh/CdZYEPF31XHquHwclZch5g= +github.com/francoispqt/gojay v1.2.13 h1:d2m3sFjloqoIUQU3TsHBgj6qg/BVGlTBeHDUmyJnXKk= +github.com/francoispqt/gojay v1.2.13/go.mod h1:ehT5mTG4ua4581f1++1WLG0vPdaA9HaiDsoyrBGkyDY= +github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= +github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= +github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA= +github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM= +github.com/getsentry/sentry-go v0.27.0 h1:Pv98CIbtB3LkMWmXi4Joa5OOcwbmnX88sF5qbK3r3Ps= +github.com/getsentry/sentry-go v0.27.0/go.mod h1:lc76E2QywIyW8WuBnwl8Lc4bkmQH4+w1gwTf25trprY= +github.com/go-errors/errors v1.4.2 h1:J6MZopCL4uSllY1OfXM374weqZFFItUbrImctkmUxIA= +github.com/go-errors/errors v1.4.2/go.mod h1:sIVyrIiJhuEF+Pj9Ebtd6P/rEYROXFi3BopGUQ5a5Og= +github.com/go-kit/kit v0.13.0 h1:OoneCcHKHQ03LfBpoQCUfCluwd2Vt3ohz+kvbJneZAU= +github.com/go-kit/kit v0.13.0/go.mod h1:phqEHMMUbyrCFCTgH48JueqrM3md2HcAZ8N3XE4FKDg= +github.com/go-kit/log v0.2.1 h1:MRVx0/zhvdseW+Gza6N9rVzU/IVzaeE1SFI4raAhmBU= +github.com/go-kit/log v0.2.1/go.mod h1:NwTd00d/i8cPZ3xOwwiv2PO5MOcx78fFErGNcVmBjv0= +github.com/go-logfmt/logfmt v0.6.0 h1:wGYYu3uicYdqXVgoYbvnkrPVXkuLM1p1ifugDMEdRi4= +github.com/go-logfmt/logfmt v0.6.0/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs= +github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 h1:tfuBGBXKqDEevZMzYi5KSi8KkcZtzBcTgAUUtapy0OI= +github.com/go-task/slim-sprig/v3 v3.0.0 h1:sUs3vkvUymDpBKi3qH1YSqBQk9+9D/8M2mN1vB6EwHI= +github.com/go-task/slim-sprig/v3 v3.0.0/go.mod h1:W848ghGpv3Qj3dhTPRyJypKRiqCdHZiAzKg9hl15HA8= +github.com/godbus/dbus/v5 v5.1.0 h1:4KLkAxT3aOY8Li4FRJe/KvhoNFFxo0m6fNuFUO8QJUk= +github.com/godbus/dbus/v5 v5.1.0/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= +github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= +github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= +github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= +github.com/golang/glog v1.2.2 h1:1+mZ9upx1Dh6FmUTFR1naJ77miKiXgALjWOZ3NVFPmY= +github.com/golang/glog v1.2.2/go.mod h1:6AhwSGph0fcJtXVM/PEHPqZlFeoLxhs7/t5UDAwmO+w= +github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da h1:oI5xCqsCo564l8iNU+DwB5epxmsaqB+rhGL0m5jtYqE= +github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= +github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= +github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= +github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= +github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= +github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= +github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= +github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= +github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= +github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= +github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= +github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek= +github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps= +github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= +github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= +github.com/google/btree v1.1.3 h1:CVpQJjYgC4VbzxeGVHfvZrv1ctoYCAI8vbl07Fcxlyg= +github.com/google/btree v1.1.3/go.mod h1:qOPhT0dTNdNzV6Z/lhRX0YXUafgPLFUh+gZMl761Gm4= +github.com/google/flatbuffers v1.12.1 h1:MVlul7pQNoDzWRLTw5imwYsl+usrS1TXG2H4jg6ImGw= +github.com/google/flatbuffers v1.12.1/go.mod h1:1AeVuKshWv4vARoZatz6mlQ0JxURH0Kv5+zNeJKJCa8= +github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= +github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= +github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= +github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= +github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/gopacket v1.1.19 h1:ves8RnFZPGiFnTS0uPQStjwru6uO6h+nlr9j6fL7kF8= +github.com/google/gopacket v1.1.19/go.mod h1:iJ8V8n6KS+z2U1A8pUwu8bW5SyEMkXJB8Yo/Vo+TKTo= +github.com/google/pprof v0.0.0-20240727154555-813a5fbdbec8 h1:FKHo8hFI3A+7w0aUQuYXQ+6EN5stWmeY/AZqtM8xk9k= +github.com/google/pprof v0.0.0-20240727154555-813a5fbdbec8/go.mod h1:K1liHPHnj73Fdn/EKuT8nrFqBihUSKXoLYU0BuatOYo= +github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= +github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg= +github.com/gorilla/websocket v1.5.3/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= +github.com/hashicorp/golang-lru/v2 v2.0.7 h1:a+bsQ5rvGLjzHuww6tVxozPZFVghXaHOwFs4luLUK2k= +github.com/hashicorp/golang-lru/v2 v2.0.7/go.mod h1:QeFd9opnmA6QUJc5vARoKUSoFhyfM2/ZepoAG6RGpeM= +github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= +github.com/huin/goupnp v1.3.0 h1:UvLUlWDNpoUdYzb2TCn+MuTWtcjXKSza2n6CBdQ0xXc= +github.com/huin/goupnp v1.3.0/go.mod h1:gnGPsThkYa7bFi/KWmEysQRf48l2dvR5bxr2OFckNX8= +github.com/ipfs/go-cid v0.4.1 h1:A/T3qGvxi4kpKWWcPC/PgbvDA2bjVLO7n4UeVwnbs/s= +github.com/ipfs/go-cid v0.4.1/go.mod h1:uQHwDeX4c6CtyrFwdqyhpNcxVewur1M7l7fNU7LKwZk= +github.com/ipfs/go-datastore v0.6.0 h1:JKyz+Gvz1QEZw0LsX1IBn+JFCJQH4SJVFtM4uWU0Myk= +github.com/ipfs/go-datastore v0.6.0/go.mod h1:rt5M3nNbSO/8q1t4LNkLyUwRs8HupMeN/8O4Vn9YAT8= +github.com/ipfs/go-detect-race v0.0.1 h1:qX/xay2W3E4Q1U7d9lNs1sU9nvguX0a7319XbyQ6cOk= +github.com/ipfs/go-detect-race v0.0.1/go.mod h1:8BNT7shDZPo99Q74BpGMK+4D8Mn4j46UU0LZ723meps= +github.com/ipfs/go-ds-badger4 v0.1.5 h1:MwrTsIUJIqH/ChuDdUOzxwxMxHx/Li1ECoSCKsCUxiA= +github.com/ipfs/go-ds-badger4 v0.1.5/go.mod h1:LUU2FbhNdmhAbJmMeoahVRbe4GsduAODSJHWJJh2Vo4= +github.com/ipfs/go-log/v2 v2.5.1 h1:1XdUzF7048prq4aBjDQQ4SL5RxftpRGdXhNRwKSAlcY= +github.com/ipfs/go-log/v2 v2.5.1/go.mod h1:prSpmC1Gpllc9UYWxDiZDreBYw7zp4Iqp1kOLU9U5UI= +github.com/jackpal/go-nat-pmp v1.0.2 h1:KzKSgb7qkJvOUTqYl9/Hg/me3pWgBmERKrTGD7BdWus= +github.com/jackpal/go-nat-pmp v1.0.2/go.mod h1:QPH045xvCAeXUZOxsnwmrtiCoxIr9eob+4orBN1SBKc= +github.com/jbenet/go-cienv v0.1.0/go.mod h1:TqNnHUmJgXau0nCzC7kXWeotg3J9W34CUv5Djy1+FlA= +github.com/jbenet/go-temp-err-catcher v0.1.0 h1:zpb3ZH6wIE8Shj2sKS+khgRvf7T7RABoLk/+KKHggpk= +github.com/jbenet/go-temp-err-catcher v0.1.0/go.mod h1:0kJRvmDZXNMIiJirNPEYfhpPwbGVtZVWC34vc5WLsDk= +github.com/jbenet/goprocess v0.1.4 h1:DRGOFReOMqqDNXwW70QkacFW0YN9QnwLV0Vqk+3oU0o= +github.com/jbenet/goprocess v0.1.4/go.mod h1:5yspPrukOVuOLORacaBi858NqyClJPQxYZlqdZVfqY4= +github.com/jmhodges/levigo v1.0.0 h1:q5EC36kV79HWeTBWsod3mG11EgStG3qArTKcvlksN1U= +github.com/jmhodges/levigo v1.0.0/go.mod h1:Q6Qx+uH3RAqyK4rFQroq9RL7mdkABMcfhEI+nNuzMJQ= +github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= +github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= +github.com/klauspost/compress v1.17.9 h1:6KIumPrER1LHsvBVuDa0r5xaG0Es51mhhB9BQB2qeMA= +github.com/klauspost/compress v1.17.9/go.mod h1:Di0epgTjJY877eYKx5yC51cX2A2Vl2ibi7bDH9ttBbw= +github.com/klauspost/cpuid/v2 v2.2.8 h1:+StwCXwm9PdpiEkPyzBXIy+M9KUb4ODm0Zarf1kS5BM= +github.com/klauspost/cpuid/v2 v2.2.8/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws= +github.com/koron/go-ssdp v0.0.4 h1:1IDwrghSKYM7yLf7XCzbByg2sJ/JcNOZRXS2jczTwz0= +github.com/koron/go-ssdp v0.0.4/go.mod h1:oDXq+E5IL5q0U8uSBcoAXzTzInwy5lEgC91HoKtbmZk= +github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= +github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= +github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= +github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= +github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= +github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= +github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/libp2p/go-buffer-pool v0.1.0 h1:oK4mSFcQz7cTQIfqbe4MIj9gLW+mnanjyFtc6cdF0Y8= +github.com/libp2p/go-buffer-pool v0.1.0/go.mod h1:N+vh8gMqimBzdKkSMVuydVDq+UV5QTWy5HSiZacSbPg= +github.com/libp2p/go-flow-metrics v0.1.0 h1:0iPhMI8PskQwzh57jB9WxIuIOQ0r+15PChFGkx3Q3WM= +github.com/libp2p/go-flow-metrics v0.1.0/go.mod h1:4Xi8MX8wj5aWNDAZttg6UPmc0ZrnFNsMtpsYUClFtro= +github.com/libp2p/go-libp2p v0.36.2 h1:BbqRkDaGC3/5xfaJakLV/BrpjlAuYqSB0lRvtzL3B/U= +github.com/libp2p/go-libp2p v0.36.2/go.mod h1:XO3joasRE4Eup8yCTTP/+kX+g92mOgRaadk46LmPhHY= +github.com/libp2p/go-libp2p-asn-util v0.4.1 h1:xqL7++IKD9TBFMgnLPZR6/6iYhawHKHl950SO9L6n94= +github.com/libp2p/go-libp2p-asn-util v0.4.1/go.mod h1:d/NI6XZ9qxw67b4e+NgpQexCIiFYJjErASrYW4PFDN8= +github.com/libp2p/go-libp2p-pubsub v0.12.0 h1:PENNZjSfk8KYxANRlpipdS7+BfLmOl3L2E/6vSNjbdI= +github.com/libp2p/go-libp2p-pubsub v0.12.0/go.mod h1:Oi0zw9aw8/Y5GC99zt+Ef2gYAl+0nZlwdJonDyOz/sE= +github.com/libp2p/go-libp2p-testing v0.12.0 h1:EPvBb4kKMWO29qP4mZGyhVzUyR25dvfUIK5WDu6iPUA= +github.com/libp2p/go-libp2p-testing v0.12.0/go.mod h1:KcGDRXyN7sQCllucn1cOOS+Dmm7ujhfEyXQL5lvkcPg= +github.com/libp2p/go-msgio v0.3.0 h1:mf3Z8B1xcFN314sWX+2vOTShIE0Mmn2TXn3YCUQGNj0= +github.com/libp2p/go-msgio v0.3.0/go.mod h1:nyRM819GmVaF9LX3l03RMh10QdOroF++NBbxAb0mmDM= +github.com/libp2p/go-nat v0.2.0 h1:Tyz+bUFAYqGyJ/ppPPymMGbIgNRH+WqC5QrT5fKrrGk= +github.com/libp2p/go-nat v0.2.0/go.mod h1:3MJr+GRpRkyT65EpVPBstXLvOlAPzUVlG6Pwg9ohLJk= +github.com/libp2p/go-netroute v0.2.1 h1:V8kVrpD8GK0Riv15/7VN6RbUQ3URNZVosw7H2v9tksU= +github.com/libp2p/go-netroute v0.2.1/go.mod h1:hraioZr0fhBjG0ZRXJJ6Zj2IVEVNx6tDTFQfSmcq7mQ= +github.com/libp2p/go-reuseport v0.4.0 h1:nR5KU7hD0WxXCJbmw7r2rhRYruNRl2koHw8fQscQm2s= +github.com/libp2p/go-reuseport v0.4.0/go.mod h1:ZtI03j/wO5hZVDFo2jKywN6bYKWLOy8Se6DrI2E1cLU= +github.com/libp2p/go-yamux/v4 v4.0.1 h1:FfDR4S1wj6Bw2Pqbc8Uz7pCxeRBPbwsBbEdfwiCypkQ= +github.com/libp2p/go-yamux/v4 v4.0.1/go.mod h1:NWjl8ZTLOGlozrXSOZ/HlfG++39iKNnM5wwmtQP1YB4= +github.com/linxGnu/grocksdb v1.8.14 h1:HTgyYalNwBSG/1qCQUIott44wU5b2Y9Kr3z7SK5OfGQ= +github.com/linxGnu/grocksdb v1.8.14/go.mod h1:QYiYypR2d4v63Wj1adOOfzglnoII0gLj3PNh4fZkcFA= +github.com/marten-seemann/tcp v0.0.0-20210406111302-dfbc87cc63fd h1:br0buuQ854V8u83wA0rVZ8ttrq5CpaPZdvrK0LP2lOk= +github.com/marten-seemann/tcp v0.0.0-20210406111302-dfbc87cc63fd/go.mod h1:QuCEs1Nt24+FYQEqAAncTDPJIuGs+LxK1MCiFL25pMU= +github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= +github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= +github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= +github.com/miekg/dns v1.1.62 h1:cN8OuEF1/x5Rq6Np+h1epln8OiyPWV+lROx9LxcGgIQ= +github.com/miekg/dns v1.1.62/go.mod h1:mvDlcItzm+br7MToIKqkglaGhlFMHJ9DTNNWONWXbNQ= +github.com/mikioh/tcpinfo v0.0.0-20190314235526-30a79bb1804b h1:z78hV3sbSMAUoyUMM0I83AUIT6Hu17AWfgjzIbtrYFc= +github.com/mikioh/tcpinfo v0.0.0-20190314235526-30a79bb1804b/go.mod h1:lxPUiZwKoFL8DUUmalo2yJJUCxbPKtm8OKfqr2/FTNU= +github.com/mikioh/tcpopt v0.0.0-20190314235656-172688c1accc h1:PTfri+PuQmWDqERdnNMiD9ZejrlswWrCpBEZgWOiTrc= +github.com/mikioh/tcpopt v0.0.0-20190314235656-172688c1accc/go.mod h1:cGKTAVKx4SxOuR/czcZ/E2RSJ3sfHs8FpHhQ5CWMf9s= +github.com/minio/blake2b-simd v0.0.0-20160723061019-3f5f724cb5b1/go.mod h1:pD8RvIylQ358TN4wwqatJ8rNavkEINozVn9DtGI3dfQ= +github.com/minio/sha256-simd v0.1.1-0.20190913151208-6de447530771/go.mod h1:B5e1o+1/KgNmWrSQK08Y6Z1Vb5pwIktudl0J58iy0KM= +github.com/minio/sha256-simd v1.0.1 h1:6kaan5IFmwTNynnKKpDHe6FWHohJOHhCPchzK49dzMM= +github.com/minio/sha256-simd v1.0.1/go.mod h1:Pz6AKMiUdngCLpeTL/RJY1M9rUuPMYujV5xJjtbRSN8= +github.com/mr-tron/base58 v1.1.2/go.mod h1:BinMc/sQntlIE1frQmRFPUoPA1Zkr8VRgBdjWI2mNwc= +github.com/mr-tron/base58 v1.2.0 h1:T/HDJBh4ZCPbU39/+c3rRvE0uKBQlU27+QI8LJ4t64o= +github.com/mr-tron/base58 v1.2.0/go.mod h1:BinMc/sQntlIE1frQmRFPUoPA1Zkr8VRgBdjWI2mNwc= +github.com/multiformats/go-base32 v0.1.0 h1:pVx9xoSPqEIQG8o+UbAe7DNi51oej1NtK+aGkbLYxPE= +github.com/multiformats/go-base32 v0.1.0/go.mod h1:Kj3tFY6zNr+ABYMqeUNeGvkIC/UYgtWibDcT0rExnbI= +github.com/multiformats/go-base36 v0.2.0 h1:lFsAbNOGeKtuKozrtBsAkSVhv1p9D0/qedU9rQyccr0= +github.com/multiformats/go-base36 v0.2.0/go.mod h1:qvnKE++v+2MWCfePClUEjE78Z7P2a1UV0xHgWc0hkp4= +github.com/multiformats/go-multiaddr v0.1.1/go.mod h1:aMKBKNEYmzmDmxfX88/vz+J5IU55txyt0p4aiWVohjo= +github.com/multiformats/go-multiaddr v0.13.0 h1:BCBzs61E3AGHcYYTv8dqRH43ZfyrqM8RXVPT8t13tLQ= +github.com/multiformats/go-multiaddr v0.13.0/go.mod h1:sBXrNzucqkFJhvKOiwwLyqamGa/P5EIXNPLovyhQCII= +github.com/multiformats/go-multiaddr-dns v0.3.1 h1:QgQgR+LQVt3NPTjbrLLpsaT2ufAA2y0Mkk+QRVJbW3A= +github.com/multiformats/go-multiaddr-dns v0.3.1/go.mod h1:G/245BRQ6FJGmryJCrOuTdB37AMA5AMOVuO6NY3JwTk= +github.com/multiformats/go-multiaddr-fmt v0.1.0 h1:WLEFClPycPkp4fnIzoFoV9FVd49/eQsuaL3/CWe167E= +github.com/multiformats/go-multiaddr-fmt v0.1.0/go.mod h1:hGtDIW4PU4BqJ50gW2quDuPVjyWNZxToGUh/HwTZYJo= +github.com/multiformats/go-multibase v0.2.0 h1:isdYCVLvksgWlMW9OZRYJEa9pZETFivncJHmHnnd87g= +github.com/multiformats/go-multibase v0.2.0/go.mod h1:bFBZX4lKCA/2lyOFSAoKH5SS6oPyjtnzK/XTFDPkNuk= +github.com/multiformats/go-multicodec v0.9.0 h1:pb/dlPnzee/Sxv/j4PmkDRxCOi3hXTz3IbPKOXWJkmg= +github.com/multiformats/go-multicodec v0.9.0/go.mod h1:L3QTQvMIaVBkXOXXtVmYE+LI16i14xuaojr/H7Ai54k= +github.com/multiformats/go-multihash v0.0.8/go.mod h1:YSLudS+Pi8NHE7o6tb3D8vrpKa63epEDmG8nTduyAew= +github.com/multiformats/go-multihash v0.2.3 h1:7Lyc8XfX/IY2jWb/gI7JP+o7JEq9hOa7BFvVU9RSh+U= +github.com/multiformats/go-multihash v0.2.3/go.mod h1:dXgKXCXjBzdscBLk9JkjINiEsCKRVch90MdaGiKsvSM= +github.com/multiformats/go-multistream v0.5.0 h1:5htLSLl7lvJk3xx3qT/8Zm9J4K8vEOf/QGkvOGQAyiE= +github.com/multiformats/go-multistream v0.5.0/go.mod h1:n6tMZiwiP2wUsR8DgfDWw1dydlEqV3l6N3/GBsX6ILA= +github.com/multiformats/go-varint v0.0.7 h1:sWSGR+f/eu5ABZA2ZpYKBILXTTs9JWpdEM/nEGOHFS8= +github.com/multiformats/go-varint v0.0.7/go.mod h1:r8PUYw/fD/SjBCiKOoDlGF6QawOELpZAu9eioSos/OU= +github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 h1:C3w9PqII01/Oq1c1nUAm88MOHcQC9l5mIlSMApZMrHA= +github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ= +github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= +github.com/nxadm/tail v1.4.11 h1:8feyoE3OzPrcshW5/MJ4sGESc5cqmGkGCWlco4l0bqY= +github.com/nxadm/tail v1.4.11/go.mod h1:OTaG3NK980DZzxbRq6lEuzgU+mug70nY11sMd4JXXHc= +github.com/oasisprotocol/curve25519-voi v0.0.0-20220708102147-0a8a51822cae h1:FatpGJD2jmJfhZiFDElaC0QhZUDQnxUeAwTGkfAHN3I= +github.com/oasisprotocol/curve25519-voi v0.0.0-20220708102147-0a8a51822cae/go.mod h1:hVoHR2EVESiICEMbg137etN/Lx+lSrHPTD39Z/uE+2s= +github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= +github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= +github.com/onsi/ginkgo v1.14.0 h1:2mOpI4JVVPBN+WQRa0WKH2eXR+Ey+uK4n7Zj0aYpIQA= +github.com/onsi/ginkgo v1.14.0/go.mod h1:iSB4RoI2tjJc9BBv4NKIKWKya62Rps+oPG/Lv9klQyY= +github.com/onsi/ginkgo/v2 v2.20.0 h1:PE84V2mHqoT1sglvHc8ZdQtPcwmvvt29WLEEO3xmdZw= +github.com/onsi/ginkgo/v2 v2.20.0/go.mod h1:lG9ey2Z29hR41WMVthyJBGUBcBhGOtoPF2VFMvBXFCI= +github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= +github.com/onsi/gomega v1.10.1 h1:o0+MgICZLuZ7xjH7Vx6zS/zcu93/BEp1VwkIW1mEXCE= +github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= +github.com/opencontainers/runtime-spec v1.2.0 h1:z97+pHb3uELt/yiAWD691HNHQIF07bE7dzrbT927iTk= +github.com/opencontainers/runtime-spec v1.2.0/go.mod h1:jwyrGlmzljRJv/Fgzds9SsS/C5hL+LL3ko9hs6T5lQ0= +github.com/pbnjay/memory v0.0.0-20210728143218-7b4eea64cf58 h1:onHthvaw9LFnH4t2DcNVpwGmV9E1BkGknEliJkfwQj0= +github.com/pbnjay/memory v0.0.0-20210728143218-7b4eea64cf58/go.mod h1:DXv8WO4yhMYhSNPKjeNKa5WY9YCIEBRbNzFFPJbWO6Y= +github.com/petermattis/goid v0.0.0-20240813172612-4fcff4a6cae7 h1:Dx7Ovyv/SFnMFw3fD4oEoeorXc6saIiQ23LrGLth0Gw= +github.com/petermattis/goid v0.0.0-20240813172612-4fcff4a6cae7/go.mod h1:pxMtw7cyUw6B2bRH0ZBANSPg+AoSud1I1iyJHI69jH4= +github.com/pingcap/errors v0.11.4 h1:lFuQV/oaUMGcD2tqt+01ROSmJs75VG1ToEOkZIZ4nE4= +github.com/pingcap/errors v0.11.4/go.mod h1:Oi8TUi2kEtXXLMJk9l1cGmz20kV3TaQ0usTwv5KuLY8= +github.com/pion/datachannel v1.5.8 h1:ph1P1NsGkazkjrvyMfhRBUAWMxugJjq2HfQifaOoSNo= +github.com/pion/datachannel v1.5.8/go.mod h1:PgmdpoaNBLX9HNzNClmdki4DYW5JtI7Yibu8QzbL3tI= +github.com/pion/dtls/v2 v2.2.12 h1:KP7H5/c1EiVAAKUmXyCzPiQe5+bCJrpOeKg/L05dunk= +github.com/pion/dtls/v2 v2.2.12/go.mod h1:d9SYc9fch0CqK90mRk1dC7AkzzpwJj6u2GU3u+9pqFE= +github.com/pion/ice/v2 v2.3.34 h1:Ic1ppYCj4tUOcPAp76U6F3fVrlSw8A9JtRXLqw6BbUM= +github.com/pion/ice/v2 v2.3.34/go.mod h1:mBF7lnigdqgtB+YHkaY/Y6s6tsyRyo4u4rPGRuOjUBQ= +github.com/pion/interceptor v0.1.30 h1:au5rlVHsgmxNi+v/mjOPazbW1SHzfx7/hYOEYQnUcxA= +github.com/pion/interceptor v0.1.30/go.mod h1:RQuKT5HTdkP2Fi0cuOS5G5WNymTjzXaGF75J4k7z2nc= +github.com/pion/logging v0.2.2 h1:M9+AIj/+pxNsDfAT64+MAVgJO0rsyLnoJKCqf//DoeY= +github.com/pion/logging v0.2.2/go.mod h1:k0/tDVsRCX2Mb2ZEmTqNa7CWsQPc+YYCB7Q+5pahoms= +github.com/pion/mdns v0.0.12 h1:CiMYlY+O0azojWDmxdNr7ADGrnZ+V6Ilfner+6mSVK8= +github.com/pion/mdns v0.0.12/go.mod h1:VExJjv8to/6Wqm1FXK+Ii/Z9tsVk/F5sD/N70cnYFbk= +github.com/pion/randutil v0.1.0 h1:CFG1UdESneORglEsnimhUjf33Rwjubwj6xfiOXBa3mA= +github.com/pion/randutil v0.1.0/go.mod h1:XcJrSMMbbMRhASFVOlj/5hQial/Y8oH/HVo7TBZq+j8= +github.com/pion/rtcp v1.2.14 h1:KCkGV3vJ+4DAJmvP0vaQShsb0xkRfWkO540Gy102KyE= +github.com/pion/rtcp v1.2.14/go.mod h1:sn6qjxvnwyAkkPzPULIbVqSKI5Dv54Rv7VG0kNxh9L4= +github.com/pion/rtp v1.8.9 h1:E2HX740TZKaqdcPmf4pw6ZZuG8u5RlMMt+l3dxeu6Wk= +github.com/pion/rtp v1.8.9/go.mod h1:pBGHaFt/yW7bf1jjWAoUjpSNoDnw98KTMg+jWWvziqU= +github.com/pion/sctp v1.8.33 h1:dSE4wX6uTJBcNm8+YlMg7lw1wqyKHggsP5uKbdj+NZw= +github.com/pion/sctp v1.8.33/go.mod h1:beTnqSzewI53KWoG3nqB282oDMGrhNxBdb+JZnkCwRM= +github.com/pion/sdp/v3 v3.0.9 h1:pX++dCHoHUwq43kuwf3PyJfHlwIj4hXA7Vrifiq0IJY= +github.com/pion/sdp/v3 v3.0.9/go.mod h1:B5xmvENq5IXJimIO4zfp6LAe1fD9N+kFv+V/1lOdz8M= +github.com/pion/srtp/v2 v2.0.20 h1:HNNny4s+OUmG280ETrCdgFndp4ufx3/uy85EawYEhTk= +github.com/pion/srtp/v2 v2.0.20/go.mod h1:0KJQjA99A6/a0DOVTu1PhDSw0CXF2jTkqOoMg3ODqdA= +github.com/pion/stun v0.6.1 h1:8lp6YejULeHBF8NmV8e2787BogQhduZugh5PdhDyyN4= +github.com/pion/stun v0.6.1/go.mod h1:/hO7APkX4hZKu/D0f2lHzNyvdkTGtIy3NDmLR7kSz/8= +github.com/pion/transport/v2 v2.2.10 h1:ucLBLE8nuxiHfvkFKnkDQRYWYfp8ejf4YBOPfaQpw6Q= +github.com/pion/transport/v2 v2.2.10/go.mod h1:sq1kSLWs+cHW9E+2fJP95QudkzbK7wscs8yYgQToO5E= +github.com/pion/turn/v2 v2.1.6 h1:Xr2niVsiPTB0FPtt+yAWKFUkU1eotQbGgpTIld4x1Gc= +github.com/pion/turn/v2 v2.1.6/go.mod h1:huEpByKKHix2/b9kmTAM3YoX6MKP+/D//0ClgUYR2fY= +github.com/pion/webrtc/v3 v3.3.0 h1:Rf4u6n6U5t5sUxhYPQk/samzU/oDv7jk6BA5hyO2F9I= +github.com/pion/webrtc/v3 v3.3.0/go.mod h1:hVmrDJvwhEertRWObeb1xzulzHGeVUoPlWvxdGzcfU0= +github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= +github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= +github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U= +github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/prometheus/client_golang v1.20.4 h1:Tgh3Yr67PaOv/uTqloMsCEdeuFTatm5zIq5+qNN23vI= +github.com/prometheus/client_golang v1.20.4/go.mod h1:PIEt8X02hGcP8JWbeHyeZ53Y/jReSnHgO035n//V5WE= +github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.6.1 h1:ZKSh/rekM+n3CeS952MLRAdFwIKqeY8b62p8ais2e9E= +github.com/prometheus/client_model v0.6.1/go.mod h1:OrxVMOVHjw3lKMa8+x6HeMGkHMQyHDk9E3jmP2AmGiY= +github.com/prometheus/common v0.59.1 h1:LXb1quJHWm1P6wq/U824uxYi4Sg0oGvNeUm1z5dJoX0= +github.com/prometheus/common v0.59.1/go.mod h1:GpWM7dewqmVYcd7SmRaiWVe9SSqjf0UrwnYnpEZNuT0= +github.com/prometheus/procfs v0.15.1 h1:YagwOFzUgYfKKHX6Dr+sHT7km/hxC76UB0learggepc= +github.com/prometheus/procfs v0.15.1/go.mod h1:fB45yRUv8NstnjriLhBQLuOUt+WW4BsoGhij/e3PBqk= +github.com/quic-go/qpack v0.4.0 h1:Cr9BXA1sQS2SmDUWjSofMPNKmvF6IiIfDRmgU0w1ZCo= +github.com/quic-go/qpack v0.4.0/go.mod h1:UZVnYIfi5GRk+zI9UMaCPsmZ2xKJP7XBUvVyT1Knj9A= +github.com/quic-go/quic-go v0.46.0 h1:uuwLClEEyk1DNvchH8uCByQVjo3yKL9opKulExNDs7Y= +github.com/quic-go/quic-go v0.46.0/go.mod h1:1dLehS7TIR64+vxGR70GDcatWTOtMX2PUtnKsjbTurI= +github.com/quic-go/webtransport-go v0.8.0 h1:HxSrwun11U+LlmwpgM1kEqIqH90IT4N8auv/cD7QFJg= +github.com/quic-go/webtransport-go v0.8.0/go.mod h1:N99tjprW432Ut5ONql/aUhSLT0YVSlwHohQsuac9WaM= +github.com/raulk/go-watchdog v1.3.0 h1:oUmdlHxdkXRJlwfG0O9omj8ukerm8MEQavSiDTEtBsk= +github.com/raulk/go-watchdog v1.3.0/go.mod h1:fIvOnLbF0b0ZwkB9YU4mOW9Did//4vPZtDqv66NfsMU= +github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= +github.com/rogpeppe/go-internal v1.11.0 h1:cWPaGQEPrBb5/AsnsZesgZZ9yb1OQ+GOISoDNXVBh4M= +github.com/rogpeppe/go-internal v1.11.0/go.mod h1:ddIwULY96R17DhadqLgMfk9H9tvdUzkipdSkR5nkCZA= +github.com/rollkit/go-sequencing v0.2.1-0.20241010053131-3134457dc4e5 h1:4qHTCZsG81CE2QvvDeu2xCSIG1fkDyYbNgGwrBx98XA= +github.com/rollkit/go-sequencing v0.2.1-0.20241010053131-3134457dc4e5/go.mod h1:P/cQXTw3rWpPqhqnCwKzlkS39XM8ugmyf2u63twBgG8= +github.com/rollkit/rollkit v0.13.8-0.20241021170619-fc65faf646a7 h1:CMcpYHFfwLHFCvbnQw+4FiINUYa+y6N96EjtiSNJGoc= +github.com/rollkit/rollkit v0.13.8-0.20241021170619-fc65faf646a7/go.mod h1:f9p7jooXjJYIndjXkaTU13jBGTlqk2Y2OL9UoxADY20= +github.com/sasha-s/go-deadlock v0.3.5 h1:tNCOEEDG6tBqrNDOX35j/7hL5FcFViG6awUGROb2NsU= +github.com/sasha-s/go-deadlock v0.3.5/go.mod h1:bugP6EGbdGYObIlx7pUZtWqlvo8k9H6vCBBsiChJQ5U= +github.com/spaolacci/murmur3 v1.1.0 h1:7c1g84S4BPRrfL5Xrdp6fOJ206sU9y293DDHaoy0bLI= +github.com/spaolacci/murmur3 v1.1.0/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= +github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= +github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= +github.com/stretchr/objx v0.5.2 h1:xuMeJ0Sdp5ZMRXx/aWO6RZxdr3beISkG5/G/aIRr3pY= +github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA= +github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= +github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= +github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= +github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= +github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7 h1:epCh84lMvA70Z7CTTCmYQn2CKbY8j86K7/FAIr141uY= +github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7/go.mod h1:q4W45IWZaF22tdD+VEXcAWRA037jwmWEB5VWYORlTpc= +github.com/wlynxg/anet v0.0.4 h1:0de1OFQxnNqAu+x2FAKKCVIrnfGKQbs7FQz++tB0+Uw= +github.com/wlynxg/anet v0.0.4/go.mod h1:eay5PRQr7fIVAMbTbchTnO9gG65Hg/uYGdc7mguHxoA= +github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= +go.etcd.io/bbolt v1.4.0-alpha.0.0.20240404170359-43604f3112c5 h1:qxen9oVGzDdIRP6ejyAJc760RwW4SnVDiTYTzwnXuxo= +go.etcd.io/bbolt v1.4.0-alpha.0.0.20240404170359-43604f3112c5/go.mod h1:eW0HG9/oHQhvRCvb1/pIXW4cOvtDqeQK+XSi3TnwaXY= +go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0= +go.opencensus.io v0.24.0/go.mod h1:vNK8G9p7aAivkbmorf4v+7Hgx+Zs0yY+0fOtgBfjQKo= +go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= +go.uber.org/dig v1.18.0 h1:imUL1UiY0Mg4bqbFfsRQO5G4CGRBec/ZujWTvSVp3pw= +go.uber.org/dig v1.18.0/go.mod h1:Us0rSJiThwCv2GteUN0Q7OKvU7n5J4dxZ9JKUXozFdE= +go.uber.org/fx v1.22.2 h1:iPW+OPxv0G8w75OemJ1RAnTUrF55zOJlXlo1TbJ0Buw= +go.uber.org/fx v1.22.2/go.mod h1:o/D9n+2mLP6v1EG+qsdT1O8wKopYAsqZasju97SDFCU= +go.uber.org/goleak v1.1.11-0.20210813005559-691160354723/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= +go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= +go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= +go.uber.org/mock v0.4.0 h1:VcM4ZOtdbR4f6VXfiOpwpVJDL6lCReaZ6mw31wqh7KU= +go.uber.org/mock v0.4.0/go.mod h1:a6FSlNadKUHUa9IP5Vyt1zh4fC7uAwxMutEAscFbkZc= +go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= +go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= +go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y= +go.uber.org/zap v1.19.1/go.mod h1:j3DNczoxDZroyBnOT1L/Q79cfUMGZxlv/9dzN7SM1rI= +go.uber.org/zap v1.27.0 h1:aJMhYGrd5QSmlpLMr2MftRKl7t8J8PTZPA732ud/XR8= +go.uber.org/zap v1.27.0/go.mod h1:GB2qFLM7cTU87MWRP2mPIjqfIDnGu+VIO4V/SdhGo2E= +golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20190611184440-5c40567a22f8/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.27.0 h1:GXm2NjJrPaiv/h1tb2UH8QfgC/hOf/+z0p6PT8o1w7A= +golang.org/x/crypto v0.27.0/go.mod h1:1Xngt8kV6Dvbssa53Ziq6Eqn0HqbZi5Z6R0ZpwQzt70= +golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= +golang.org/x/exp v0.0.0-20240808152545-0cdaa3abc0fa h1:ELnwvuAXPNtPk1TJRuGkI9fDTwym6AYBu0qzT8AcHdI= +golang.org/x/exp v0.0.0-20240808152545-0cdaa3abc0fa/go.mod h1:akd2r19cwCdwSwWeIdzYQGa/EZZyqcOdwWiwj5L5eKQ= +golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= +golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= +golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= +golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= +golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.20.0 h1:utOm6MM3R3dnawAiJgn0y+xvuYRsm1RKM/4giyfDgV0= +golang.org/x/mod v0.20.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c= +golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= +golang.org/x/net v0.0.0-20200813134508-3edf25e44fcc/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= +golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= +golang.org/x/net v0.29.0 h1:5ORfpBpCs4HzDYoodCDBbwHzdR5UrLBZ3sOnUJmFoHo= +golang.org/x/net v0.29.0/go.mod h1:gLkgy8jTGERgjzMic6DS9+SP0ajcu6Xu3Orq/SpETg0= +golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= +golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ= +golang.org/x/sync v0.8.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= +golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200519105757-fe76b779f299/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200814200057-3d37ad5750ed/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20221010170243-090e33056c14/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.25.0 h1:r+8e+loiHxRqhXVl6ML1nO3l1+oFoWbnlu2Ehimmi34= +golang.org/x/sys v0.25.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= +golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= +golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.18.0 h1:XvMDiNzPAl0jr17s6W9lcaIhGUfUORdGCNsuLmPG224= +golang.org/x/text v0.18.0/go.mod h1:BuEKDfySbSR4drPmRPG/7iBdf8hvFMuRexcpahXilzY= +golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= +golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= +golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.24.0 h1:J1shsA93PJUEVaUSaay7UXAyE8aimq3GW0pjlolpa24= +golang.org/x/tools v0.24.0/go.mod h1:YhNqVBIfWHdzvTLs0d8LCuMhkKUgSUKldakyV7W/WDQ= +golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20231012003039-104605ab7028 h1:+cNy6SZtPcJQH3LJVLOSmiC7MMxXNOb3PU/VUEz+EhU= +golang.org/x/xerrors v0.0.0-20231012003039-104605ab7028/go.mod h1:NDW/Ps6MPRej6fsCIbMTohpP40sJ/P/vI1MoTEGwX90= +google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= +google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= +google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= +google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= +google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240827150818-7e3bb234dfed h1:J6izYgfBXAI3xTKLgxzTmUltdYaLsuBxFCgDHWJ/eXg= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240827150818-7e3bb234dfed/go.mod h1:UqMtugtsSgubUsoxbuAoiCXvqvErP7Gf0so0mK9tHxU= +google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= +google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= +google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= +google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= +google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= +google.golang.org/grpc v1.67.1 h1:zWnc1Vrcno+lHZCOofnIMvycFcc0QRGIzm9dhnDX68E= +google.golang.org/grpc v1.67.1/go.mod h1:1gLDyUQU7CTLJI90u3nXZ9ekeghjeM7pTDZlqFNg2AA= +google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= +google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= +google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= +google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= +google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= +google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= +google.golang.org/protobuf v1.35.1 h1:m3LfL6/Ca+fqnjnlqQXNpFPABW1UD7mjh8KO2mKFytA= +google.golang.org/protobuf v1.35.1/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= +gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= +gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= +gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= +gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +lukechampine.com/blake3 v1.3.0 h1:sJ3XhFINmHSrYCgl958hscfIa3bw8x4DqMP3u1YvoYE= +lukechampine.com/blake3 v1.3.0/go.mod h1:0OFRp7fBtAylGVCO40o87sbupkyIGgbpv1+M1k1LM6k= From 29b6d20dab7648149150e70f84985e8ef71e226c Mon Sep 17 00:00:00 2001 From: Jay Jie Date: Thu, 24 Oct 2024 22:14:13 -0700 Subject: [PATCH 02/10] chore: update module path in go.mod --- go.mod | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/go.mod b/go.mod index 2cbb42e..63aa975 100644 --- a/go.mod +++ b/go.mod @@ -1,4 +1,4 @@ -module github.com/rollkit/template-da-repo +module github.com/LastL2/go-execution-abci go 1.22.7 From 5a28a5d4527510cdcde5e8828cdbb389f6b0c558 Mon Sep 17 00:00:00 2001 From: Jay Jie Date: Mon, 28 Oct 2024 21:55:41 -0700 Subject: [PATCH 03/10] chore: change github org name --- execution.go | 2 +- go.mod | 6 +++--- go.sum | 4 ++-- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/execution.go b/execution.go index cb0df80..ca0213d 100644 --- a/execution.go +++ b/execution.go @@ -7,12 +7,12 @@ import ( "fmt" "time" - execution "github.com/LastL2/go-execution" abci "github.com/cometbft/cometbft/abci/types" cmbytes "github.com/cometbft/cometbft/libs/bytes" cmproto "github.com/cometbft/cometbft/proto/tendermint/types" "github.com/cometbft/cometbft/proxy" cmtypes "github.com/cometbft/cometbft/types" + execution "github.com/rollkit/go-execution" "github.com/rollkit/rollkit/mempool" "github.com/rollkit/rollkit/state" "github.com/rollkit/rollkit/store" diff --git a/go.mod b/go.mod index 63aa975..c02dd79 100644 --- a/go.mod +++ b/go.mod @@ -1,14 +1,14 @@ -module github.com/LastL2/go-execution-abci +module github.com/rollkit/go-execution-abci go 1.22.7 toolchain go1.22.8 -replace github.com/LastL2/go-execution => github.com/LastL2/go-execution v0.0.0-20241025044830-6028e95ddb3a +replace github.com/rollkit/go-execution => github.com/lastdotnet/go-execution v0.0.0-20241029045146-b7513b533b24 require ( - github.com/LastL2/go-execution v0.0.0-00010101000000-000000000000 github.com/cometbft/cometbft v0.38.13 + github.com/rollkit/go-execution v0.0.0-00010101000000-000000000000 github.com/rollkit/rollkit v0.13.8-0.20241021170619-fc65faf646a7 ) diff --git a/go.sum b/go.sum index 02dbdbf..50e413b 100644 --- a/go.sum +++ b/go.sum @@ -2,8 +2,6 @@ cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMT github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= github.com/DataDog/zstd v1.4.5 h1:EndNeuB0l9syBZhut0wns3gV1hL8zX8LIu6ZiVHWLIQ= github.com/DataDog/zstd v1.4.5/go.mod h1:1jcaCB/ufaK+sKp1NBhlGmpz41jOoPQ35bpF36t7BBo= -github.com/LastL2/go-execution v0.0.0-20241025044830-6028e95ddb3a h1:l3EnFurDODyP+abpbtfb6v/F7TQmkMIYKRVu6xygO9c= -github.com/LastL2/go-execution v0.0.0-20241025044830-6028e95ddb3a/go.mod h1:PM6JVuwMV/RZa3SYq6vznGpzcfKFy6KYaDauAPQ13qc= github.com/VividCortex/gohistogram v1.0.0 h1:6+hBz+qvs0JOrrNhhmR7lFxo5sINxBCGXrdtl/UvroE= github.com/VividCortex/gohistogram v1.0.0/go.mod h1:Pf5mBqqDxYaXu3hDrrU+w6nw50o/4+TcAqDqk/vUH7g= github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= @@ -182,6 +180,8 @@ github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/lastdotnet/go-execution v0.0.0-20241029045146-b7513b533b24 h1:F/XQxFKbZA1w8daNqHdh/8waKQfWLAyZTmkQVVxlgVk= +github.com/lastdotnet/go-execution v0.0.0-20241029045146-b7513b533b24/go.mod h1:fo+uH57fdmebAEYEVtLhsSJ7Jw8CKRSydCnyqJZRBAA= github.com/libp2p/go-buffer-pool v0.1.0 h1:oK4mSFcQz7cTQIfqbe4MIj9gLW+mnanjyFtc6cdF0Y8= github.com/libp2p/go-buffer-pool v0.1.0/go.mod h1:N+vh8gMqimBzdKkSMVuydVDq+UV5QTWy5HSiZacSbPg= github.com/libp2p/go-flow-metrics v0.1.0 h1:0iPhMI8PskQwzh57jB9WxIuIOQ0r+15PChFGkx3Q3WM= From 3b5211afd15148fdd14d333c933eaf95e86e3720 Mon Sep 17 00:00:00 2001 From: Jay Jie Date: Tue, 29 Oct 2024 21:48:45 -0700 Subject: [PATCH 04/10] feat: support json rpc proxy --- README.md | 25 ++ execution.go | 707 +--------------------------------------------- proxy/client.go | 731 ++++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 771 insertions(+), 692 deletions(-) create mode 100644 README.md create mode 100644 proxy/client.go diff --git a/README.md b/README.md new file mode 100644 index 0000000..5de9bfa --- /dev/null +++ b/README.md @@ -0,0 +1,25 @@ +## Architecture + +```mermaid +graph TB + subgraph go-execution + R[Execute Interface] + end + subgraph "go-execution-abci" + AEC[ABCIExecutionClient] + PC[ProxyClient] + end + subgraph "CometBFT" + ABCI[ABCI Application] + MP[Mempool] + EB[EventBus] + end + R -->|implements| AEC + AEC -->|Delegates| PC + PC -->|ABCI Calls| ABCI + PC -->|Tx Management| MP + PC -->|Events| EB + classDef default fill:#f9f9f9,stroke:#333,stroke-width:2px; + classDef highlight fill:#e1f7d5,stroke:#333,stroke-width:2px; + class AEC,PC highlight; +``` diff --git a/execution.go b/execution.go index ca0213d..8a36079 100644 --- a/execution.go +++ b/execution.go @@ -1,64 +1,19 @@ package execution import ( - "bytes" - "context" - "errors" - "fmt" "time" - abci "github.com/cometbft/cometbft/abci/types" - cmbytes "github.com/cometbft/cometbft/libs/bytes" - cmproto "github.com/cometbft/cometbft/proto/tendermint/types" - "github.com/cometbft/cometbft/proxy" - cmtypes "github.com/cometbft/cometbft/types" execution "github.com/rollkit/go-execution" - "github.com/rollkit/rollkit/mempool" - "github.com/rollkit/rollkit/state" - "github.com/rollkit/rollkit/store" - "github.com/rollkit/rollkit/third_party/log" - "github.com/rollkit/rollkit/types" - abciconv "github.com/rollkit/rollkit/types/abci" + rollkitTypes "github.com/rollkit/rollkit/types" ) -// ErrEmptyValSetGenerated is returned when applying the validator changes would result in empty set. -var ErrEmptyValSetGenerated = errors.New("applying the validator changes would result in empty set") - -// ErrAddingValidatorToBased is returned when trying to add a validator to an empty validator set. -var ErrAddingValidatorToBased = errors.New("cannot add validators to empty validator set") - type ABCIExecutionClient struct { - // abci specific - proxyApp proxy.AppConnConsensus - eventBus *cmtypes.EventBus - genesis *cmtypes.GenesisDoc - maxBytes uint64 - proposerAddress []byte - chainID string - - // rollkit specific - mempool mempool.Mempool - mempoolReaper *mempool.CListMempoolReaper - logger log.Logger - metrics *state.Metrics - state *types.State - store store.Store + execute execution.Execute } -func NewABCIExecutionClient(proposerAddress []byte, chainID string, mempool mempool.Mempool, mempoolReaper *mempool.CListMempoolReaper, proxyApp proxy.AppConnConsensus, eventBus *cmtypes.EventBus, maxBytes uint64, logger log.Logger, metrics *state.Metrics, store store.Store, genesis *cmtypes.GenesisDoc, state *types.State) *ABCIExecutionClient { +func NewABCIExecutionClient(execute execution.Execute) *ABCIExecutionClient { return &ABCIExecutionClient{ - proxyApp: proxyApp, - eventBus: eventBus, - genesis: genesis, - maxBytes: maxBytes, - proposerAddress: proposerAddress, - chainID: chainID, - mempool: mempool, - mempoolReaper: mempoolReaper, - logger: logger, - metrics: metrics, - store: store, - state: state, + execute: execute, } } @@ -69,658 +24,26 @@ func (c *ABCIExecutionClient) InitChain( genesisTime time.Time, initialHeight uint64, chainID string, -) (types.Hash, uint64, error) { - genesis := &cmtypes.GenesisDoc{ - GenesisTime: genesisTime, - ChainID: chainID, - ConsensusParams: c.genesis.ConsensusParams, - Validators: c.genesis.Validators, - AppState: c.genesis.AppState, - InitialHeight: int64(initialHeight), - } - - response, err := c.initChain(genesis) - if err != nil { - return types.Hash{}, 0, err - } - - stateRoot := types.Hash(response.AppHash) - maxBytes := response.ConsensusParams.Block.MaxBytes - - return stateRoot, uint64(maxBytes), nil -} - -// initChain calls InitChainSync using consensus connection to app. -func (c *ABCIExecutionClient) initChain(genesis *cmtypes.GenesisDoc) (*abci.ResponseInitChain, error) { - params := genesis.ConsensusParams - - validators := make([]*cmtypes.Validator, len(genesis.Validators)) - for i, v := range genesis.Validators { - validators[i] = cmtypes.NewValidator(v.PubKey, v.Power) - } - - return c.proxyApp.InitChain(context.Background(), &abci.RequestInitChain{ - Time: genesis.GenesisTime, - ChainId: genesis.ChainID, - ConsensusParams: &cmproto.ConsensusParams{ - Block: &cmproto.BlockParams{ - MaxBytes: params.Block.MaxBytes, - MaxGas: params.Block.MaxGas, - }, - Evidence: &cmproto.EvidenceParams{ - MaxAgeNumBlocks: params.Evidence.MaxAgeNumBlocks, - MaxAgeDuration: params.Evidence.MaxAgeDuration, - MaxBytes: params.Evidence.MaxBytes, - }, - Validator: &cmproto.ValidatorParams{ - PubKeyTypes: params.Validator.PubKeyTypes, - }, - Version: &cmproto.VersionParams{ - App: params.Version.App, - }, - Abci: &cmproto.ABCIParams{ - VoteExtensionsEnableHeight: params.ABCI.VoteExtensionsEnableHeight, - }, - }, - Validators: cmtypes.TM2PB.ValidatorUpdates(cmtypes.NewValidatorSet(validators)), - AppStateBytes: genesis.AppState, - InitialHeight: genesis.InitialHeight, - }) +) (rollkitTypes.Hash, uint64, error) { + return c.execute.InitChain(genesisTime, initialHeight, chainID) } -// GetTxs retrieves all available transactions from the mempool. -func (c *ABCIExecutionClient) GetTxs() ([]types.Tx, error) { - state, err := c.store.GetState(context.Background()) - if err != nil { - return nil, fmt.Errorf("failed to get current state: %w", err) - } - - maxBytes := state.ConsensusParams.Block.MaxBytes - if maxBytes == -1 { - maxBytes = int64(cmtypes.MaxBlockSizeBytes) - } - if maxBytes > int64(c.maxBytes) { - c.logger.Debug("limiting maxBytes to", "maxBytes", c.maxBytes) - maxBytes = int64(c.maxBytes) - } - - cmTxs := c.mempool.ReapMaxTxs(int(maxBytes)) - - rollkitTxs := make([]types.Tx, len(cmTxs)) - for i, tx := range cmTxs { - rollkitTxs[i] = types.Tx(tx) - } - - return rollkitTxs, nil +// GetTxs retrieves transactions from the transaction pool. +func (c *ABCIExecutionClient) GetTxs() ([]rollkitTypes.Tx, error) { + return c.execute.GetTxs() } -// ExecuteTxs executes a set of transactions to produce a new block. +// ExecuteTxs executes the given transactions and returns the new state root and gas used. func (c *ABCIExecutionClient) ExecuteTxs( - txs []types.Tx, + txs []rollkitTypes.Tx, blockHeight uint64, timestamp time.Time, - prevStateRoot types.Hash, -) (types.Hash, uint64, error) { - ctx := context.Background() - - state, err := c.store.GetState(ctx) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to get current state: %w", err) - } - - cmTxs := fromRollkitTxs(txs) - - var lastSignature *types.Signature - var lastHeaderHash types.Hash - var lastExtendedCommit abci.ExtendedCommitInfo - - if blockHeight == uint64(c.genesis.InitialHeight) { - lastSignature = &types.Signature{} - lastHeaderHash = types.Hash{} - lastExtendedCommit = abci.ExtendedCommitInfo{} - } else { - lastSignature, err = c.store.GetSignature(ctx, blockHeight-1) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("error while loading last commit: %w", err) - } - - lastHeader, _, err := c.store.GetBlockData(ctx, blockHeight-1) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("error while loading last block: %w", err) - } - lastHeaderHash = lastHeader.Hash() - - extCommit, err := c.store.GetExtendedCommit(ctx, blockHeight-1) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to load extended commit for height %d: %w", blockHeight-1, err) - } - if extCommit != nil { - lastExtendedCommit = *extCommit - } - } - - header, data, err := c.CreateBlock( - blockHeight, - lastSignature, - lastExtendedCommit, - lastHeaderHash, - state, - cmTxs, - timestamp, - ) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to create block: %w", err) - } - - isValid, err := c.ProcessProposal(header, data, state) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to process proposal: %w", err) - } - if !isValid { - return types.Hash{}, 0, fmt.Errorf("proposal was not valid") - } - - newState, resp, err := c.ApplyBlock(ctx, state, header, data) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to apply block: %w", err) - } - - appHash, _, err := c.Commit(ctx, newState, header, data, resp) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to commit: %w", err) - } - - return types.Hash(appHash), uint64(newState.ConsensusParams.Block.MaxBytes), nil + prevStateRoot rollkitTypes.Hash, +) (rollkitTypes.Hash, uint64, error) { + return c.execute.ExecuteTxs(txs, blockHeight, timestamp, prevStateRoot) } // SetFinal marks a block at the given height as final. func (c *ABCIExecutionClient) SetFinal(blockHeight uint64) error { - ctx := context.Background() - - header, data, err := c.store.GetBlockData(ctx, blockHeight) - if err != nil { - return fmt.Errorf("failed to get block data for height %d: %w", blockHeight, err) - } - - state, err := c.store.GetState(ctx) - if err != nil { - return fmt.Errorf("failed to get current state: %w", err) - } - - resp, err := c.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ - Hash: header.Hash(), - Height: int64(blockHeight), - Time: header.Time(), - Txs: data.Txs.ToSliceOfBytes(), - ProposerAddress: header.ProposerAddress, - NextValidatorsHash: state.Validators.Hash(), - }) - if err != nil { - return fmt.Errorf("failed to finalize block at height %d: %w", blockHeight, err) - } - - state.AppHash = resp.AppHash - if err := c.store.UpdateState(ctx, state); err != nil { - return fmt.Errorf("failed to update state after finalizing block %d: %w", blockHeight, err) - } - - c.logger.Info("Block finalized", "height", blockHeight, "hash", header.Hash()) - - return nil -} - -// CreateBlock reaps transactions from mempool and builds a block. -func (c *ABCIExecutionClient) CreateBlock(height uint64, lastSignature *types.Signature, lastExtendedCommit abci.ExtendedCommitInfo, lastHeaderHash types.Hash, state types.State, txs cmtypes.Txs, timestamp time.Time) (*types.SignedHeader, *types.Data, error) { - maxBytes := state.ConsensusParams.Block.MaxBytes - emptyMaxBytes := maxBytes == -1 - if emptyMaxBytes { - maxBytes = int64(cmtypes.MaxBlockSizeBytes) - } - if maxBytes > int64(c.maxBytes) { //nolint:gosec - c.logger.Debug("limiting maxBytes to", "e.maxBytes=%d", c.maxBytes) - maxBytes = int64(c.maxBytes) //nolint:gosec - } - - header := &types.SignedHeader{ - Header: types.Header{ - Version: types.Version{ - Block: state.Version.Consensus.Block, - App: state.Version.Consensus.App, - }, - BaseHeader: types.BaseHeader{ - ChainID: c.chainID, - Height: height, - Time: uint64(timestamp.UnixNano()), //nolint:gosec - }, - DataHash: make(types.Hash, 32), - ConsensusHash: make(types.Hash, 32), - AppHash: state.AppHash, - LastResultsHash: state.LastResultsHash, - ProposerAddress: c.proposerAddress, - }, - Signature: *lastSignature, - } - data := &types.Data{ - Txs: toRollkitTxs(txs), - // IntermediateStateRoots: types.IntermediateStateRoots{RawRootsList: nil}, - // Note: Temporarily remove Evidence #896 - // Evidence: types.EvidenceData{Evidence: nil}, - } - - rpp, err := c.proxyApp.PrepareProposal( - context.TODO(), - &abci.RequestPrepareProposal{ - MaxTxBytes: maxBytes, - Txs: txs.ToSliceOfBytes(), - LocalLastCommit: lastExtendedCommit, - Misbehavior: []abci.Misbehavior{}, - Height: int64(header.Height()), //nolint:gosec - Time: header.Time(), //TODO: replace with sequencer timestamp - NextValidatorsHash: state.Validators.Hash(), - ProposerAddress: c.proposerAddress, - }, - ) - if err != nil { - // The App MUST ensure that only valid (and hence 'processable') transactions - // enter the mempool. Hence, at this point, we can't have any non-processable - // transaction causing an error. - // - // Also, the App can simply skip any transaction that could cause any kind of trouble. - // Either way, we cannot recover in a meaningful way, unless we skip proposing - // this block, repair what caused the error and try again. Hence, we return an - // error for now (the production code calling this function is expected to panic). - return nil, nil, err - } - - txl := cmtypes.ToTxs(rpp.Txs) - if err := txl.Validate(maxBytes); err != nil { - return nil, nil, err - } - - data.Txs = toRollkitTxs(txl) - // Note: This is hash of an ABCI type commit equivalent of the last signature in the signed header. - header.LastCommitHash = lastSignature.GetCommitHash(&header.Header, c.proposerAddress) - header.LastHeaderHash = lastHeaderHash - - return header, data, nil -} - -// ProcessProposal calls the corresponding ABCI method on the app. -func (c *ABCIExecutionClient) ProcessProposal( - header *types.SignedHeader, - data *types.Data, - state types.State, -) (bool, error) { - resp, err := c.proxyApp.ProcessProposal(context.TODO(), &abci.RequestProcessProposal{ - Hash: header.Hash(), - Height: int64(header.Height()), //nolint:gosec - Time: header.Time(), - Txs: data.Txs.ToSliceOfBytes(), - ProposedLastCommit: abci.CommitInfo{ - Round: 0, - Votes: []abci.VoteInfo{ - { - Validator: abci.Validator{ - Address: header.Validators.GetProposer().Address, - Power: header.Validators.GetProposer().VotingPower, - }, - BlockIdFlag: cmproto.BlockIDFlagCommit, - }, - }, - }, - Misbehavior: []abci.Misbehavior{}, - ProposerAddress: c.proposerAddress, - NextValidatorsHash: state.Validators.Hash(), - }) - if err != nil { - return false, err - } - if resp.IsStatusUnknown() { - panic(fmt.Sprintf("ProcessProposal responded with status %s", resp.Status.String())) - } - - return resp.IsAccepted(), nil -} - -// ApplyBlock validates and executes the block. -func (c *ABCIExecutionClient) ApplyBlock(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data) (types.State, *abci.ResponseFinalizeBlock, error) { - isAppValid, err := c.ProcessProposal(header, data, state) - if err != nil { - return types.State{}, nil, err - } - if !isAppValid { - return types.State{}, nil, fmt.Errorf("proposal processing resulted in an invalid application state") - } - - err = c.Validate(state, header, data) - if err != nil { - return types.State{}, nil, err - } - // This makes calls to the AppClient - resp, err := c.execute(ctx, state, header, data) - if err != nil { - return types.State{}, nil, err - } - abciValUpdates := resp.ValidatorUpdates - - validatorUpdates, err := cmtypes.PB2TM.ValidatorUpdates(abciValUpdates) - if err != nil { - return state, nil, err - } - - if resp.ConsensusParamUpdates != nil { - c.metrics.ConsensusParamUpdates.Add(1) - } - - state, err = c.updateState(state, header, data, resp, validatorUpdates) - if err != nil { - return types.State{}, nil, err - } - - if state.ConsensusParams.Block.MaxBytes <= 0 { - c.logger.Error("maxBytes<=0", "state.ConsensusParams.Block", state.ConsensusParams.Block, "header", header) - } - - return state, resp, nil -} - -// ExtendVote calls the ExtendVote ABCI method on the proxy app. -func (c *ABCIExecutionClient) ExtendVote(ctx context.Context, header *types.SignedHeader, data *types.Data) ([]byte, error) { - resp, err := c.proxyApp.ExtendVote(ctx, &abci.RequestExtendVote{ - Hash: header.Hash(), - Height: int64(header.Height()), //nolint:gosec - Time: header.Time(), - Txs: data.Txs.ToSliceOfBytes(), - ProposedLastCommit: abci.CommitInfo{ - Votes: []abci.VoteInfo{{ - Validator: abci.Validator{ - Address: header.Validators.GetProposer().Address, - Power: header.Validators.GetProposer().VotingPower, - }, - BlockIdFlag: cmproto.BlockIDFlagCommit, - }}, - }, - Misbehavior: nil, - NextValidatorsHash: header.ValidatorHash, - ProposerAddress: header.ProposerAddress, - }) - if err != nil { - return nil, err - } - return resp.VoteExtension, nil -} - -// Commit commits the block -func (c *ABCIExecutionClient) Commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { - appHash, retainHeight, err := c.commit(ctx, state, header, data, resp) - if err != nil { - return []byte{}, 0, err - } - - state.AppHash = appHash - - c.publishEvents(resp, header, data, state) - - return appHash, retainHeight, nil -} - -// updateConsensusParams updates the consensus parameters based on the provided updates. -func (c *ABCIExecutionClient) updateConsensusParams(height uint64, params cmtypes.ConsensusParams, consensusParamUpdates *cmproto.ConsensusParams) (cmproto.ConsensusParams, uint64, error) { - nextParams := params.Update(consensusParamUpdates) - if err := types.ConsensusParamsValidateBasic(nextParams); err != nil { - return cmproto.ConsensusParams{}, 0, fmt.Errorf("validating new consensus params: %w", err) - } - if err := nextParams.ValidateUpdate(consensusParamUpdates, int64(height)); err != nil { //nolint:gosec - return cmproto.ConsensusParams{}, 0, fmt.Errorf("updating consensus params: %w", err) - } - return nextParams.ToProto(), nextParams.Version.App, nil -} - -func (c *ABCIExecutionClient) updateState(state types.State, header *types.SignedHeader, data *types.Data, finalizeBlockResponse *abci.ResponseFinalizeBlock, validatorUpdates []*cmtypes.Validator) (types.State, error) { - height := header.Height() - if finalizeBlockResponse.ConsensusParamUpdates != nil { - nextParamsProto, appVersion, err := c.updateConsensusParams(height, types.ConsensusParamsFromProto(state.ConsensusParams), finalizeBlockResponse.ConsensusParamUpdates) - if err != nil { - return types.State{}, err - } - // Change results from this height but only applies to the next height. - state.LastHeightConsensusParamsChanged = height + 1 - state.Version.Consensus.App = appVersion - state.ConsensusParams = nextParamsProto - } - - nValSet := state.NextValidators.Copy() - lastHeightValSetChanged := state.LastHeightValidatorsChanged - - if len(nValSet.Validators) > 0 { - err := nValSet.UpdateWithChangeSet(validatorUpdates) - if err != nil { - if err.Error() != ErrEmptyValSetGenerated.Error() { - return state, err - } - nValSet = &cmtypes.ValidatorSet{ - Validators: make([]*cmtypes.Validator, 0), - Proposer: nil, - } - } - // Change results from this height but only applies to the next next height. - lastHeightValSetChanged = int64(header.Header.Height() + 1 + 1) //nolint:gosec - - if len(nValSet.Validators) > 0 { - nValSet.IncrementProposerPriority(1) - } - } - - s := types.State{ - Version: state.Version, - ChainID: state.ChainID, - InitialHeight: state.InitialHeight, - LastBlockHeight: height, - LastBlockTime: header.Time(), - LastBlockID: cmtypes.BlockID{ - Hash: cmbytes.HexBytes(header.Hash()), - // for now, we don't care about part set headers - }, - ConsensusParams: state.ConsensusParams, - LastHeightConsensusParamsChanged: state.LastHeightConsensusParamsChanged, - AppHash: finalizeBlockResponse.AppHash, - Validators: state.NextValidators.Copy(), - NextValidators: nValSet, - LastHeightValidatorsChanged: lastHeightValSetChanged, - LastValidators: state.Validators.Copy(), - } - copy(s.LastResultsHash[:], cmtypes.NewResults(finalizeBlockResponse.TxResults).Hash()) - - return s, nil -} - -func (e *ABCIExecutionClient) commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { - e.mempool.Lock() - defer e.mempool.Unlock() - - err := e.mempool.FlushAppConn() - if err != nil { - return nil, 0, err - } - - commitResp, err := e.proxyApp.Commit(ctx) - if err != nil { - return nil, 0, err - } - - maxBytes := state.ConsensusParams.Block.MaxBytes - maxGas := state.ConsensusParams.Block.MaxGas - cTxs := fromRollkitTxs(data.Txs) - e.mempoolReaper.UpdateCommitedTxs(cTxs) - err = e.mempool.Update(header.Height(), cTxs, resp.TxResults, mempool.PreCheckMaxBytes(maxBytes), mempool.PostCheckMaxGas(maxGas)) - if err != nil { - return nil, 0, err - } - - return resp.AppHash, uint64(commitResp.RetainHeight), err //nolint:gosec -} - -// Validate validates the state and the block for the executor -func (e *ABCIExecutionClient) Validate(state types.State, header *types.SignedHeader, data *types.Data) error { - if err := header.ValidateBasic(); err != nil { - return err - } - if err := data.ValidateBasic(); err != nil { - return err - } - if err := types.Validate(header, data); err != nil { - return err - } - if header.Version.App != state.Version.Consensus.App || - header.Version.Block != state.Version.Consensus.Block { - return errors.New("block version mismatch") - } - if state.LastBlockHeight <= 0 && header.Height() != state.InitialHeight { - return errors.New("initial block height mismatch") - } - if state.LastBlockHeight > 0 && header.Height() != state.LastBlockHeight+1 { - return errors.New("block height mismatch") - } - if !bytes.Equal(header.AppHash[:], state.AppHash[:]) { - return errors.New("AppHash mismatch") - } - - if !bytes.Equal(header.LastResultsHash[:], state.LastResultsHash[:]) { - return errors.New("LastResultsHash mismatch") - } - - return nil -} - -func (e *ABCIExecutionClient) execute(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data) (*abci.ResponseFinalizeBlock, error) { - // Only execute if the node hasn't already shut down - select { - case <-ctx.Done(): - return nil, ctx.Err() - default: - } - abciHeader, err := abciconv.ToABCIHeaderPB(&header.Header) - if err != nil { - return nil, err - } - abciHeader.ChainID = e.chainID - abciBlock, err := abciconv.ToABCIBlock(header, data) - if err != nil { - return nil, err - } - - startTime := time.Now().UnixNano() - finalizeBlockResponse, err := e.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ - Hash: header.Hash(), - NextValidatorsHash: state.Validators.Hash(), - ProposerAddress: abciHeader.ProposerAddress, - Height: abciHeader.Height, - Time: abciHeader.Time, - DecidedLastCommit: abci.CommitInfo{ - Round: 0, - Votes: nil, - }, - Misbehavior: abciBlock.Evidence.Evidence.ToABCI(), - Txs: abciBlock.Txs.ToSliceOfBytes(), - }) - endTime := time.Now().UnixNano() - e.metrics.BlockProcessingTime.Observe(float64(endTime-startTime) / 1000000) - if err != nil { - e.logger.Error("error in proxyAppConn.FinalizeBlock", "err", err) - return nil, err - } - - e.logger.Info( - "finalized block", - "height", abciBlock.Height, - "num_txs_res", len(finalizeBlockResponse.TxResults), - "num_val_updates", len(finalizeBlockResponse.ValidatorUpdates), - "block_app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash), - ) - - // Assert that the application correctly returned tx results for each of the transactions provided in the block - if len(abciBlock.Data.Txs) != len(finalizeBlockResponse.TxResults) { - return nil, fmt.Errorf("expected tx results length to match size of transactions in block. Expected %d, got %d", len(data.Txs), len(finalizeBlockResponse.TxResults)) - } - - e.logger.Info("executed block", "height", abciHeader.Height, "app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash)) - - return finalizeBlockResponse, nil -} - -func (e *ABCIExecutionClient) publishEvents(resp *abci.ResponseFinalizeBlock, header *types.SignedHeader, data *types.Data, state types.State) { - if e.eventBus == nil { - return - } - - abciBlock, err := abciconv.ToABCIBlock(header, data) - if err != nil { - return - } - - if err := e.eventBus.PublishEventNewBlock(cmtypes.EventDataNewBlock{ - Block: abciBlock, - BlockID: cmtypes.BlockID{ - Hash: cmbytes.HexBytes(header.Hash()), - // for now, we don't care about part set headers - }, - ResultFinalizeBlock: *resp, - }); err != nil { - e.logger.Error("failed publishing new block", "err", err) - } - - if err := e.eventBus.PublishEventNewBlockHeader(cmtypes.EventDataNewBlockHeader{ - Header: abciBlock.Header, - }); err != nil { - e.logger.Error("failed publishing new block header", "err", err) - } - - if err := e.eventBus.PublishEventNewBlockEvents(cmtypes.EventDataNewBlockEvents{ - Height: abciBlock.Height, - Events: resp.Events, - NumTxs: int64(len(abciBlock.Txs)), - }); err != nil { - e.logger.Error("failed publishing new block events", "err", err) - } - - if len(abciBlock.Evidence.Evidence) != 0 { - for _, ev := range abciBlock.Evidence.Evidence { - if err := e.eventBus.PublishEventNewEvidence(cmtypes.EventDataNewEvidence{ - Evidence: ev, - Height: int64(header.Header.Height()), //nolint:gosec - }); err != nil { - e.logger.Error("failed publishing new evidence", "err", err) - } - } - } - - for i, tx := range abciBlock.Data.Txs { - err := e.eventBus.PublishEventTx(cmtypes.EventDataTx{ - TxResult: abci.TxResult{ - Height: abciBlock.Height, - Index: uint32(i), //nolint:gosec - Tx: tx, - Result: *(resp.TxResults[i]), - }, - }) - if err != nil { - e.logger.Error("failed publishing event TX", "err", err) - } - } -} - -func toRollkitTxs(txs cmtypes.Txs) types.Txs { - rollkitTxs := make(types.Txs, len(txs)) - for i := range txs { - rollkitTxs[i] = []byte(txs[i]) - } - return rollkitTxs -} - -func fromRollkitTxs(rollkitTxs types.Txs) cmtypes.Txs { - txs := make(cmtypes.Txs, len(rollkitTxs)) - for i := range rollkitTxs { - txs[i] = []byte(rollkitTxs[i]) - } - return txs + return c.execute.SetFinal(blockHeight) } diff --git a/proxy/client.go b/proxy/client.go new file mode 100644 index 0000000..ef37ff8 --- /dev/null +++ b/proxy/client.go @@ -0,0 +1,731 @@ +package proxy + +import ( + "bytes" + "context" + "errors" + "fmt" + "time" + + abci "github.com/cometbft/cometbft/abci/types" + cmbytes "github.com/cometbft/cometbft/libs/bytes" + cmproto "github.com/cometbft/cometbft/proto/tendermint/types" + "github.com/cometbft/cometbft/proxy" + cmtypes "github.com/cometbft/cometbft/types" + execution "github.com/rollkit/go-execution" + proxyJsonrpc "github.com/rollkit/go-execution/proxy/jsonrpc" + "github.com/rollkit/rollkit/mempool" + "github.com/rollkit/rollkit/state" + "github.com/rollkit/rollkit/store" + "github.com/rollkit/rollkit/third_party/log" + "github.com/rollkit/rollkit/types" + abciconv "github.com/rollkit/rollkit/types/abci" +) + +// ErrEmptyValSetGenerated is returned when applying the validator changes would result in empty set. +var ErrEmptyValSetGenerated = errors.New("applying the validator changes would result in empty set") + +// ErrAddingValidatorToBased is returned when trying to add a validator to an empty validator set. +var ErrAddingValidatorToBased = errors.New("cannot add validators to empty validator set") + +// Ensure ProxyClient implements Execute interface +var _ execution.Execute = (*ProxyClient)(nil) + +// ProxyClient implements the Execute interface in go-execution +type ProxyClient struct { + client *proxyJsonrpc.Client + // abci specific + proxyApp proxy.AppConnConsensus + eventBus *cmtypes.EventBus + genesis *cmtypes.GenesisDoc + maxBytes uint64 + proposerAddress []byte + chainID string + + // rollkit specific + mempool mempool.Mempool + mempoolReaper *mempool.CListMempoolReaper + logger log.Logger + metrics *state.Metrics + state *types.State + store store.Store +} + +func NewClient(client *proxyJsonrpc.Client, proxyApp proxy.AppConnConsensus, eventBus *cmtypes.EventBus, genesis *cmtypes.GenesisDoc, maxBytes uint64, proposerAddress []byte, chainID string, mempool mempool.Mempool, mempoolReaper *mempool.CListMempoolReaper, logger log.Logger, metrics *state.Metrics, state *types.State, store store.Store) *ProxyClient { + return &ProxyClient{ + client: client, + proxyApp: proxyApp, + eventBus: eventBus, + genesis: genesis, + maxBytes: maxBytes, + proposerAddress: proposerAddress, + chainID: chainID, + mempool: mempool, + mempoolReaper: mempoolReaper, + logger: logger, + metrics: metrics, + state: state, + store: store, + } +} + +// InitChain initializes the blockchain with genesis information. +func (c *ProxyClient) InitChain( + genesisTime time.Time, + initialHeight uint64, + chainID string, +) (types.Hash, uint64, error) { + genesis := &cmtypes.GenesisDoc{ + GenesisTime: genesisTime, + ChainID: chainID, + ConsensusParams: c.genesis.ConsensusParams, + Validators: c.genesis.Validators, + AppState: c.genesis.AppState, + InitialHeight: int64(initialHeight), + } + + response, err := c.initChain(genesis) + if err != nil { + return types.Hash{}, 0, err + } + + stateRoot := types.Hash(response.AppHash) + maxBytes := response.ConsensusParams.Block.MaxBytes + + return stateRoot, uint64(maxBytes), nil +} + +// initChain calls InitChainSync using consensus connection to app. +func (c *ProxyClient) initChain(genesis *cmtypes.GenesisDoc) (*abci.ResponseInitChain, error) { + params := genesis.ConsensusParams + + validators := make([]*cmtypes.Validator, len(genesis.Validators)) + for i, v := range genesis.Validators { + validators[i] = cmtypes.NewValidator(v.PubKey, v.Power) + } + + return c.proxyApp.InitChain(context.Background(), &abci.RequestInitChain{ + Time: genesis.GenesisTime, + ChainId: genesis.ChainID, + ConsensusParams: &cmproto.ConsensusParams{ + Block: &cmproto.BlockParams{ + MaxBytes: params.Block.MaxBytes, + MaxGas: params.Block.MaxGas, + }, + Evidence: &cmproto.EvidenceParams{ + MaxAgeNumBlocks: params.Evidence.MaxAgeNumBlocks, + MaxAgeDuration: params.Evidence.MaxAgeDuration, + MaxBytes: params.Evidence.MaxBytes, + }, + Validator: &cmproto.ValidatorParams{ + PubKeyTypes: params.Validator.PubKeyTypes, + }, + Version: &cmproto.VersionParams{ + App: params.Version.App, + }, + Abci: &cmproto.ABCIParams{ + VoteExtensionsEnableHeight: params.ABCI.VoteExtensionsEnableHeight, + }, + }, + Validators: cmtypes.TM2PB.ValidatorUpdates(cmtypes.NewValidatorSet(validators)), + AppStateBytes: genesis.AppState, + InitialHeight: genesis.InitialHeight, + }) +} + +// GetTxs retrieves all available transactions from the mempool. +func (c *ProxyClient) GetTxs() ([]types.Tx, error) { + state, err := c.store.GetState(context.Background()) + if err != nil { + return nil, fmt.Errorf("failed to get current state: %w", err) + } + + maxBytes := state.ConsensusParams.Block.MaxBytes + if maxBytes == -1 { + maxBytes = int64(cmtypes.MaxBlockSizeBytes) + } + if maxBytes > int64(c.maxBytes) { + c.logger.Debug("limiting maxBytes to", "maxBytes", c.maxBytes) + maxBytes = int64(c.maxBytes) + } + + cmTxs := c.mempool.ReapMaxTxs(int(maxBytes)) + + rollkitTxs := make([]types.Tx, len(cmTxs)) + for i, tx := range cmTxs { + rollkitTxs[i] = types.Tx(tx) + } + + return rollkitTxs, nil +} + +// ExecuteTxs executes a set of transactions to produce a new block. +func (c *ProxyClient) ExecuteTxs( + txs []types.Tx, + blockHeight uint64, + timestamp time.Time, + prevStateRoot types.Hash, +) (types.Hash, uint64, error) { + ctx := context.Background() + + state, err := c.store.GetState(ctx) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to get current state: %w", err) + } + + cmTxs := fromRollkitTxs(txs) + + var lastSignature *types.Signature + var lastHeaderHash types.Hash + var lastExtendedCommit abci.ExtendedCommitInfo + + if blockHeight == uint64(c.genesis.InitialHeight) { + lastSignature = &types.Signature{} + lastHeaderHash = types.Hash{} + lastExtendedCommit = abci.ExtendedCommitInfo{} + } else { + lastSignature, err = c.store.GetSignature(ctx, blockHeight-1) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("error while loading last commit: %w", err) + } + + lastHeader, _, err := c.store.GetBlockData(ctx, blockHeight-1) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("error while loading last block: %w", err) + } + lastHeaderHash = lastHeader.Hash() + + extCommit, err := c.store.GetExtendedCommit(ctx, blockHeight-1) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to load extended commit for height %d: %w", blockHeight-1, err) + } + if extCommit != nil { + lastExtendedCommit = *extCommit + } + } + + header, data, err := c.CreateBlock( + blockHeight, + lastSignature, + lastExtendedCommit, + lastHeaderHash, + state, + cmTxs, + timestamp, + ) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to create block: %w", err) + } + + isValid, err := c.ProcessProposal(header, data, state) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to process proposal: %w", err) + } + if !isValid { + return types.Hash{}, 0, fmt.Errorf("proposal was not valid") + } + + newState, resp, err := c.ApplyBlock(ctx, state, header, data) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to apply block: %w", err) + } + + appHash, _, err := c.Commit(ctx, newState, header, data, resp) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to commit: %w", err) + } + + return types.Hash(appHash), uint64(newState.ConsensusParams.Block.MaxBytes), nil +} + +// SetFinal marks a block at the given height as final. +func (c *ProxyClient) SetFinal(blockHeight uint64) error { + ctx := context.Background() + + header, data, err := c.store.GetBlockData(ctx, blockHeight) + if err != nil { + return fmt.Errorf("failed to get block data for height %d: %w", blockHeight, err) + } + + state, err := c.store.GetState(ctx) + if err != nil { + return fmt.Errorf("failed to get current state: %w", err) + } + + resp, err := c.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ + Hash: header.Hash(), + Height: int64(blockHeight), + Time: header.Time(), + Txs: data.Txs.ToSliceOfBytes(), + ProposerAddress: header.ProposerAddress, + NextValidatorsHash: state.Validators.Hash(), + }) + if err != nil { + return fmt.Errorf("failed to finalize block at height %d: %w", blockHeight, err) + } + + state.AppHash = resp.AppHash + if err := c.store.UpdateState(ctx, state); err != nil { + return fmt.Errorf("failed to update state after finalizing block %d: %w", blockHeight, err) + } + + c.logger.Info("Block finalized", "height", blockHeight, "hash", header.Hash()) + + return nil +} + +// CreateBlock reaps transactions from mempool and builds a block. +func (c *ProxyClient) CreateBlock(height uint64, lastSignature *types.Signature, lastExtendedCommit abci.ExtendedCommitInfo, lastHeaderHash types.Hash, state types.State, txs cmtypes.Txs, timestamp time.Time) (*types.SignedHeader, *types.Data, error) { + maxBytes := state.ConsensusParams.Block.MaxBytes + emptyMaxBytes := maxBytes == -1 + if emptyMaxBytes { + maxBytes = int64(cmtypes.MaxBlockSizeBytes) + } + if maxBytes > int64(c.maxBytes) { //nolint:gosec + c.logger.Debug("limiting maxBytes to", "e.maxBytes=%d", c.maxBytes) + maxBytes = int64(c.maxBytes) //nolint:gosec + } + + header := &types.SignedHeader{ + Header: types.Header{ + Version: types.Version{ + Block: state.Version.Consensus.Block, + App: state.Version.Consensus.App, + }, + BaseHeader: types.BaseHeader{ + ChainID: c.chainID, + Height: height, + Time: uint64(timestamp.UnixNano()), //nolint:gosec + }, + DataHash: make(types.Hash, 32), + ConsensusHash: make(types.Hash, 32), + AppHash: state.AppHash, + LastResultsHash: state.LastResultsHash, + ProposerAddress: c.proposerAddress, + }, + Signature: *lastSignature, + } + data := &types.Data{ + Txs: toRollkitTxs(txs), + // IntermediateStateRoots: types.IntermediateStateRoots{RawRootsList: nil}, + // Note: Temporarily remove Evidence #896 + // Evidence: types.EvidenceData{Evidence: nil}, + } + + rpp, err := c.proxyApp.PrepareProposal( + context.TODO(), + &abci.RequestPrepareProposal{ + MaxTxBytes: maxBytes, + Txs: txs.ToSliceOfBytes(), + LocalLastCommit: lastExtendedCommit, + Misbehavior: []abci.Misbehavior{}, + Height: int64(header.Height()), //nolint:gosec + Time: header.Time(), //TODO: replace with sequencer timestamp + NextValidatorsHash: state.Validators.Hash(), + ProposerAddress: c.proposerAddress, + }, + ) + if err != nil { + // The App MUST ensure that only valid (and hence 'processable') transactions + // enter the mempool. Hence, at this point, we can't have any non-processable + // transaction causing an error. + // + // Also, the App can simply skip any transaction that could cause any kind of trouble. + // Either way, we cannot recover in a meaningful way, unless we skip proposing + // this block, repair what caused the error and try again. Hence, we return an + // error for now (the production code calling this function is expected to panic). + return nil, nil, err + } + + txl := cmtypes.ToTxs(rpp.Txs) + if err := txl.Validate(maxBytes); err != nil { + return nil, nil, err + } + + data.Txs = toRollkitTxs(txl) + // Note: This is hash of an ABCI type commit equivalent of the last signature in the signed header. + header.LastCommitHash = lastSignature.GetCommitHash(&header.Header, c.proposerAddress) + header.LastHeaderHash = lastHeaderHash + + return header, data, nil +} + +// ProcessProposal calls the corresponding ABCI method on the app. +func (c *ProxyClient) ProcessProposal( + header *types.SignedHeader, + data *types.Data, + state types.State, +) (bool, error) { + resp, err := c.proxyApp.ProcessProposal(context.TODO(), &abci.RequestProcessProposal{ + Hash: header.Hash(), + Height: int64(header.Height()), //nolint:gosec + Time: header.Time(), + Txs: data.Txs.ToSliceOfBytes(), + ProposedLastCommit: abci.CommitInfo{ + Round: 0, + Votes: []abci.VoteInfo{ + { + Validator: abci.Validator{ + Address: header.Validators.GetProposer().Address, + Power: header.Validators.GetProposer().VotingPower, + }, + BlockIdFlag: cmproto.BlockIDFlagCommit, + }, + }, + }, + Misbehavior: []abci.Misbehavior{}, + ProposerAddress: c.proposerAddress, + NextValidatorsHash: state.Validators.Hash(), + }) + if err != nil { + return false, err + } + if resp.IsStatusUnknown() { + panic(fmt.Sprintf("ProcessProposal responded with status %s", resp.Status.String())) + } + + return resp.IsAccepted(), nil +} + +// ApplyBlock validates and executes the block. +func (c *ProxyClient) ApplyBlock(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data) (types.State, *abci.ResponseFinalizeBlock, error) { + isAppValid, err := c.ProcessProposal(header, data, state) + if err != nil { + return types.State{}, nil, err + } + if !isAppValid { + return types.State{}, nil, fmt.Errorf("proposal processing resulted in an invalid application state") + } + + err = c.Validate(state, header, data) + if err != nil { + return types.State{}, nil, err + } + // This makes calls to the AppClient + resp, err := c.execute(ctx, state, header, data) + if err != nil { + return types.State{}, nil, err + } + abciValUpdates := resp.ValidatorUpdates + + validatorUpdates, err := cmtypes.PB2TM.ValidatorUpdates(abciValUpdates) + if err != nil { + return state, nil, err + } + + if resp.ConsensusParamUpdates != nil { + c.metrics.ConsensusParamUpdates.Add(1) + } + + state, err = c.updateState(state, header, data, resp, validatorUpdates) + if err != nil { + return types.State{}, nil, err + } + + if state.ConsensusParams.Block.MaxBytes <= 0 { + c.logger.Error("maxBytes<=0", "state.ConsensusParams.Block", state.ConsensusParams.Block, "header", header) + } + + return state, resp, nil +} + +// ExtendVote calls the ExtendVote ABCI method on the proxy app. +func (c *ProxyClient) ExtendVote(ctx context.Context, header *types.SignedHeader, data *types.Data) ([]byte, error) { + resp, err := c.proxyApp.ExtendVote(ctx, &abci.RequestExtendVote{ + Hash: header.Hash(), + Height: int64(header.Height()), //nolint:gosec + Time: header.Time(), + Txs: data.Txs.ToSliceOfBytes(), + ProposedLastCommit: abci.CommitInfo{ + Votes: []abci.VoteInfo{{ + Validator: abci.Validator{ + Address: header.Validators.GetProposer().Address, + Power: header.Validators.GetProposer().VotingPower, + }, + BlockIdFlag: cmproto.BlockIDFlagCommit, + }}, + }, + Misbehavior: nil, + NextValidatorsHash: header.ValidatorHash, + ProposerAddress: header.ProposerAddress, + }) + if err != nil { + return nil, err + } + return resp.VoteExtension, nil +} + +// Commit commits the block +func (c *ProxyClient) Commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { + appHash, retainHeight, err := c.commit(ctx, state, header, data, resp) + if err != nil { + return []byte{}, 0, err + } + + state.AppHash = appHash + + c.publishEvents(resp, header, data, state) + + return appHash, retainHeight, nil +} + +// updateConsensusParams updates the consensus parameters based on the provided updates. +func (c *ProxyClient) updateConsensusParams(height uint64, params cmtypes.ConsensusParams, consensusParamUpdates *cmproto.ConsensusParams) (cmproto.ConsensusParams, uint64, error) { + nextParams := params.Update(consensusParamUpdates) + if err := types.ConsensusParamsValidateBasic(nextParams); err != nil { + return cmproto.ConsensusParams{}, 0, fmt.Errorf("validating new consensus params: %w", err) + } + if err := nextParams.ValidateUpdate(consensusParamUpdates, int64(height)); err != nil { //nolint:gosec + return cmproto.ConsensusParams{}, 0, fmt.Errorf("updating consensus params: %w", err) + } + return nextParams.ToProto(), nextParams.Version.App, nil +} + +func (c *ProxyClient) updateState(state types.State, header *types.SignedHeader, data *types.Data, finalizeBlockResponse *abci.ResponseFinalizeBlock, validatorUpdates []*cmtypes.Validator) (types.State, error) { + height := header.Height() + if finalizeBlockResponse.ConsensusParamUpdates != nil { + nextParamsProto, appVersion, err := c.updateConsensusParams(height, types.ConsensusParamsFromProto(state.ConsensusParams), finalizeBlockResponse.ConsensusParamUpdates) + if err != nil { + return types.State{}, err + } + // Change results from this height but only applies to the next height. + state.LastHeightConsensusParamsChanged = height + 1 + state.Version.Consensus.App = appVersion + state.ConsensusParams = nextParamsProto + } + + nValSet := state.NextValidators.Copy() + lastHeightValSetChanged := state.LastHeightValidatorsChanged + + if len(nValSet.Validators) > 0 { + err := nValSet.UpdateWithChangeSet(validatorUpdates) + if err != nil { + if err.Error() != ErrEmptyValSetGenerated.Error() { + return state, err + } + nValSet = &cmtypes.ValidatorSet{ + Validators: make([]*cmtypes.Validator, 0), + Proposer: nil, + } + } + // Change results from this height but only applies to the next next height. + lastHeightValSetChanged = int64(header.Header.Height() + 1 + 1) //nolint:gosec + + if len(nValSet.Validators) > 0 { + nValSet.IncrementProposerPriority(1) + } + } + + s := types.State{ + Version: state.Version, + ChainID: state.ChainID, + InitialHeight: state.InitialHeight, + LastBlockHeight: height, + LastBlockTime: header.Time(), + LastBlockID: cmtypes.BlockID{ + Hash: cmbytes.HexBytes(header.Hash()), + // for now, we don't care about part set headers + }, + ConsensusParams: state.ConsensusParams, + LastHeightConsensusParamsChanged: state.LastHeightConsensusParamsChanged, + AppHash: finalizeBlockResponse.AppHash, + Validators: state.NextValidators.Copy(), + NextValidators: nValSet, + LastHeightValidatorsChanged: lastHeightValSetChanged, + LastValidators: state.Validators.Copy(), + } + copy(s.LastResultsHash[:], cmtypes.NewResults(finalizeBlockResponse.TxResults).Hash()) + + return s, nil +} + +func (c *ProxyClient) commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { + c.mempool.Lock() + defer c.mempool.Unlock() + + err := c.mempool.FlushAppConn() + if err != nil { + return nil, 0, err + } + + commitResp, err := c.proxyApp.Commit(ctx) + if err != nil { + return nil, 0, err + } + + maxBytes := state.ConsensusParams.Block.MaxBytes + maxGas := state.ConsensusParams.Block.MaxGas + cTxs := fromRollkitTxs(data.Txs) + c.mempoolReaper.UpdateCommitedTxs(cTxs) + err = c.mempool.Update(header.Height(), cTxs, resp.TxResults, mempool.PreCheckMaxBytes(maxBytes), mempool.PostCheckMaxGas(maxGas)) + if err != nil { + return nil, 0, err + } + + return resp.AppHash, uint64(commitResp.RetainHeight), err //nolint:gosec +} + +// Validate validates the state and the block for the executor +func (c *ProxyClient) Validate(state types.State, header *types.SignedHeader, data *types.Data) error { + if err := header.ValidateBasic(); err != nil { + return err + } + if err := data.ValidateBasic(); err != nil { + return err + } + if err := types.Validate(header, data); err != nil { + return err + } + if header.Version.App != state.Version.Consensus.App || + header.Version.Block != state.Version.Consensus.Block { + return errors.New("block version mismatch") + } + if state.LastBlockHeight <= 0 && header.Height() != state.InitialHeight { + return errors.New("initial block height mismatch") + } + if state.LastBlockHeight > 0 && header.Height() != state.LastBlockHeight+1 { + return errors.New("block height mismatch") + } + if !bytes.Equal(header.AppHash[:], state.AppHash[:]) { + return errors.New("AppHash mismatch") + } + + if !bytes.Equal(header.LastResultsHash[:], state.LastResultsHash[:]) { + return errors.New("LastResultsHash mismatch") + } + + return nil +} + +func (c *ProxyClient) execute(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data) (*abci.ResponseFinalizeBlock, error) { + // Only execute if the node hasn't already shut down + select { + case <-ctx.Done(): + return nil, ctx.Err() + default: + } + abciHeader, err := abciconv.ToABCIHeaderPB(&header.Header) + if err != nil { + return nil, err + } + abciHeader.ChainID = c.chainID + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return nil, err + } + + startTime := time.Now().UnixNano() + finalizeBlockResponse, err := c.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ + Hash: header.Hash(), + NextValidatorsHash: state.Validators.Hash(), + ProposerAddress: abciHeader.ProposerAddress, + Height: abciHeader.Height, + Time: abciHeader.Time, + DecidedLastCommit: abci.CommitInfo{ + Round: 0, + Votes: nil, + }, + Misbehavior: abciBlock.Evidence.Evidence.ToABCI(), + Txs: abciBlock.Txs.ToSliceOfBytes(), + }) + endTime := time.Now().UnixNano() + c.metrics.BlockProcessingTime.Observe(float64(endTime-startTime) / 1000000) + if err != nil { + c.logger.Error("error in proxyAppConn.FinalizeBlock", "err", err) + return nil, err + } + + c.logger.Info( + "finalized block", + "height", abciBlock.Height, + "num_txs_res", len(finalizeBlockResponse.TxResults), + "num_val_updates", len(finalizeBlockResponse.ValidatorUpdates), + "block_app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash), + ) + + // Assert that the application correctly returned tx results for each of the transactions provided in the block + if len(abciBlock.Data.Txs) != len(finalizeBlockResponse.TxResults) { + return nil, fmt.Errorf("expected tx results length to match size of transactions in block. Expected %d, got %d", len(data.Txs), len(finalizeBlockResponse.TxResults)) + } + + c.logger.Info("executed block", "height", abciHeader.Height, "app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash)) + + return finalizeBlockResponse, nil +} + +func (e *ProxyClient) publishEvents(resp *abci.ResponseFinalizeBlock, header *types.SignedHeader, data *types.Data, state types.State) { + if e.eventBus == nil { + return + } + + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return + } + + if err := e.eventBus.PublishEventNewBlock(cmtypes.EventDataNewBlock{ + Block: abciBlock, + BlockID: cmtypes.BlockID{ + Hash: cmbytes.HexBytes(header.Hash()), + // for now, we don't care about part set headers + }, + ResultFinalizeBlock: *resp, + }); err != nil { + e.logger.Error("failed publishing new block", "err", err) + } + + if err := e.eventBus.PublishEventNewBlockHeader(cmtypes.EventDataNewBlockHeader{ + Header: abciBlock.Header, + }); err != nil { + e.logger.Error("failed publishing new block header", "err", err) + } + + if err := e.eventBus.PublishEventNewBlockEvents(cmtypes.EventDataNewBlockEvents{ + Height: abciBlock.Height, + Events: resp.Events, + NumTxs: int64(len(abciBlock.Txs)), + }); err != nil { + e.logger.Error("failed publishing new block events", "err", err) + } + + if len(abciBlock.Evidence.Evidence) != 0 { + for _, ev := range abciBlock.Evidence.Evidence { + if err := e.eventBus.PublishEventNewEvidence(cmtypes.EventDataNewEvidence{ + Evidence: ev, + Height: int64(header.Header.Height()), //nolint:gosec + }); err != nil { + e.logger.Error("failed publishing new evidence", "err", err) + } + } + } + + for i, tx := range abciBlock.Data.Txs { + err := e.eventBus.PublishEventTx(cmtypes.EventDataTx{ + TxResult: abci.TxResult{ + Height: abciBlock.Height, + Index: uint32(i), //nolint:gosec + Tx: tx, + Result: *(resp.TxResults[i]), + }, + }) + if err != nil { + e.logger.Error("failed publishing event TX", "err", err) + } + } +} + +func toRollkitTxs(txs cmtypes.Txs) types.Txs { + rollkitTxs := make(types.Txs, len(txs)) + for i := range txs { + rollkitTxs[i] = []byte(txs[i]) + } + return rollkitTxs +} + +func fromRollkitTxs(rollkitTxs types.Txs) cmtypes.Txs { + txs := make(cmtypes.Txs, len(rollkitTxs)) + for i := range rollkitTxs { + txs[i] = []byte(rollkitTxs[i]) + } + return txs +} From 8a61a892003206e57aff366de2b13d32fc2cbab4 Mon Sep 17 00:00:00 2001 From: Jay Jie Date: Tue, 29 Oct 2024 21:50:41 -0700 Subject: [PATCH 05/10] chore: switch to grpc proxy --- proxy/client.go | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/proxy/client.go b/proxy/client.go index ef37ff8..e018b60 100644 --- a/proxy/client.go +++ b/proxy/client.go @@ -13,7 +13,7 @@ import ( "github.com/cometbft/cometbft/proxy" cmtypes "github.com/cometbft/cometbft/types" execution "github.com/rollkit/go-execution" - proxyJsonrpc "github.com/rollkit/go-execution/proxy/jsonrpc" + proxy_grpc "github.com/rollkit/go-execution/proxy/grpc" "github.com/rollkit/rollkit/mempool" "github.com/rollkit/rollkit/state" "github.com/rollkit/rollkit/store" @@ -33,7 +33,7 @@ var _ execution.Execute = (*ProxyClient)(nil) // ProxyClient implements the Execute interface in go-execution type ProxyClient struct { - client *proxyJsonrpc.Client + client *proxy_grpc.Client // abci specific proxyApp proxy.AppConnConsensus eventBus *cmtypes.EventBus @@ -51,7 +51,7 @@ type ProxyClient struct { store store.Store } -func NewClient(client *proxyJsonrpc.Client, proxyApp proxy.AppConnConsensus, eventBus *cmtypes.EventBus, genesis *cmtypes.GenesisDoc, maxBytes uint64, proposerAddress []byte, chainID string, mempool mempool.Mempool, mempoolReaper *mempool.CListMempoolReaper, logger log.Logger, metrics *state.Metrics, state *types.State, store store.Store) *ProxyClient { +func NewClient(client *proxy_grpc.Client, proxyApp proxy.AppConnConsensus, eventBus *cmtypes.EventBus, genesis *cmtypes.GenesisDoc, maxBytes uint64, proposerAddress []byte, chainID string, mempool mempool.Mempool, mempoolReaper *mempool.CListMempoolReaper, logger log.Logger, metrics *state.Metrics, state *types.State, store store.Store) *ProxyClient { return &ProxyClient{ client: client, proxyApp: proxyApp, From c9120e40525c1249f3c040fa5fcde343f26823d0 Mon Sep 17 00:00:00 2001 From: Jay Jie Date: Wed, 6 Nov 2024 12:41:33 -0800 Subject: [PATCH 06/10] refactor: remove proxy layer --- execution.go | 717 +++++++++++++++++++++++++++++++++++++++++++++-- proxy/client.go | 731 ------------------------------------------------ 2 files changed, 700 insertions(+), 748 deletions(-) delete mode 100644 proxy/client.go diff --git a/execution.go b/execution.go index 8a36079..80666ce 100644 --- a/execution.go +++ b/execution.go @@ -1,49 +1,732 @@ package execution import ( + "bytes" + "context" + "errors" + "fmt" "time" - execution "github.com/rollkit/go-execution" - rollkitTypes "github.com/rollkit/rollkit/types" + abci "github.com/cometbft/cometbft/abci/types" + cmbytes "github.com/cometbft/cometbft/libs/bytes" + cmproto "github.com/cometbft/cometbft/proto/tendermint/types" + "github.com/cometbft/cometbft/proxy" + cmtypes "github.com/cometbft/cometbft/types" + proxy_grpc "github.com/rollkit/go-execution/proxy/grpc" + "github.com/rollkit/rollkit/mempool" + "github.com/rollkit/rollkit/state" + "github.com/rollkit/rollkit/store" + "github.com/rollkit/rollkit/third_party/log" + "github.com/rollkit/rollkit/types" + abciconv "github.com/rollkit/rollkit/types/abci" ) +// ErrEmptyValSetGenerated is returned when applying the validator changes would result in empty set. +var ErrEmptyValSetGenerated = errors.New("applying the validator changes would result in empty set") + +// ErrAddingValidatorToBased is returned when trying to add a validator to an empty validator set. +var ErrAddingValidatorToBased = errors.New("cannot add validators to empty validator set") + +// ABCIExecutionClient implements the Execute interface for ABCI applications type ABCIExecutionClient struct { - execute execution.Execute + client *proxy_grpc.Client + // abci specific + proxyApp proxy.AppConnConsensus + eventBus *cmtypes.EventBus + genesis *cmtypes.GenesisDoc + maxBytes uint64 + proposerAddress []byte + chainID string + + // rollkit specific + mempool mempool.Mempool + mempoolReaper *mempool.CListMempoolReaper + logger log.Logger + metrics *state.Metrics + state *types.State + store store.Store } -func NewABCIExecutionClient(execute execution.Execute) *ABCIExecutionClient { +// NewABCIExecutionClient creates a new ABCIExecutionClient +func NewABCIExecutionClient( + client *proxy_grpc.Client, + proxyApp proxy.AppConnConsensus, + eventBus *cmtypes.EventBus, + genesis *cmtypes.GenesisDoc, + maxBytes uint64, + proposerAddress []byte, + chainID string, + mempool mempool.Mempool, + mempoolReaper *mempool.CListMempoolReaper, + logger log.Logger, + metrics *state.Metrics, + state *types.State, + store store.Store, +) *ABCIExecutionClient { return &ABCIExecutionClient{ - execute: execute, + client: client, + proxyApp: proxyApp, + eventBus: eventBus, + genesis: genesis, + maxBytes: maxBytes, + proposerAddress: proposerAddress, + chainID: chainID, + mempool: mempool, + mempoolReaper: mempoolReaper, + logger: logger, + metrics: metrics, + state: state, + store: store, } } -var _ execution.Execute = (*ABCIExecutionClient)(nil) - // InitChain initializes the blockchain with genesis information. func (c *ABCIExecutionClient) InitChain( genesisTime time.Time, initialHeight uint64, chainID string, -) (rollkitTypes.Hash, uint64, error) { - return c.execute.InitChain(genesisTime, initialHeight, chainID) +) (types.Hash, uint64, error) { + genesis := &cmtypes.GenesisDoc{ + GenesisTime: genesisTime, + ChainID: chainID, + ConsensusParams: c.genesis.ConsensusParams, + Validators: c.genesis.Validators, + AppState: c.genesis.AppState, + InitialHeight: int64(initialHeight), + } + + response, err := c.initChain(genesis) + if err != nil { + return types.Hash{}, 0, err + } + + stateRoot := types.Hash(response.AppHash) + maxBytes := response.ConsensusParams.Block.MaxBytes + + return stateRoot, uint64(maxBytes), nil } -// GetTxs retrieves transactions from the transaction pool. -func (c *ABCIExecutionClient) GetTxs() ([]rollkitTypes.Tx, error) { - return c.execute.GetTxs() +// initChain calls InitChainSync using consensus connection to app. +func (c *ABCIExecutionClient) initChain(genesis *cmtypes.GenesisDoc) (*abci.ResponseInitChain, error) { + params := genesis.ConsensusParams + + validators := make([]*cmtypes.Validator, len(genesis.Validators)) + for i, v := range genesis.Validators { + validators[i] = cmtypes.NewValidator(v.PubKey, v.Power) + } + + return c.proxyApp.InitChain(context.Background(), &abci.RequestInitChain{ + Time: genesis.GenesisTime, + ChainId: genesis.ChainID, + ConsensusParams: &cmproto.ConsensusParams{ + Block: &cmproto.BlockParams{ + MaxBytes: params.Block.MaxBytes, + MaxGas: params.Block.MaxGas, + }, + Evidence: &cmproto.EvidenceParams{ + MaxAgeNumBlocks: params.Evidence.MaxAgeNumBlocks, + MaxAgeDuration: params.Evidence.MaxAgeDuration, + MaxBytes: params.Evidence.MaxBytes, + }, + Validator: &cmproto.ValidatorParams{ + PubKeyTypes: params.Validator.PubKeyTypes, + }, + Version: &cmproto.VersionParams{ + App: params.Version.App, + }, + Abci: &cmproto.ABCIParams{ + VoteExtensionsEnableHeight: params.ABCI.VoteExtensionsEnableHeight, + }, + }, + Validators: cmtypes.TM2PB.ValidatorUpdates(cmtypes.NewValidatorSet(validators)), + AppStateBytes: genesis.AppState, + InitialHeight: genesis.InitialHeight, + }) +} + +// GetTxs retrieves all available transactions from the mempool. +func (c *ABCIExecutionClient) GetTxs() ([]types.Tx, error) { + state, err := c.store.GetState(context.Background()) + if err != nil { + return nil, fmt.Errorf("failed to get current state: %w", err) + } + + maxBytes := state.ConsensusParams.Block.MaxBytes + if maxBytes == -1 { + maxBytes = int64(cmtypes.MaxBlockSizeBytes) + } + if maxBytes > int64(c.maxBytes) { + c.logger.Debug("limiting maxBytes to", "maxBytes", c.maxBytes) + maxBytes = int64(c.maxBytes) + } + + cmTxs := c.mempool.ReapMaxTxs(int(maxBytes)) + + rollkitTxs := make([]types.Tx, len(cmTxs)) + for i, tx := range cmTxs { + rollkitTxs[i] = types.Tx(tx) + } + + return rollkitTxs, nil } // ExecuteTxs executes the given transactions and returns the new state root and gas used. func (c *ABCIExecutionClient) ExecuteTxs( - txs []rollkitTypes.Tx, + txs []types.Tx, blockHeight uint64, timestamp time.Time, - prevStateRoot rollkitTypes.Hash, -) (rollkitTypes.Hash, uint64, error) { - return c.execute.ExecuteTxs(txs, blockHeight, timestamp, prevStateRoot) + prevStateRoot types.Hash, +) (types.Hash, uint64, error) { + ctx := context.Background() + + state, err := c.store.GetState(ctx) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to get current state: %w", err) + } + + cmTxs := fromRollkitTxs(txs) + + var lastSignature *types.Signature + var lastHeaderHash types.Hash + var lastExtendedCommit abci.ExtendedCommitInfo + + if blockHeight == uint64(c.genesis.InitialHeight) { + lastSignature = &types.Signature{} + lastHeaderHash = types.Hash{} + lastExtendedCommit = abci.ExtendedCommitInfo{} + } else { + lastSignature, err = c.store.GetSignature(ctx, blockHeight-1) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("error while loading last commit: %w", err) + } + + lastHeader, _, err := c.store.GetBlockData(ctx, blockHeight-1) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("error while loading last block: %w", err) + } + lastHeaderHash = lastHeader.Hash() + + extCommit, err := c.store.GetExtendedCommit(ctx, blockHeight-1) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to load extended commit for height %d: %w", blockHeight-1, err) + } + if extCommit != nil { + lastExtendedCommit = *extCommit + } + } + + header, data, err := c.CreateBlock( + blockHeight, + lastSignature, + lastExtendedCommit, + lastHeaderHash, + state, + cmTxs, + timestamp, + ) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to create block: %w", err) + } + + isValid, err := c.ProcessProposal(header, data, state) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to process proposal: %w", err) + } + if !isValid { + return types.Hash{}, 0, fmt.Errorf("proposal was not valid") + } + + newState, resp, err := c.ApplyBlock(ctx, state, header, data) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to apply block: %w", err) + } + + appHash, _, err := c.Commit(ctx, newState, header, data, resp) + if err != nil { + return types.Hash{}, 0, fmt.Errorf("failed to commit: %w", err) + } + + return types.Hash(appHash), uint64(newState.ConsensusParams.Block.MaxBytes), nil } // SetFinal marks a block at the given height as final. func (c *ABCIExecutionClient) SetFinal(blockHeight uint64) error { - return c.execute.SetFinal(blockHeight) + ctx := context.Background() + + header, data, err := c.store.GetBlockData(ctx, blockHeight) + if err != nil { + return fmt.Errorf("failed to get block data for height %d: %w", blockHeight, err) + } + + state, err := c.store.GetState(ctx) + if err != nil { + return fmt.Errorf("failed to get current state: %w", err) + } + + resp, err := c.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ + Hash: header.Hash(), + Height: int64(blockHeight), + Time: header.Time(), + Txs: data.Txs.ToSliceOfBytes(), + ProposerAddress: header.ProposerAddress, + NextValidatorsHash: state.Validators.Hash(), + }) + if err != nil { + return fmt.Errorf("failed to finalize block at height %d: %w", blockHeight, err) + } + + state.AppHash = resp.AppHash + if err := c.store.UpdateState(ctx, state); err != nil { + return fmt.Errorf("failed to update state after finalizing block %d: %w", blockHeight, err) + } + + c.logger.Info("Block finalized", "height", blockHeight, "hash", header.Hash()) + + return nil +} + +// CreateBlock reaps transactions from mempool and builds a block. +func (c *ABCIExecutionClient) CreateBlock(height uint64, lastSignature *types.Signature, lastExtendedCommit abci.ExtendedCommitInfo, lastHeaderHash types.Hash, state types.State, txs cmtypes.Txs, timestamp time.Time) (*types.SignedHeader, *types.Data, error) { + maxBytes := state.ConsensusParams.Block.MaxBytes + emptyMaxBytes := maxBytes == -1 + if emptyMaxBytes { + maxBytes = int64(cmtypes.MaxBlockSizeBytes) + } + if maxBytes > int64(c.maxBytes) { + c.logger.Debug("limiting maxBytes to", "e.maxBytes=%d", c.maxBytes) + maxBytes = int64(c.maxBytes) + } + + header := &types.SignedHeader{ + Header: types.Header{ + Version: types.Version{ + Block: state.Version.Consensus.Block, + App: state.Version.Consensus.App, + }, + BaseHeader: types.BaseHeader{ + ChainID: c.chainID, + Height: height, + Time: uint64(timestamp.UnixNano()), + }, + DataHash: make(types.Hash, 32), + ConsensusHash: make(types.Hash, 32), + AppHash: state.AppHash, + LastResultsHash: state.LastResultsHash, + ProposerAddress: c.proposerAddress, + }, + Signature: *lastSignature, + } + data := &types.Data{ + Txs: toRollkitTxs(txs), + } + + rpp, err := c.proxyApp.PrepareProposal( + context.TODO(), + &abci.RequestPrepareProposal{ + MaxTxBytes: maxBytes, + Txs: txs.ToSliceOfBytes(), + LocalLastCommit: lastExtendedCommit, + Misbehavior: []abci.Misbehavior{}, + Height: int64(header.Height()), + Time: header.Time(), + NextValidatorsHash: state.Validators.Hash(), + ProposerAddress: c.proposerAddress, + }, + ) + if err != nil { + return nil, nil, err + } + + txl := cmtypes.ToTxs(rpp.Txs) + if err := txl.Validate(maxBytes); err != nil { + return nil, nil, err + } + + data.Txs = toRollkitTxs(txl) + header.LastCommitHash = lastSignature.GetCommitHash(&header.Header, c.proposerAddress) + header.LastHeaderHash = lastHeaderHash + + return header, data, nil +} + +// ProcessProposal calls the corresponding ABCI method on the app. +func (c *ABCIExecutionClient) ProcessProposal( + header *types.SignedHeader, + data *types.Data, + state types.State, +) (bool, error) { + resp, err := c.proxyApp.ProcessProposal(context.TODO(), &abci.RequestProcessProposal{ + Hash: header.Hash(), + Height: int64(header.Height()), + Time: header.Time(), + Txs: data.Txs.ToSliceOfBytes(), + ProposedLastCommit: abci.CommitInfo{ + Round: 0, + Votes: []abci.VoteInfo{ + { + Validator: abci.Validator{ + Address: header.Validators.GetProposer().Address, + Power: header.Validators.GetProposer().VotingPower, + }, + BlockIdFlag: cmproto.BlockIDFlagCommit, + }, + }, + }, + Misbehavior: []abci.Misbehavior{}, + ProposerAddress: c.proposerAddress, + NextValidatorsHash: state.Validators.Hash(), + }) + if err != nil { + return false, err + } + if resp.IsStatusUnknown() { + panic(fmt.Sprintf("ProcessProposal responded with status %s", resp.Status.String())) + } + + return resp.IsAccepted(), nil +} + +// ApplyBlock validates and executes the block. +func (c *ABCIExecutionClient) ApplyBlock(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data) (types.State, *abci.ResponseFinalizeBlock, error) { + isAppValid, err := c.ProcessProposal(header, data, state) + if err != nil { + return types.State{}, nil, err + } + if !isAppValid { + return types.State{}, nil, fmt.Errorf("proposal processing resulted in an invalid application state") + } + + err = c.Validate(state, header, data) + if err != nil { + return types.State{}, nil, err + } + + resp, err := c.execute(ctx, state, header, data) + if err != nil { + return types.State{}, nil, err + } + abciValUpdates := resp.ValidatorUpdates + + validatorUpdates, err := cmtypes.PB2TM.ValidatorUpdates(abciValUpdates) + if err != nil { + return state, nil, err + } + + if resp.ConsensusParamUpdates != nil { + c.metrics.ConsensusParamUpdates.Add(1) + } + + state, err = c.updateState(state, header, data, resp, validatorUpdates) + if err != nil { + return types.State{}, nil, err + } + + if state.ConsensusParams.Block.MaxBytes <= 0 { + c.logger.Error("maxBytes<=0", "state.ConsensusParams.Block", state.ConsensusParams.Block, "header", header) + } + + return state, resp, nil +} + +// ExtendVote calls the ExtendVote ABCI method on the proxy app. +func (c *ABCIExecutionClient) ExtendVote(ctx context.Context, header *types.SignedHeader, data *types.Data) ([]byte, error) { + resp, err := c.proxyApp.ExtendVote(ctx, &abci.RequestExtendVote{ + Hash: header.Hash(), + Height: int64(header.Height()), + Time: header.Time(), + Txs: data.Txs.ToSliceOfBytes(), + ProposedLastCommit: abci.CommitInfo{ + Votes: []abci.VoteInfo{{ + Validator: abci.Validator{ + Address: header.Validators.GetProposer().Address, + Power: header.Validators.GetProposer().VotingPower, + }, + BlockIdFlag: cmproto.BlockIDFlagCommit, + }}, + }, + Misbehavior: nil, + NextValidatorsHash: header.ValidatorHash, + ProposerAddress: header.ProposerAddress, + }) + if err != nil { + return nil, err + } + return resp.VoteExtension, nil +} + +// Commit commits the block +func (c *ABCIExecutionClient) Commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { + appHash, retainHeight, err := c.commit(ctx, state, header, data, resp) + if err != nil { + return []byte{}, 0, err + } + + state.AppHash = appHash + + c.publishEvents(resp, header, data, state) + + return appHash, retainHeight, nil +} + +// updateConsensusParams updates the consensus parameters based on the provided updates. +func (c *ABCIExecutionClient) updateConsensusParams(height uint64, params cmtypes.ConsensusParams, consensusParamUpdates *cmproto.ConsensusParams) (cmproto.ConsensusParams, uint64, error) { + nextParams := params.Update(consensusParamUpdates) + if err := types.ConsensusParamsValidateBasic(nextParams); err != nil { + return cmproto.ConsensusParams{}, 0, fmt.Errorf("validating new consensus params: %w", err) + } + if err := nextParams.ValidateUpdate(consensusParamUpdates, int64(height)); err != nil { + return cmproto.ConsensusParams{}, 0, fmt.Errorf("updating consensus params: %w", err) + } + return nextParams.ToProto(), nextParams.Version.App, nil +} + +// I'll continue with the remaining helper methods in the next message... + +func (c *ABCIExecutionClient) updateState(state types.State, header *types.SignedHeader, data *types.Data, finalizeBlockResponse *abci.ResponseFinalizeBlock, validatorUpdates []*cmtypes.Validator) (types.State, error) { + height := header.Height() + if finalizeBlockResponse.ConsensusParamUpdates != nil { + nextParamsProto, appVersion, err := c.updateConsensusParams(height, types.ConsensusParamsFromProto(state.ConsensusParams), finalizeBlockResponse.ConsensusParamUpdates) + if err != nil { + return types.State{}, err + } + // Change results from this height but only applies to the next height. + state.LastHeightConsensusParamsChanged = height + 1 + state.Version.Consensus.App = appVersion + state.ConsensusParams = nextParamsProto + } + + nValSet := state.NextValidators.Copy() + lastHeightValSetChanged := state.LastHeightValidatorsChanged + + if len(nValSet.Validators) > 0 { + err := nValSet.UpdateWithChangeSet(validatorUpdates) + if err != nil { + if err.Error() != ErrEmptyValSetGenerated.Error() { + return state, err + } + nValSet = &cmtypes.ValidatorSet{ + Validators: make([]*cmtypes.Validator, 0), + Proposer: nil, + } + } + // Change results from this height but only applies to the next next height. + lastHeightValSetChanged = int64(header.Header.Height() + 1 + 1) + + if len(nValSet.Validators) > 0 { + nValSet.IncrementProposerPriority(1) + } + } + + s := types.State{ + Version: state.Version, + ChainID: state.ChainID, + InitialHeight: state.InitialHeight, + LastBlockHeight: height, + LastBlockTime: header.Time(), + LastBlockID: cmtypes.BlockID{ + Hash: cmbytes.HexBytes(header.Hash()), + // for now, we don't care about part set headers + }, + ConsensusParams: state.ConsensusParams, + LastHeightConsensusParamsChanged: state.LastHeightConsensusParamsChanged, + AppHash: finalizeBlockResponse.AppHash, + Validators: state.NextValidators.Copy(), + NextValidators: nValSet, + LastHeightValidatorsChanged: lastHeightValSetChanged, + LastValidators: state.Validators.Copy(), + } + copy(s.LastResultsHash[:], cmtypes.NewResults(finalizeBlockResponse.TxResults).Hash()) + + return s, nil +} + +func (c *ABCIExecutionClient) commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { + c.mempool.Lock() + defer c.mempool.Unlock() + + err := c.mempool.FlushAppConn() + if err != nil { + return nil, 0, err + } + + commitResp, err := c.proxyApp.Commit(ctx) + if err != nil { + return nil, 0, err + } + + maxBytes := state.ConsensusParams.Block.MaxBytes + maxGas := state.ConsensusParams.Block.MaxGas + cTxs := fromRollkitTxs(data.Txs) + c.mempoolReaper.UpdateCommitedTxs(cTxs) + err = c.mempool.Update(header.Height(), cTxs, resp.TxResults, mempool.PreCheckMaxBytes(maxBytes), mempool.PostCheckMaxGas(maxGas)) + if err != nil { + return nil, 0, err + } + + return resp.AppHash, uint64(commitResp.RetainHeight), err +} + +// Validate validates the state and the block for the executor +func (c *ABCIExecutionClient) Validate(state types.State, header *types.SignedHeader, data *types.Data) error { + if err := header.ValidateBasic(); err != nil { + return err + } + if err := data.ValidateBasic(); err != nil { + return err + } + if err := types.Validate(header, data); err != nil { + return err + } + if header.Version.App != state.Version.Consensus.App || + header.Version.Block != state.Version.Consensus.Block { + return errors.New("block version mismatch") + } + if state.LastBlockHeight <= 0 && header.Height() != state.InitialHeight { + return errors.New("initial block height mismatch") + } + if state.LastBlockHeight > 0 && header.Height() != state.LastBlockHeight+1 { + return errors.New("block height mismatch") + } + if !bytes.Equal(header.AppHash[:], state.AppHash[:]) { + return errors.New("AppHash mismatch") + } + + if !bytes.Equal(header.LastResultsHash[:], state.LastResultsHash[:]) { + return errors.New("LastResultsHash mismatch") + } + + return nil +} + +func (c *ABCIExecutionClient) execute(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data) (*abci.ResponseFinalizeBlock, error) { + // Only execute if the node hasn't already shut down + select { + case <-ctx.Done(): + return nil, ctx.Err() + default: + } + abciHeader, err := abciconv.ToABCIHeaderPB(&header.Header) + if err != nil { + return nil, err + } + abciHeader.ChainID = c.chainID + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return nil, err + } + + startTime := time.Now().UnixNano() + finalizeBlockResponse, err := c.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ + Hash: header.Hash(), + NextValidatorsHash: state.Validators.Hash(), + ProposerAddress: abciHeader.ProposerAddress, + Height: abciHeader.Height, + Time: abciHeader.Time, + DecidedLastCommit: abci.CommitInfo{ + Round: 0, + Votes: nil, + }, + Misbehavior: abciBlock.Evidence.Evidence.ToABCI(), + Txs: abciBlock.Txs.ToSliceOfBytes(), + }) + endTime := time.Now().UnixNano() + c.metrics.BlockProcessingTime.Observe(float64(endTime-startTime) / 1000000) + if err != nil { + c.logger.Error("error in proxyAppConn.FinalizeBlock", "err", err) + return nil, err + } + + c.logger.Info( + "finalized block", + "height", abciBlock.Height, + "num_txs_res", len(finalizeBlockResponse.TxResults), + "num_val_updates", len(finalizeBlockResponse.ValidatorUpdates), + "block_app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash), + ) + + // Assert that the application correctly returned tx results for each of the transactions provided in the block + if len(abciBlock.Data.Txs) != len(finalizeBlockResponse.TxResults) { + return nil, fmt.Errorf("expected tx results length to match size of transactions in block. Expected %d, got %d", len(data.Txs), len(finalizeBlockResponse.TxResults)) + } + + c.logger.Info("executed block", "height", abciHeader.Height, "app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash)) + + return finalizeBlockResponse, nil +} + +func (c *ABCIExecutionClient) publishEvents(resp *abci.ResponseFinalizeBlock, header *types.SignedHeader, data *types.Data, state types.State) { + if c.eventBus == nil { + return + } + + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return + } + + if err := c.eventBus.PublishEventNewBlock(cmtypes.EventDataNewBlock{ + Block: abciBlock, + BlockID: cmtypes.BlockID{ + Hash: cmbytes.HexBytes(header.Hash()), + // for now, we don't care about part set headers + }, + ResultFinalizeBlock: *resp, + }); err != nil { + c.logger.Error("failed publishing new block", "err", err) + } + + if err := c.eventBus.PublishEventNewBlockHeader(cmtypes.EventDataNewBlockHeader{ + Header: abciBlock.Header, + }); err != nil { + c.logger.Error("failed publishing new block header", "err", err) + } + + if err := c.eventBus.PublishEventNewBlockEvents(cmtypes.EventDataNewBlockEvents{ + Height: abciBlock.Height, + Events: resp.Events, + NumTxs: int64(len(abciBlock.Txs)), + }); err != nil { + c.logger.Error("failed publishing new block events", "err", err) + } + + if len(abciBlock.Evidence.Evidence) != 0 { + for _, ev := range abciBlock.Evidence.Evidence { + if err := c.eventBus.PublishEventNewEvidence(cmtypes.EventDataNewEvidence{ + Evidence: ev, + Height: int64(header.Header.Height()), + }); err != nil { + c.logger.Error("failed publishing new evidence", "err", err) + } + } + } + + for i, tx := range abciBlock.Data.Txs { + err := c.eventBus.PublishEventTx(cmtypes.EventDataTx{ + TxResult: abci.TxResult{ + Height: abciBlock.Height, + Index: uint32(i), + Tx: tx, + Result: *(resp.TxResults[i]), + }, + }) + if err != nil { + c.logger.Error("failed publishing event TX", "err", err) + } + } +} + +func toRollkitTxs(txs cmtypes.Txs) types.Txs { + rollkitTxs := make(types.Txs, len(txs)) + for i := range txs { + rollkitTxs[i] = []byte(txs[i]) + } + return rollkitTxs +} + +func fromRollkitTxs(rollkitTxs types.Txs) cmtypes.Txs { + txs := make(cmtypes.Txs, len(rollkitTxs)) + for i := range rollkitTxs { + txs[i] = []byte(rollkitTxs[i]) + } + return txs } diff --git a/proxy/client.go b/proxy/client.go deleted file mode 100644 index e018b60..0000000 --- a/proxy/client.go +++ /dev/null @@ -1,731 +0,0 @@ -package proxy - -import ( - "bytes" - "context" - "errors" - "fmt" - "time" - - abci "github.com/cometbft/cometbft/abci/types" - cmbytes "github.com/cometbft/cometbft/libs/bytes" - cmproto "github.com/cometbft/cometbft/proto/tendermint/types" - "github.com/cometbft/cometbft/proxy" - cmtypes "github.com/cometbft/cometbft/types" - execution "github.com/rollkit/go-execution" - proxy_grpc "github.com/rollkit/go-execution/proxy/grpc" - "github.com/rollkit/rollkit/mempool" - "github.com/rollkit/rollkit/state" - "github.com/rollkit/rollkit/store" - "github.com/rollkit/rollkit/third_party/log" - "github.com/rollkit/rollkit/types" - abciconv "github.com/rollkit/rollkit/types/abci" -) - -// ErrEmptyValSetGenerated is returned when applying the validator changes would result in empty set. -var ErrEmptyValSetGenerated = errors.New("applying the validator changes would result in empty set") - -// ErrAddingValidatorToBased is returned when trying to add a validator to an empty validator set. -var ErrAddingValidatorToBased = errors.New("cannot add validators to empty validator set") - -// Ensure ProxyClient implements Execute interface -var _ execution.Execute = (*ProxyClient)(nil) - -// ProxyClient implements the Execute interface in go-execution -type ProxyClient struct { - client *proxy_grpc.Client - // abci specific - proxyApp proxy.AppConnConsensus - eventBus *cmtypes.EventBus - genesis *cmtypes.GenesisDoc - maxBytes uint64 - proposerAddress []byte - chainID string - - // rollkit specific - mempool mempool.Mempool - mempoolReaper *mempool.CListMempoolReaper - logger log.Logger - metrics *state.Metrics - state *types.State - store store.Store -} - -func NewClient(client *proxy_grpc.Client, proxyApp proxy.AppConnConsensus, eventBus *cmtypes.EventBus, genesis *cmtypes.GenesisDoc, maxBytes uint64, proposerAddress []byte, chainID string, mempool mempool.Mempool, mempoolReaper *mempool.CListMempoolReaper, logger log.Logger, metrics *state.Metrics, state *types.State, store store.Store) *ProxyClient { - return &ProxyClient{ - client: client, - proxyApp: proxyApp, - eventBus: eventBus, - genesis: genesis, - maxBytes: maxBytes, - proposerAddress: proposerAddress, - chainID: chainID, - mempool: mempool, - mempoolReaper: mempoolReaper, - logger: logger, - metrics: metrics, - state: state, - store: store, - } -} - -// InitChain initializes the blockchain with genesis information. -func (c *ProxyClient) InitChain( - genesisTime time.Time, - initialHeight uint64, - chainID string, -) (types.Hash, uint64, error) { - genesis := &cmtypes.GenesisDoc{ - GenesisTime: genesisTime, - ChainID: chainID, - ConsensusParams: c.genesis.ConsensusParams, - Validators: c.genesis.Validators, - AppState: c.genesis.AppState, - InitialHeight: int64(initialHeight), - } - - response, err := c.initChain(genesis) - if err != nil { - return types.Hash{}, 0, err - } - - stateRoot := types.Hash(response.AppHash) - maxBytes := response.ConsensusParams.Block.MaxBytes - - return stateRoot, uint64(maxBytes), nil -} - -// initChain calls InitChainSync using consensus connection to app. -func (c *ProxyClient) initChain(genesis *cmtypes.GenesisDoc) (*abci.ResponseInitChain, error) { - params := genesis.ConsensusParams - - validators := make([]*cmtypes.Validator, len(genesis.Validators)) - for i, v := range genesis.Validators { - validators[i] = cmtypes.NewValidator(v.PubKey, v.Power) - } - - return c.proxyApp.InitChain(context.Background(), &abci.RequestInitChain{ - Time: genesis.GenesisTime, - ChainId: genesis.ChainID, - ConsensusParams: &cmproto.ConsensusParams{ - Block: &cmproto.BlockParams{ - MaxBytes: params.Block.MaxBytes, - MaxGas: params.Block.MaxGas, - }, - Evidence: &cmproto.EvidenceParams{ - MaxAgeNumBlocks: params.Evidence.MaxAgeNumBlocks, - MaxAgeDuration: params.Evidence.MaxAgeDuration, - MaxBytes: params.Evidence.MaxBytes, - }, - Validator: &cmproto.ValidatorParams{ - PubKeyTypes: params.Validator.PubKeyTypes, - }, - Version: &cmproto.VersionParams{ - App: params.Version.App, - }, - Abci: &cmproto.ABCIParams{ - VoteExtensionsEnableHeight: params.ABCI.VoteExtensionsEnableHeight, - }, - }, - Validators: cmtypes.TM2PB.ValidatorUpdates(cmtypes.NewValidatorSet(validators)), - AppStateBytes: genesis.AppState, - InitialHeight: genesis.InitialHeight, - }) -} - -// GetTxs retrieves all available transactions from the mempool. -func (c *ProxyClient) GetTxs() ([]types.Tx, error) { - state, err := c.store.GetState(context.Background()) - if err != nil { - return nil, fmt.Errorf("failed to get current state: %w", err) - } - - maxBytes := state.ConsensusParams.Block.MaxBytes - if maxBytes == -1 { - maxBytes = int64(cmtypes.MaxBlockSizeBytes) - } - if maxBytes > int64(c.maxBytes) { - c.logger.Debug("limiting maxBytes to", "maxBytes", c.maxBytes) - maxBytes = int64(c.maxBytes) - } - - cmTxs := c.mempool.ReapMaxTxs(int(maxBytes)) - - rollkitTxs := make([]types.Tx, len(cmTxs)) - for i, tx := range cmTxs { - rollkitTxs[i] = types.Tx(tx) - } - - return rollkitTxs, nil -} - -// ExecuteTxs executes a set of transactions to produce a new block. -func (c *ProxyClient) ExecuteTxs( - txs []types.Tx, - blockHeight uint64, - timestamp time.Time, - prevStateRoot types.Hash, -) (types.Hash, uint64, error) { - ctx := context.Background() - - state, err := c.store.GetState(ctx) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to get current state: %w", err) - } - - cmTxs := fromRollkitTxs(txs) - - var lastSignature *types.Signature - var lastHeaderHash types.Hash - var lastExtendedCommit abci.ExtendedCommitInfo - - if blockHeight == uint64(c.genesis.InitialHeight) { - lastSignature = &types.Signature{} - lastHeaderHash = types.Hash{} - lastExtendedCommit = abci.ExtendedCommitInfo{} - } else { - lastSignature, err = c.store.GetSignature(ctx, blockHeight-1) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("error while loading last commit: %w", err) - } - - lastHeader, _, err := c.store.GetBlockData(ctx, blockHeight-1) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("error while loading last block: %w", err) - } - lastHeaderHash = lastHeader.Hash() - - extCommit, err := c.store.GetExtendedCommit(ctx, blockHeight-1) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to load extended commit for height %d: %w", blockHeight-1, err) - } - if extCommit != nil { - lastExtendedCommit = *extCommit - } - } - - header, data, err := c.CreateBlock( - blockHeight, - lastSignature, - lastExtendedCommit, - lastHeaderHash, - state, - cmTxs, - timestamp, - ) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to create block: %w", err) - } - - isValid, err := c.ProcessProposal(header, data, state) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to process proposal: %w", err) - } - if !isValid { - return types.Hash{}, 0, fmt.Errorf("proposal was not valid") - } - - newState, resp, err := c.ApplyBlock(ctx, state, header, data) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to apply block: %w", err) - } - - appHash, _, err := c.Commit(ctx, newState, header, data, resp) - if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to commit: %w", err) - } - - return types.Hash(appHash), uint64(newState.ConsensusParams.Block.MaxBytes), nil -} - -// SetFinal marks a block at the given height as final. -func (c *ProxyClient) SetFinal(blockHeight uint64) error { - ctx := context.Background() - - header, data, err := c.store.GetBlockData(ctx, blockHeight) - if err != nil { - return fmt.Errorf("failed to get block data for height %d: %w", blockHeight, err) - } - - state, err := c.store.GetState(ctx) - if err != nil { - return fmt.Errorf("failed to get current state: %w", err) - } - - resp, err := c.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ - Hash: header.Hash(), - Height: int64(blockHeight), - Time: header.Time(), - Txs: data.Txs.ToSliceOfBytes(), - ProposerAddress: header.ProposerAddress, - NextValidatorsHash: state.Validators.Hash(), - }) - if err != nil { - return fmt.Errorf("failed to finalize block at height %d: %w", blockHeight, err) - } - - state.AppHash = resp.AppHash - if err := c.store.UpdateState(ctx, state); err != nil { - return fmt.Errorf("failed to update state after finalizing block %d: %w", blockHeight, err) - } - - c.logger.Info("Block finalized", "height", blockHeight, "hash", header.Hash()) - - return nil -} - -// CreateBlock reaps transactions from mempool and builds a block. -func (c *ProxyClient) CreateBlock(height uint64, lastSignature *types.Signature, lastExtendedCommit abci.ExtendedCommitInfo, lastHeaderHash types.Hash, state types.State, txs cmtypes.Txs, timestamp time.Time) (*types.SignedHeader, *types.Data, error) { - maxBytes := state.ConsensusParams.Block.MaxBytes - emptyMaxBytes := maxBytes == -1 - if emptyMaxBytes { - maxBytes = int64(cmtypes.MaxBlockSizeBytes) - } - if maxBytes > int64(c.maxBytes) { //nolint:gosec - c.logger.Debug("limiting maxBytes to", "e.maxBytes=%d", c.maxBytes) - maxBytes = int64(c.maxBytes) //nolint:gosec - } - - header := &types.SignedHeader{ - Header: types.Header{ - Version: types.Version{ - Block: state.Version.Consensus.Block, - App: state.Version.Consensus.App, - }, - BaseHeader: types.BaseHeader{ - ChainID: c.chainID, - Height: height, - Time: uint64(timestamp.UnixNano()), //nolint:gosec - }, - DataHash: make(types.Hash, 32), - ConsensusHash: make(types.Hash, 32), - AppHash: state.AppHash, - LastResultsHash: state.LastResultsHash, - ProposerAddress: c.proposerAddress, - }, - Signature: *lastSignature, - } - data := &types.Data{ - Txs: toRollkitTxs(txs), - // IntermediateStateRoots: types.IntermediateStateRoots{RawRootsList: nil}, - // Note: Temporarily remove Evidence #896 - // Evidence: types.EvidenceData{Evidence: nil}, - } - - rpp, err := c.proxyApp.PrepareProposal( - context.TODO(), - &abci.RequestPrepareProposal{ - MaxTxBytes: maxBytes, - Txs: txs.ToSliceOfBytes(), - LocalLastCommit: lastExtendedCommit, - Misbehavior: []abci.Misbehavior{}, - Height: int64(header.Height()), //nolint:gosec - Time: header.Time(), //TODO: replace with sequencer timestamp - NextValidatorsHash: state.Validators.Hash(), - ProposerAddress: c.proposerAddress, - }, - ) - if err != nil { - // The App MUST ensure that only valid (and hence 'processable') transactions - // enter the mempool. Hence, at this point, we can't have any non-processable - // transaction causing an error. - // - // Also, the App can simply skip any transaction that could cause any kind of trouble. - // Either way, we cannot recover in a meaningful way, unless we skip proposing - // this block, repair what caused the error and try again. Hence, we return an - // error for now (the production code calling this function is expected to panic). - return nil, nil, err - } - - txl := cmtypes.ToTxs(rpp.Txs) - if err := txl.Validate(maxBytes); err != nil { - return nil, nil, err - } - - data.Txs = toRollkitTxs(txl) - // Note: This is hash of an ABCI type commit equivalent of the last signature in the signed header. - header.LastCommitHash = lastSignature.GetCommitHash(&header.Header, c.proposerAddress) - header.LastHeaderHash = lastHeaderHash - - return header, data, nil -} - -// ProcessProposal calls the corresponding ABCI method on the app. -func (c *ProxyClient) ProcessProposal( - header *types.SignedHeader, - data *types.Data, - state types.State, -) (bool, error) { - resp, err := c.proxyApp.ProcessProposal(context.TODO(), &abci.RequestProcessProposal{ - Hash: header.Hash(), - Height: int64(header.Height()), //nolint:gosec - Time: header.Time(), - Txs: data.Txs.ToSliceOfBytes(), - ProposedLastCommit: abci.CommitInfo{ - Round: 0, - Votes: []abci.VoteInfo{ - { - Validator: abci.Validator{ - Address: header.Validators.GetProposer().Address, - Power: header.Validators.GetProposer().VotingPower, - }, - BlockIdFlag: cmproto.BlockIDFlagCommit, - }, - }, - }, - Misbehavior: []abci.Misbehavior{}, - ProposerAddress: c.proposerAddress, - NextValidatorsHash: state.Validators.Hash(), - }) - if err != nil { - return false, err - } - if resp.IsStatusUnknown() { - panic(fmt.Sprintf("ProcessProposal responded with status %s", resp.Status.String())) - } - - return resp.IsAccepted(), nil -} - -// ApplyBlock validates and executes the block. -func (c *ProxyClient) ApplyBlock(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data) (types.State, *abci.ResponseFinalizeBlock, error) { - isAppValid, err := c.ProcessProposal(header, data, state) - if err != nil { - return types.State{}, nil, err - } - if !isAppValid { - return types.State{}, nil, fmt.Errorf("proposal processing resulted in an invalid application state") - } - - err = c.Validate(state, header, data) - if err != nil { - return types.State{}, nil, err - } - // This makes calls to the AppClient - resp, err := c.execute(ctx, state, header, data) - if err != nil { - return types.State{}, nil, err - } - abciValUpdates := resp.ValidatorUpdates - - validatorUpdates, err := cmtypes.PB2TM.ValidatorUpdates(abciValUpdates) - if err != nil { - return state, nil, err - } - - if resp.ConsensusParamUpdates != nil { - c.metrics.ConsensusParamUpdates.Add(1) - } - - state, err = c.updateState(state, header, data, resp, validatorUpdates) - if err != nil { - return types.State{}, nil, err - } - - if state.ConsensusParams.Block.MaxBytes <= 0 { - c.logger.Error("maxBytes<=0", "state.ConsensusParams.Block", state.ConsensusParams.Block, "header", header) - } - - return state, resp, nil -} - -// ExtendVote calls the ExtendVote ABCI method on the proxy app. -func (c *ProxyClient) ExtendVote(ctx context.Context, header *types.SignedHeader, data *types.Data) ([]byte, error) { - resp, err := c.proxyApp.ExtendVote(ctx, &abci.RequestExtendVote{ - Hash: header.Hash(), - Height: int64(header.Height()), //nolint:gosec - Time: header.Time(), - Txs: data.Txs.ToSliceOfBytes(), - ProposedLastCommit: abci.CommitInfo{ - Votes: []abci.VoteInfo{{ - Validator: abci.Validator{ - Address: header.Validators.GetProposer().Address, - Power: header.Validators.GetProposer().VotingPower, - }, - BlockIdFlag: cmproto.BlockIDFlagCommit, - }}, - }, - Misbehavior: nil, - NextValidatorsHash: header.ValidatorHash, - ProposerAddress: header.ProposerAddress, - }) - if err != nil { - return nil, err - } - return resp.VoteExtension, nil -} - -// Commit commits the block -func (c *ProxyClient) Commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { - appHash, retainHeight, err := c.commit(ctx, state, header, data, resp) - if err != nil { - return []byte{}, 0, err - } - - state.AppHash = appHash - - c.publishEvents(resp, header, data, state) - - return appHash, retainHeight, nil -} - -// updateConsensusParams updates the consensus parameters based on the provided updates. -func (c *ProxyClient) updateConsensusParams(height uint64, params cmtypes.ConsensusParams, consensusParamUpdates *cmproto.ConsensusParams) (cmproto.ConsensusParams, uint64, error) { - nextParams := params.Update(consensusParamUpdates) - if err := types.ConsensusParamsValidateBasic(nextParams); err != nil { - return cmproto.ConsensusParams{}, 0, fmt.Errorf("validating new consensus params: %w", err) - } - if err := nextParams.ValidateUpdate(consensusParamUpdates, int64(height)); err != nil { //nolint:gosec - return cmproto.ConsensusParams{}, 0, fmt.Errorf("updating consensus params: %w", err) - } - return nextParams.ToProto(), nextParams.Version.App, nil -} - -func (c *ProxyClient) updateState(state types.State, header *types.SignedHeader, data *types.Data, finalizeBlockResponse *abci.ResponseFinalizeBlock, validatorUpdates []*cmtypes.Validator) (types.State, error) { - height := header.Height() - if finalizeBlockResponse.ConsensusParamUpdates != nil { - nextParamsProto, appVersion, err := c.updateConsensusParams(height, types.ConsensusParamsFromProto(state.ConsensusParams), finalizeBlockResponse.ConsensusParamUpdates) - if err != nil { - return types.State{}, err - } - // Change results from this height but only applies to the next height. - state.LastHeightConsensusParamsChanged = height + 1 - state.Version.Consensus.App = appVersion - state.ConsensusParams = nextParamsProto - } - - nValSet := state.NextValidators.Copy() - lastHeightValSetChanged := state.LastHeightValidatorsChanged - - if len(nValSet.Validators) > 0 { - err := nValSet.UpdateWithChangeSet(validatorUpdates) - if err != nil { - if err.Error() != ErrEmptyValSetGenerated.Error() { - return state, err - } - nValSet = &cmtypes.ValidatorSet{ - Validators: make([]*cmtypes.Validator, 0), - Proposer: nil, - } - } - // Change results from this height but only applies to the next next height. - lastHeightValSetChanged = int64(header.Header.Height() + 1 + 1) //nolint:gosec - - if len(nValSet.Validators) > 0 { - nValSet.IncrementProposerPriority(1) - } - } - - s := types.State{ - Version: state.Version, - ChainID: state.ChainID, - InitialHeight: state.InitialHeight, - LastBlockHeight: height, - LastBlockTime: header.Time(), - LastBlockID: cmtypes.BlockID{ - Hash: cmbytes.HexBytes(header.Hash()), - // for now, we don't care about part set headers - }, - ConsensusParams: state.ConsensusParams, - LastHeightConsensusParamsChanged: state.LastHeightConsensusParamsChanged, - AppHash: finalizeBlockResponse.AppHash, - Validators: state.NextValidators.Copy(), - NextValidators: nValSet, - LastHeightValidatorsChanged: lastHeightValSetChanged, - LastValidators: state.Validators.Copy(), - } - copy(s.LastResultsHash[:], cmtypes.NewResults(finalizeBlockResponse.TxResults).Hash()) - - return s, nil -} - -func (c *ProxyClient) commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { - c.mempool.Lock() - defer c.mempool.Unlock() - - err := c.mempool.FlushAppConn() - if err != nil { - return nil, 0, err - } - - commitResp, err := c.proxyApp.Commit(ctx) - if err != nil { - return nil, 0, err - } - - maxBytes := state.ConsensusParams.Block.MaxBytes - maxGas := state.ConsensusParams.Block.MaxGas - cTxs := fromRollkitTxs(data.Txs) - c.mempoolReaper.UpdateCommitedTxs(cTxs) - err = c.mempool.Update(header.Height(), cTxs, resp.TxResults, mempool.PreCheckMaxBytes(maxBytes), mempool.PostCheckMaxGas(maxGas)) - if err != nil { - return nil, 0, err - } - - return resp.AppHash, uint64(commitResp.RetainHeight), err //nolint:gosec -} - -// Validate validates the state and the block for the executor -func (c *ProxyClient) Validate(state types.State, header *types.SignedHeader, data *types.Data) error { - if err := header.ValidateBasic(); err != nil { - return err - } - if err := data.ValidateBasic(); err != nil { - return err - } - if err := types.Validate(header, data); err != nil { - return err - } - if header.Version.App != state.Version.Consensus.App || - header.Version.Block != state.Version.Consensus.Block { - return errors.New("block version mismatch") - } - if state.LastBlockHeight <= 0 && header.Height() != state.InitialHeight { - return errors.New("initial block height mismatch") - } - if state.LastBlockHeight > 0 && header.Height() != state.LastBlockHeight+1 { - return errors.New("block height mismatch") - } - if !bytes.Equal(header.AppHash[:], state.AppHash[:]) { - return errors.New("AppHash mismatch") - } - - if !bytes.Equal(header.LastResultsHash[:], state.LastResultsHash[:]) { - return errors.New("LastResultsHash mismatch") - } - - return nil -} - -func (c *ProxyClient) execute(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data) (*abci.ResponseFinalizeBlock, error) { - // Only execute if the node hasn't already shut down - select { - case <-ctx.Done(): - return nil, ctx.Err() - default: - } - abciHeader, err := abciconv.ToABCIHeaderPB(&header.Header) - if err != nil { - return nil, err - } - abciHeader.ChainID = c.chainID - abciBlock, err := abciconv.ToABCIBlock(header, data) - if err != nil { - return nil, err - } - - startTime := time.Now().UnixNano() - finalizeBlockResponse, err := c.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ - Hash: header.Hash(), - NextValidatorsHash: state.Validators.Hash(), - ProposerAddress: abciHeader.ProposerAddress, - Height: abciHeader.Height, - Time: abciHeader.Time, - DecidedLastCommit: abci.CommitInfo{ - Round: 0, - Votes: nil, - }, - Misbehavior: abciBlock.Evidence.Evidence.ToABCI(), - Txs: abciBlock.Txs.ToSliceOfBytes(), - }) - endTime := time.Now().UnixNano() - c.metrics.BlockProcessingTime.Observe(float64(endTime-startTime) / 1000000) - if err != nil { - c.logger.Error("error in proxyAppConn.FinalizeBlock", "err", err) - return nil, err - } - - c.logger.Info( - "finalized block", - "height", abciBlock.Height, - "num_txs_res", len(finalizeBlockResponse.TxResults), - "num_val_updates", len(finalizeBlockResponse.ValidatorUpdates), - "block_app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash), - ) - - // Assert that the application correctly returned tx results for each of the transactions provided in the block - if len(abciBlock.Data.Txs) != len(finalizeBlockResponse.TxResults) { - return nil, fmt.Errorf("expected tx results length to match size of transactions in block. Expected %d, got %d", len(data.Txs), len(finalizeBlockResponse.TxResults)) - } - - c.logger.Info("executed block", "height", abciHeader.Height, "app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash)) - - return finalizeBlockResponse, nil -} - -func (e *ProxyClient) publishEvents(resp *abci.ResponseFinalizeBlock, header *types.SignedHeader, data *types.Data, state types.State) { - if e.eventBus == nil { - return - } - - abciBlock, err := abciconv.ToABCIBlock(header, data) - if err != nil { - return - } - - if err := e.eventBus.PublishEventNewBlock(cmtypes.EventDataNewBlock{ - Block: abciBlock, - BlockID: cmtypes.BlockID{ - Hash: cmbytes.HexBytes(header.Hash()), - // for now, we don't care about part set headers - }, - ResultFinalizeBlock: *resp, - }); err != nil { - e.logger.Error("failed publishing new block", "err", err) - } - - if err := e.eventBus.PublishEventNewBlockHeader(cmtypes.EventDataNewBlockHeader{ - Header: abciBlock.Header, - }); err != nil { - e.logger.Error("failed publishing new block header", "err", err) - } - - if err := e.eventBus.PublishEventNewBlockEvents(cmtypes.EventDataNewBlockEvents{ - Height: abciBlock.Height, - Events: resp.Events, - NumTxs: int64(len(abciBlock.Txs)), - }); err != nil { - e.logger.Error("failed publishing new block events", "err", err) - } - - if len(abciBlock.Evidence.Evidence) != 0 { - for _, ev := range abciBlock.Evidence.Evidence { - if err := e.eventBus.PublishEventNewEvidence(cmtypes.EventDataNewEvidence{ - Evidence: ev, - Height: int64(header.Header.Height()), //nolint:gosec - }); err != nil { - e.logger.Error("failed publishing new evidence", "err", err) - } - } - } - - for i, tx := range abciBlock.Data.Txs { - err := e.eventBus.PublishEventTx(cmtypes.EventDataTx{ - TxResult: abci.TxResult{ - Height: abciBlock.Height, - Index: uint32(i), //nolint:gosec - Tx: tx, - Result: *(resp.TxResults[i]), - }, - }) - if err != nil { - e.logger.Error("failed publishing event TX", "err", err) - } - } -} - -func toRollkitTxs(txs cmtypes.Txs) types.Txs { - rollkitTxs := make(types.Txs, len(txs)) - for i := range txs { - rollkitTxs[i] = []byte(txs[i]) - } - return rollkitTxs -} - -func fromRollkitTxs(rollkitTxs types.Txs) cmtypes.Txs { - txs := make(cmtypes.Txs, len(rollkitTxs)) - for i := range rollkitTxs { - txs[i] = []byte(rollkitTxs[i]) - } - return txs -} From 06035993c19ef74b09aff702a14683a0cb2d585a Mon Sep 17 00:00:00 2001 From: Jay Jie Date: Wed, 6 Nov 2024 15:48:55 -0800 Subject: [PATCH 07/10] chore: update to go-execution tip --- go.mod | 2 +- go.sum | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/go.mod b/go.mod index c02dd79..ec8a0ab 100644 --- a/go.mod +++ b/go.mod @@ -4,7 +4,7 @@ go 1.22.7 toolchain go1.22.8 -replace github.com/rollkit/go-execution => github.com/lastdotnet/go-execution v0.0.0-20241029045146-b7513b533b24 +replace github.com/rollkit/go-execution => github.com/lastdotnet/go-execution v0.0.0-20241106160445-7810bc1e5d3c require ( github.com/cometbft/cometbft v0.38.13 diff --git a/go.sum b/go.sum index 50e413b..9096fcc 100644 --- a/go.sum +++ b/go.sum @@ -180,8 +180,8 @@ github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= -github.com/lastdotnet/go-execution v0.0.0-20241029045146-b7513b533b24 h1:F/XQxFKbZA1w8daNqHdh/8waKQfWLAyZTmkQVVxlgVk= -github.com/lastdotnet/go-execution v0.0.0-20241029045146-b7513b533b24/go.mod h1:fo+uH57fdmebAEYEVtLhsSJ7Jw8CKRSydCnyqJZRBAA= +github.com/lastdotnet/go-execution v0.0.0-20241106160445-7810bc1e5d3c h1:A8/nKchjQ5waN5PEwVNsTaAdTm2iukmjRlcLcrwunjA= +github.com/lastdotnet/go-execution v0.0.0-20241106160445-7810bc1e5d3c/go.mod h1:WgzV+v1zOka30IQbuE9Rmk7xkfF3xhCmptpit2v5Zo0= github.com/libp2p/go-buffer-pool v0.1.0 h1:oK4mSFcQz7cTQIfqbe4MIj9gLW+mnanjyFtc6cdF0Y8= github.com/libp2p/go-buffer-pool v0.1.0/go.mod h1:N+vh8gMqimBzdKkSMVuydVDq+UV5QTWy5HSiZacSbPg= github.com/libp2p/go-flow-metrics v0.1.0 h1:0iPhMI8PskQwzh57jB9WxIuIOQ0r+15PChFGkx3Q3WM= From 8d6f65b2147ce53b2b503d11858b41726bf16e6e Mon Sep 17 00:00:00 2001 From: Jay Jie Date: Wed, 6 Nov 2024 15:59:46 -0800 Subject: [PATCH 08/10] chore: update architecture diagram --- README.md | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 5de9bfa..311a615 100644 --- a/README.md +++ b/README.md @@ -7,7 +7,6 @@ graph TB end subgraph "go-execution-abci" AEC[ABCIExecutionClient] - PC[ProxyClient] end subgraph "CometBFT" ABCI[ABCI Application] @@ -15,11 +14,10 @@ graph TB EB[EventBus] end R -->|implements| AEC - AEC -->|Delegates| PC - PC -->|ABCI Calls| ABCI - PC -->|Tx Management| MP - PC -->|Events| EB + AEC -->|ABCI Calls| ABCI + AEC -->|Tx Management| MP + AEC -->|Events| EB classDef default fill:#f9f9f9,stroke:#333,stroke-width:2px; classDef highlight fill:#e1f7d5,stroke:#333,stroke-width:2px; - class AEC,PC highlight; + class AEC highlight; ``` From c99483f569c03ca1a81dd9788136ea0a3a4e6712 Mon Sep 17 00:00:00 2001 From: Jay Jie Date: Fri, 8 Nov 2024 17:41:26 -0800 Subject: [PATCH 09/10] refactor: client implementation cleanup --- execution.go | 388 ++++++++++++++++++++++++++++++++++++++------------- go.mod | 4 +- go.sum | 8 +- 3 files changed, 296 insertions(+), 104 deletions(-) diff --git a/execution.go b/execution.go index 80666ce..b8844f2 100644 --- a/execution.go +++ b/execution.go @@ -12,7 +12,9 @@ import ( cmproto "github.com/cometbft/cometbft/proto/tendermint/types" "github.com/cometbft/cometbft/proxy" cmtypes "github.com/cometbft/cometbft/types" + execution "github.com/rollkit/go-execution" proxy_grpc "github.com/rollkit/go-execution/proxy/grpc" + execution_types "github.com/rollkit/go-execution/types" "github.com/rollkit/rollkit/mempool" "github.com/rollkit/rollkit/state" "github.com/rollkit/rollkit/store" @@ -47,6 +49,9 @@ type ABCIExecutionClient struct { store store.Store } +// Ensure ABCIExecutionClient implements the execution.Execute interface +var _ execution.Executor = (*ABCIExecutionClient)(nil) + // NewABCIExecutionClient creates a new ABCIExecutionClient func NewABCIExecutionClient( client *proxy_grpc.Client, @@ -145,7 +150,7 @@ func (c *ABCIExecutionClient) initChain(genesis *cmtypes.GenesisDoc) (*abci.Resp } // GetTxs retrieves all available transactions from the mempool. -func (c *ABCIExecutionClient) GetTxs() ([]types.Tx, error) { +func (c *ABCIExecutionClient) GetTxs() ([]execution_types.Tx, error) { state, err := c.store.GetState(context.Background()) if err != nil { return nil, fmt.Errorf("failed to get current state: %w", err) @@ -162,17 +167,17 @@ func (c *ABCIExecutionClient) GetTxs() ([]types.Tx, error) { cmTxs := c.mempool.ReapMaxTxs(int(maxBytes)) - rollkitTxs := make([]types.Tx, len(cmTxs)) + txs := make([]execution_types.Tx, len(cmTxs)) for i, tx := range cmTxs { - rollkitTxs[i] = types.Tx(tx) + txs[i] = execution_types.Tx(tx) } - return rollkitTxs, nil + return txs, nil } // ExecuteTxs executes the given transactions and returns the new state root and gas used. func (c *ABCIExecutionClient) ExecuteTxs( - txs []types.Tx, + txs []execution_types.Tx, blockHeight uint64, timestamp time.Time, prevStateRoot types.Hash, @@ -181,10 +186,10 @@ func (c *ABCIExecutionClient) ExecuteTxs( state, err := c.store.GetState(ctx) if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to get current state: %w", err) + return types.Hash{}, 0, wrapStateError(err, "get current state") } - cmTxs := fromRollkitTxs(txs) + cmTxs := fromExecutionTxs(txs) var lastSignature *types.Signature var lastHeaderHash types.Hash @@ -197,18 +202,18 @@ func (c *ABCIExecutionClient) ExecuteTxs( } else { lastSignature, err = c.store.GetSignature(ctx, blockHeight-1) if err != nil { - return types.Hash{}, 0, fmt.Errorf("error while loading last commit: %w", err) + return types.Hash{}, 0, wrapBlockError(blockHeight-1, err, "load last commit") } lastHeader, _, err := c.store.GetBlockData(ctx, blockHeight-1) if err != nil { - return types.Hash{}, 0, fmt.Errorf("error while loading last block: %w", err) + return types.Hash{}, 0, wrapBlockError(blockHeight-1, err, "load last block") } lastHeaderHash = lastHeader.Hash() extCommit, err := c.store.GetExtendedCommit(ctx, blockHeight-1) if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to load extended commit for height %d: %w", blockHeight-1, err) + return types.Hash{}, 0, wrapBlockError(blockHeight-1, err, "load extended commit") } if extCommit != nil { lastExtendedCommit = *extCommit @@ -225,25 +230,25 @@ func (c *ABCIExecutionClient) ExecuteTxs( timestamp, ) if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to create block: %w", err) + return types.Hash{}, 0, wrapBlockError(blockHeight, err, "create block") } isValid, err := c.ProcessProposal(header, data, state) if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to process proposal: %w", err) + return types.Hash{}, 0, wrapBlockError(blockHeight, err, "process proposal") } if !isValid { - return types.Hash{}, 0, fmt.Errorf("proposal was not valid") + return types.Hash{}, 0, fmt.Errorf("proposal at height %d was not valid", blockHeight) } newState, resp, err := c.ApplyBlock(ctx, state, header, data) if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to apply block: %w", err) + return types.Hash{}, 0, wrapBlockError(blockHeight, err, "apply block") } appHash, _, err := c.Commit(ctx, newState, header, data, resp) if err != nil { - return types.Hash{}, 0, fmt.Errorf("failed to commit: %w", err) + return types.Hash{}, 0, wrapBlockError(blockHeight, err, "commit block") } return types.Hash(appHash), uint64(newState.ConsensusParams.Block.MaxBytes), nil @@ -255,12 +260,12 @@ func (c *ABCIExecutionClient) SetFinal(blockHeight uint64) error { header, data, err := c.store.GetBlockData(ctx, blockHeight) if err != nil { - return fmt.Errorf("failed to get block data for height %d: %w", blockHeight, err) + return wrapBlockError(blockHeight, err, "get block data") } state, err := c.store.GetState(ctx) if err != nil { - return fmt.Errorf("failed to get current state: %w", err) + return wrapStateError(err, "get current state") } resp, err := c.proxyApp.FinalizeBlock(ctx, &abci.RequestFinalizeBlock{ @@ -272,12 +277,12 @@ func (c *ABCIExecutionClient) SetFinal(blockHeight uint64) error { NextValidatorsHash: state.Validators.Hash(), }) if err != nil { - return fmt.Errorf("failed to finalize block at height %d: %w", blockHeight, err) + return wrapBlockError(blockHeight, err, "finalize block") } state.AppHash = resp.AppHash if err := c.store.UpdateState(ctx, state); err != nil { - return fmt.Errorf("failed to update state after finalizing block %d: %w", blockHeight, err) + return wrapBlockError(blockHeight, err, "update state after finalizing block") } c.logger.Info("Block finalized", "height", blockHeight, "hash", header.Hash()) @@ -469,32 +474,28 @@ func (c *ABCIExecutionClient) Commit(ctx context.Context, state types.State, hea } // updateConsensusParams updates the consensus parameters based on the provided updates. -func (c *ABCIExecutionClient) updateConsensusParams(height uint64, params cmtypes.ConsensusParams, consensusParamUpdates *cmproto.ConsensusParams) (cmproto.ConsensusParams, uint64, error) { - nextParams := params.Update(consensusParamUpdates) +func (c *ABCIExecutionClient) updateConsensusParams( + height uint64, + currentParams cmtypes.ConsensusParams, + updates *cmproto.ConsensusParams, +) (cmproto.ConsensusParams, uint64, error) { + if updates == nil { + return currentParams.ToProto(), currentParams.Version.App, nil + } + + nextParams := currentParams.Update(updates) if err := types.ConsensusParamsValidateBasic(nextParams); err != nil { return cmproto.ConsensusParams{}, 0, fmt.Errorf("validating new consensus params: %w", err) } - if err := nextParams.ValidateUpdate(consensusParamUpdates, int64(height)); err != nil { + + if err := nextParams.ValidateUpdate(updates, int64(height)); err != nil { return cmproto.ConsensusParams{}, 0, fmt.Errorf("updating consensus params: %w", err) } + return nextParams.ToProto(), nextParams.Version.App, nil } -// I'll continue with the remaining helper methods in the next message... - -func (c *ABCIExecutionClient) updateState(state types.State, header *types.SignedHeader, data *types.Data, finalizeBlockResponse *abci.ResponseFinalizeBlock, validatorUpdates []*cmtypes.Validator) (types.State, error) { - height := header.Height() - if finalizeBlockResponse.ConsensusParamUpdates != nil { - nextParamsProto, appVersion, err := c.updateConsensusParams(height, types.ConsensusParamsFromProto(state.ConsensusParams), finalizeBlockResponse.ConsensusParamUpdates) - if err != nil { - return types.State{}, err - } - // Change results from this height but only applies to the next height. - state.LastHeightConsensusParamsChanged = height + 1 - state.Version.Consensus.App = appVersion - state.ConsensusParams = nextParamsProto - } - +func (c *ABCIExecutionClient) updateValidatorSet(state types.State, height uint64, validatorUpdates []*cmtypes.Validator) (*cmtypes.ValidatorSet, int64, error) { nValSet := state.NextValidators.Copy() lastHeightValSetChanged := state.LastHeightValidatorsChanged @@ -502,42 +503,94 @@ func (c *ABCIExecutionClient) updateState(state types.State, header *types.Signe err := nValSet.UpdateWithChangeSet(validatorUpdates) if err != nil { if err.Error() != ErrEmptyValSetGenerated.Error() { - return state, err + return nil, 0, fmt.Errorf("failed to update validator set: %w", err) } nValSet = &cmtypes.ValidatorSet{ Validators: make([]*cmtypes.Validator, 0), Proposer: nil, } } - // Change results from this height but only applies to the next next height. - lastHeightValSetChanged = int64(header.Header.Height() + 1 + 1) + + lastHeightValSetChanged = int64(height + 1 + 1) if len(nValSet.Validators) > 0 { nValSet.IncrementProposerPriority(1) } } - s := types.State{ + return nValSet, lastHeightValSetChanged, nil +} + +func (c *ABCIExecutionClient) createNewState( + state types.State, + header *types.SignedHeader, + resp *abci.ResponseFinalizeBlock, + nValSet *cmtypes.ValidatorSet, + lastHeightValSetChanged int64, + nextParams cmproto.ConsensusParams, + nextParamsHeight uint64, +) types.State { + newState := types.State{ Version: state.Version, ChainID: state.ChainID, InitialHeight: state.InitialHeight, - LastBlockHeight: height, + LastBlockHeight: header.Height(), LastBlockTime: header.Time(), LastBlockID: cmtypes.BlockID{ Hash: cmbytes.HexBytes(header.Hash()), - // for now, we don't care about part set headers }, - ConsensusParams: state.ConsensusParams, - LastHeightConsensusParamsChanged: state.LastHeightConsensusParamsChanged, - AppHash: finalizeBlockResponse.AppHash, + ConsensusParams: nextParams, + LastHeightConsensusParamsChanged: nextParamsHeight, + AppHash: resp.AppHash, Validators: state.NextValidators.Copy(), NextValidators: nValSet, LastHeightValidatorsChanged: lastHeightValSetChanged, LastValidators: state.Validators.Copy(), } - copy(s.LastResultsHash[:], cmtypes.NewResults(finalizeBlockResponse.TxResults).Hash()) - return s, nil + copy(newState.LastResultsHash[:], cmtypes.NewResults(resp.TxResults).Hash()) + return newState +} + +// Update the main updateState function to use these helpers +func (c *ABCIExecutionClient) updateState( + state types.State, + header *types.SignedHeader, + _ *types.Data, + finalizeBlockResponse *abci.ResponseFinalizeBlock, + validatorUpdates []*cmtypes.Validator, +) (types.State, error) { + height := header.Height() + nextParams, appVersion, err := c.updateConsensusParams( + height, + types.ConsensusParamsFromProto(state.ConsensusParams), + finalizeBlockResponse.ConsensusParamUpdates, + ) + if err != nil { + return types.State{}, fmt.Errorf("failed to update consensus params: %w", err) + } + + nValSet, lastHeightValSetChanged, err := c.updateValidatorSet(state, height, validatorUpdates) + if err != nil { + return types.State{}, fmt.Errorf("failed to update validator set: %w", err) + } + + nextParamsHeight := height + 1 + if finalizeBlockResponse.ConsensusParamUpdates != nil { + state.Version.Consensus.App = appVersion + } + + newState := c.createNewState( + state, + header, + finalizeBlockResponse, + nValSet, + lastHeightValSetChanged, + nextParams, + nextParamsHeight, + ) + + return newState, nil } func (c *ABCIExecutionClient) commit(ctx context.Context, state types.State, header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) ([]byte, uint64, error) { @@ -556,9 +609,10 @@ func (c *ABCIExecutionClient) commit(ctx context.Context, state types.State, hea maxBytes := state.ConsensusParams.Block.MaxBytes maxGas := state.ConsensusParams.Block.MaxGas - cTxs := fromRollkitTxs(data.Txs) - c.mempoolReaper.UpdateCommitedTxs(cTxs) - err = c.mempool.Update(header.Height(), cTxs, resp.TxResults, mempool.PreCheckMaxBytes(maxBytes), mempool.PostCheckMaxGas(maxGas)) + execTxs := fromRollkitTxs(data.Txs) + cmTxs := fromExecutionTxs(execTxs) + c.mempoolReaper.UpdateCommitedTxs(cmTxs) + err = c.mempool.Update(header.Height(), cmTxs, resp.TxResults, mempool.PreCheckMaxBytes(maxBytes), mempool.PostCheckMaxGas(maxGas)) if err != nil { return nil, 0, err } @@ -569,30 +623,27 @@ func (c *ABCIExecutionClient) commit(ctx context.Context, state types.State, hea // Validate validates the state and the block for the executor func (c *ABCIExecutionClient) Validate(state types.State, header *types.SignedHeader, data *types.Data) error { if err := header.ValidateBasic(); err != nil { - return err + return fmt.Errorf("invalid header: %w", err) } + if err := data.ValidateBasic(); err != nil { - return err + return fmt.Errorf("invalid data: %w", err) } + if err := types.Validate(header, data); err != nil { - return err - } - if header.Version.App != state.Version.Consensus.App || - header.Version.Block != state.Version.Consensus.Block { - return errors.New("block version mismatch") - } - if state.LastBlockHeight <= 0 && header.Height() != state.InitialHeight { - return errors.New("initial block height mismatch") + return fmt.Errorf("invalid block: %w", err) } - if state.LastBlockHeight > 0 && header.Height() != state.LastBlockHeight+1 { - return errors.New("block height mismatch") + + if err := c.validateVersions(state, header); err != nil { + return err } - if !bytes.Equal(header.AppHash[:], state.AppHash[:]) { - return errors.New("AppHash mismatch") + + if err := c.validateHeight(state, header); err != nil { + return err } - if !bytes.Equal(header.LastResultsHash[:], state.LastResultsHash[:]) { - return errors.New("LastResultsHash mismatch") + if err := c.validateHashes(state, header); err != nil { + return err } return nil @@ -605,6 +656,7 @@ func (c *ABCIExecutionClient) execute(ctx context.Context, state types.State, he return nil, ctx.Err() default: } + abciHeader, err := abciconv.ToABCIHeaderPB(&header.Header) if err != nil { return nil, err @@ -644,7 +696,6 @@ func (c *ABCIExecutionClient) execute(ctx context.Context, state types.State, he "block_app_hash", fmt.Sprintf("%X", finalizeBlockResponse.AppHash), ) - // Assert that the application correctly returned tx results for each of the transactions provided in the block if len(abciBlock.Data.Txs) != len(finalizeBlockResponse.TxResults) { return nil, fmt.Errorf("expected tx results length to match size of transactions in block. Expected %d, got %d", len(data.Txs), len(finalizeBlockResponse.TxResults)) } @@ -654,79 +705,220 @@ func (c *ABCIExecutionClient) execute(ctx context.Context, state types.State, he return finalizeBlockResponse, nil } -func (c *ABCIExecutionClient) publishEvents(resp *abci.ResponseFinalizeBlock, header *types.SignedHeader, data *types.Data, state types.State) { +func (c *ABCIExecutionClient) publishEvents(resp *abci.ResponseFinalizeBlock, header *types.SignedHeader, data *types.Data, _ types.State) { if c.eventBus == nil { return } + if err := c.publishBlockEvent(header, data, resp); err != nil { + c.logger.Error("failed publishing block event", "err", err) + } + + if err := c.publishBlockHeaderEvent(header, data); err != nil { + c.logger.Error("failed publishing block header event", "err", err) + } + + if err := c.publishBlockEventsData(header, data, resp); err != nil { + c.logger.Error("failed publishing block events", "err", err) + } + abciBlock, err := abciconv.ToABCIBlock(header, data) if err != nil { + c.logger.Error("failed to convert block for evidence check", "err", err) return } - if err := c.eventBus.PublishEventNewBlock(cmtypes.EventDataNewBlock{ + if len(abciBlock.Evidence.Evidence) > 0 { + if err := c.publishEvidenceEvents(header, data); err != nil { + c.logger.Error("failed publishing evidence events", "err", err) + } + } + + if err := c.publishTxEvents(header, data, resp); err != nil { + c.logger.Error("failed publishing tx events", "err", err) + } +} + +func (c *ABCIExecutionClient) publishBlockEvent(header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) error { + if c.eventBus == nil { + return nil + } + + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return fmt.Errorf("failed to convert block: %w", err) + } + + return c.eventBus.PublishEventNewBlock(cmtypes.EventDataNewBlock{ Block: abciBlock, BlockID: cmtypes.BlockID{ Hash: cmbytes.HexBytes(header.Hash()), - // for now, we don't care about part set headers }, ResultFinalizeBlock: *resp, - }); err != nil { - c.logger.Error("failed publishing new block", "err", err) + }) +} + +func (c *ABCIExecutionClient) publishBlockHeaderEvent(header *types.SignedHeader, data *types.Data) error { + if c.eventBus == nil { + return nil + } + + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return fmt.Errorf("failed to convert block header: %w", err) } - if err := c.eventBus.PublishEventNewBlockHeader(cmtypes.EventDataNewBlockHeader{ + return c.eventBus.PublishEventNewBlockHeader(cmtypes.EventDataNewBlockHeader{ Header: abciBlock.Header, - }); err != nil { - c.logger.Error("failed publishing new block header", "err", err) + }) +} + +func (c *ABCIExecutionClient) publishBlockEventsData(header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) error { + if c.eventBus == nil { + return nil } - if err := c.eventBus.PublishEventNewBlockEvents(cmtypes.EventDataNewBlockEvents{ + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return fmt.Errorf("failed to convert block events: %w", err) + } + + return c.eventBus.PublishEventNewBlockEvents(cmtypes.EventDataNewBlockEvents{ Height: abciBlock.Height, Events: resp.Events, NumTxs: int64(len(abciBlock.Txs)), - }); err != nil { - c.logger.Error("failed publishing new block events", "err", err) + }) +} + +func (c *ABCIExecutionClient) publishEvidenceEvents(header *types.SignedHeader, data *types.Data) error { + if c.eventBus == nil { + return nil } - if len(abciBlock.Evidence.Evidence) != 0 { - for _, ev := range abciBlock.Evidence.Evidence { - if err := c.eventBus.PublishEventNewEvidence(cmtypes.EventDataNewEvidence{ - Evidence: ev, - Height: int64(header.Header.Height()), - }); err != nil { - c.logger.Error("failed publishing new evidence", "err", err) - } + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return fmt.Errorf("failed to convert block for evidence: %w", err) + } + + for _, ev := range abciBlock.Evidence.Evidence { + if err := c.eventBus.PublishEventNewEvidence(cmtypes.EventDataNewEvidence{ + Evidence: ev, + Height: int64(header.Header.Height()), + }); err != nil { + return fmt.Errorf("failed to publish evidence: %w", err) } } + return nil +} + +func (c *ABCIExecutionClient) publishTxEvents(header *types.SignedHeader, data *types.Data, resp *abci.ResponseFinalizeBlock) error { + if c.eventBus == nil { + return nil + } + + abciBlock, err := abciconv.ToABCIBlock(header, data) + if err != nil { + return fmt.Errorf("failed to convert block for tx events: %w", err) + } + for i, tx := range abciBlock.Data.Txs { - err := c.eventBus.PublishEventTx(cmtypes.EventDataTx{ + if err := c.eventBus.PublishEventTx(cmtypes.EventDataTx{ TxResult: abci.TxResult{ Height: abciBlock.Height, Index: uint32(i), Tx: tx, Result: *(resp.TxResults[i]), }, - }) - if err != nil { - c.logger.Error("failed publishing event TX", "err", err) + }); err != nil { + return fmt.Errorf("failed to publish tx event at index %d: %w", i, err) } } + + return nil } func toRollkitTxs(txs cmtypes.Txs) types.Txs { + if len(txs) == 0 { + return nil + } + rollkitTxs := make(types.Txs, len(txs)) - for i := range txs { - rollkitTxs[i] = []byte(txs[i]) + for i, tx := range txs { + rollkitTxs[i] = types.Tx(tx) } return rollkitTxs } -func fromRollkitTxs(rollkitTxs types.Txs) cmtypes.Txs { - txs := make(cmtypes.Txs, len(rollkitTxs)) - for i := range rollkitTxs { - txs[i] = []byte(rollkitTxs[i]) +func fromRollkitTxs(txs types.Txs) []execution_types.Tx { + if len(txs) == 0 { + return nil + } + + execTxs := make([]execution_types.Tx, len(txs)) + for i, tx := range txs { + execTxs[i] = execution_types.Tx(tx) + } + return execTxs +} + +func fromExecutionTxs(txs []execution_types.Tx) cmtypes.Txs { + if len(txs) == 0 { + return nil + } + + cmTxs := make(cmtypes.Txs, len(txs)) + for i, tx := range txs { + cmTxs[i] = cmtypes.Tx(tx) + } + return cmTxs +} + +func wrapBlockError(height uint64, err error, msg string) error { + return fmt.Errorf("failed to %s at height %d: %w", msg, height, err) +} + +func wrapStateError(err error, msg string) error { + return fmt.Errorf("failed to %s: %w", msg, err) +} + +// Add these validation helper functions +func (c *ABCIExecutionClient) validateVersions(state types.State, header *types.SignedHeader) error { + if header.Version.App != state.Version.Consensus.App || + header.Version.Block != state.Version.Consensus.Block { + return fmt.Errorf("block version mismatch: got app=%d,block=%d want app=%d,block=%d", + header.Version.App, header.Version.Block, + state.Version.Consensus.App, state.Version.Consensus.Block) + } + return nil +} + +func (c *ABCIExecutionClient) validateHeight(state types.State, header *types.SignedHeader) error { + if state.LastBlockHeight <= 0 { + if header.Height() != state.InitialHeight { + return fmt.Errorf("initial block height mismatch: got %d, want %d", + header.Height(), state.InitialHeight) + } + return nil + } + + expectedHeight := state.LastBlockHeight + 1 + if header.Height() != expectedHeight { + return fmt.Errorf("block height mismatch: got %d, want %d", + header.Height(), expectedHeight) } - return txs + return nil +} + +func (c *ABCIExecutionClient) validateHashes(state types.State, header *types.SignedHeader) error { + if !bytes.Equal(header.AppHash[:], state.AppHash[:]) { + return fmt.Errorf("AppHash mismatch: got %X, want %X", + header.AppHash, state.AppHash) + } + + if !bytes.Equal(header.LastResultsHash[:], state.LastResultsHash[:]) { + return fmt.Errorf("LastResultsHash mismatch: got %X, want %X", + header.LastResultsHash, state.LastResultsHash) + } + return nil } diff --git a/go.mod b/go.mod index ec8a0ab..8c5ceb7 100644 --- a/go.mod +++ b/go.mod @@ -4,7 +4,7 @@ go 1.22.7 toolchain go1.22.8 -replace github.com/rollkit/go-execution => github.com/lastdotnet/go-execution v0.0.0-20241106160445-7810bc1e5d3c +replace github.com/rollkit/go-execution => github.com/lastdotnet/go-execution v0.0.0-20241107213138-d1712b8c4d58 require ( github.com/cometbft/cometbft v0.38.13 @@ -15,7 +15,7 @@ require ( require ( github.com/DataDog/zstd v1.4.5 // indirect github.com/beorn7/perks v1.0.1 // indirect - github.com/celestiaorg/go-header v0.6.2 // indirect + github.com/celestiaorg/go-header v0.6.3 // indirect github.com/cespare/xxhash/v2 v2.3.0 // indirect github.com/cockroachdb/errors v1.11.3 // indirect github.com/cockroachdb/fifo v0.0.0-20240606204812-0bbfbd93a7ce // indirect diff --git a/go.sum b/go.sum index 9096fcc..12a244d 100644 --- a/go.sum +++ b/go.sum @@ -11,8 +11,8 @@ github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= github.com/btcsuite/btcd/btcutil v1.1.6 h1:zFL2+c3Lb9gEgqKNzowKUPQNb8jV7v5Oaodi/AYFd6c= github.com/btcsuite/btcd/btcutil v1.1.6/go.mod h1:9dFymx8HpuLqBnsPELrImQeTQfKBQqzqGbbV3jK55aE= -github.com/celestiaorg/go-header v0.6.2 h1:qgWyJQg+/x6k4QAfN1rPt2HXHZjQOmCqD0ct4dFBIZY= -github.com/celestiaorg/go-header v0.6.2/go.mod h1:Az4S4NxMOJ1eAzOaF8u5AZt5UzsSzg92uqpdXS3yOZE= +github.com/celestiaorg/go-header v0.6.3 h1:VI+fsNxFLeUS7cNn0LgHP6Db66uslnKp/fgMg5nxqHg= +github.com/celestiaorg/go-header v0.6.3/go.mod h1:Az4S4NxMOJ1eAzOaF8u5AZt5UzsSzg92uqpdXS3yOZE= github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= @@ -180,8 +180,8 @@ github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= -github.com/lastdotnet/go-execution v0.0.0-20241106160445-7810bc1e5d3c h1:A8/nKchjQ5waN5PEwVNsTaAdTm2iukmjRlcLcrwunjA= -github.com/lastdotnet/go-execution v0.0.0-20241106160445-7810bc1e5d3c/go.mod h1:WgzV+v1zOka30IQbuE9Rmk7xkfF3xhCmptpit2v5Zo0= +github.com/lastdotnet/go-execution v0.0.0-20241107213138-d1712b8c4d58 h1:sjinTQV5doakYIhNFcDUzcfdmJzdIhKvdZuXo+z2+Jc= +github.com/lastdotnet/go-execution v0.0.0-20241107213138-d1712b8c4d58/go.mod h1:JYUo1RXqdVQjuOFhU8oO6g7pKOk29ZFsdizMXkHwVdA= github.com/libp2p/go-buffer-pool v0.1.0 h1:oK4mSFcQz7cTQIfqbe4MIj9gLW+mnanjyFtc6cdF0Y8= github.com/libp2p/go-buffer-pool v0.1.0/go.mod h1:N+vh8gMqimBzdKkSMVuydVDq+UV5QTWy5HSiZacSbPg= github.com/libp2p/go-flow-metrics v0.1.0 h1:0iPhMI8PskQwzh57jB9WxIuIOQ0r+15PChFGkx3Q3WM= From c716a014ba91db5677082b930b719e26ff4caa77 Mon Sep 17 00:00:00 2001 From: Jay Jie Date: Mon, 11 Nov 2024 18:02:21 -0800 Subject: [PATCH 10/10] chore: add ctx in executor methods --- execution.go | 12 +++++------- go.mod | 2 +- go.sum | 4 ++-- 3 files changed, 8 insertions(+), 10 deletions(-) diff --git a/execution.go b/execution.go index b8844f2..99d51e0 100644 --- a/execution.go +++ b/execution.go @@ -87,6 +87,7 @@ func NewABCIExecutionClient( // InitChain initializes the blockchain with genesis information. func (c *ABCIExecutionClient) InitChain( + ctx context.Context, genesisTime time.Time, initialHeight uint64, chainID string, @@ -150,8 +151,8 @@ func (c *ABCIExecutionClient) initChain(genesis *cmtypes.GenesisDoc) (*abci.Resp } // GetTxs retrieves all available transactions from the mempool. -func (c *ABCIExecutionClient) GetTxs() ([]execution_types.Tx, error) { - state, err := c.store.GetState(context.Background()) +func (c *ABCIExecutionClient) GetTxs(ctx context.Context) ([]execution_types.Tx, error) { + state, err := c.store.GetState(ctx) if err != nil { return nil, fmt.Errorf("failed to get current state: %w", err) } @@ -177,13 +178,12 @@ func (c *ABCIExecutionClient) GetTxs() ([]execution_types.Tx, error) { // ExecuteTxs executes the given transactions and returns the new state root and gas used. func (c *ABCIExecutionClient) ExecuteTxs( + ctx context.Context, txs []execution_types.Tx, blockHeight uint64, timestamp time.Time, prevStateRoot types.Hash, ) (types.Hash, uint64, error) { - ctx := context.Background() - state, err := c.store.GetState(ctx) if err != nil { return types.Hash{}, 0, wrapStateError(err, "get current state") @@ -255,9 +255,7 @@ func (c *ABCIExecutionClient) ExecuteTxs( } // SetFinal marks a block at the given height as final. -func (c *ABCIExecutionClient) SetFinal(blockHeight uint64) error { - ctx := context.Background() - +func (c *ABCIExecutionClient) SetFinal(ctx context.Context, blockHeight uint64) error { header, data, err := c.store.GetBlockData(ctx, blockHeight) if err != nil { return wrapBlockError(blockHeight, err, "get block data") diff --git a/go.mod b/go.mod index 8c5ceb7..314c2a9 100644 --- a/go.mod +++ b/go.mod @@ -4,7 +4,7 @@ go 1.22.7 toolchain go1.22.8 -replace github.com/rollkit/go-execution => github.com/lastdotnet/go-execution v0.0.0-20241107213138-d1712b8c4d58 +replace github.com/rollkit/go-execution => github.com/lastdotnet/go-execution v0.0.0-20241108025553-291f75953069 require ( github.com/cometbft/cometbft v0.38.13 diff --git a/go.sum b/go.sum index 12a244d..3b9df57 100644 --- a/go.sum +++ b/go.sum @@ -180,8 +180,8 @@ github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= -github.com/lastdotnet/go-execution v0.0.0-20241107213138-d1712b8c4d58 h1:sjinTQV5doakYIhNFcDUzcfdmJzdIhKvdZuXo+z2+Jc= -github.com/lastdotnet/go-execution v0.0.0-20241107213138-d1712b8c4d58/go.mod h1:JYUo1RXqdVQjuOFhU8oO6g7pKOk29ZFsdizMXkHwVdA= +github.com/lastdotnet/go-execution v0.0.0-20241108025553-291f75953069 h1:GnaHqG+ixzkaBtWzvaZCpNgNZnsf/liacgK39t55nHE= +github.com/lastdotnet/go-execution v0.0.0-20241108025553-291f75953069/go.mod h1:JYUo1RXqdVQjuOFhU8oO6g7pKOk29ZFsdizMXkHwVdA= github.com/libp2p/go-buffer-pool v0.1.0 h1:oK4mSFcQz7cTQIfqbe4MIj9gLW+mnanjyFtc6cdF0Y8= github.com/libp2p/go-buffer-pool v0.1.0/go.mod h1:N+vh8gMqimBzdKkSMVuydVDq+UV5QTWy5HSiZacSbPg= github.com/libp2p/go-flow-metrics v0.1.0 h1:0iPhMI8PskQwzh57jB9WxIuIOQ0r+15PChFGkx3Q3WM=