mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-19 00:54:12 +00:00
148 lines
5.1 KiB
Go
148 lines
5.1 KiB
Go
package handler
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
"strconv"
|
|
|
|
"github.com/ledgerwatch/erigon/cl/beacon/beaconhttp"
|
|
"github.com/ledgerwatch/erigon/cl/persistence/beacon_indicies"
|
|
state_accessors "github.com/ledgerwatch/erigon/cl/persistence/state"
|
|
"github.com/ledgerwatch/erigon/cl/phase1/core/state"
|
|
)
|
|
|
|
type committeeResponse struct {
|
|
Index uint64 `json:"index,string"`
|
|
Slot uint64 `json:"slot,string"`
|
|
Validators []string `json:"validators"` // do string directly but it is still a base10 number
|
|
}
|
|
|
|
func (a *ApiHandler) getCommittees(w http.ResponseWriter, r *http.Request) (*beaconResponse, error) {
|
|
ctx := r.Context()
|
|
|
|
epochReq, err := uint64FromQueryParams(r, "epoch")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
index, err := uint64FromQueryParams(r, "index")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
slotFilter, err := uint64FromQueryParams(r, "slot")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
tx, err := a.indiciesDB.BeginRo(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer tx.Rollback()
|
|
blockId, err := stateIdFromRequest(r)
|
|
if err != nil {
|
|
return nil, beaconhttp.NewEndpointError(http.StatusBadRequest, err.Error())
|
|
}
|
|
|
|
blockRoot, httpStatus, err := a.blockRootFromStateId(ctx, tx, blockId)
|
|
if err != nil {
|
|
return nil, beaconhttp.NewEndpointError(httpStatus, err.Error())
|
|
}
|
|
|
|
slotPtr, err := beacon_indicies.ReadBlockSlotByBlockRoot(tx, blockRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if slotPtr == nil {
|
|
return nil, beaconhttp.NewEndpointError(http.StatusNotFound, fmt.Sprintf("could not read block slot: %x", blockRoot))
|
|
}
|
|
slot := *slotPtr
|
|
epoch := slot / a.beaconChainCfg.SlotsPerEpoch
|
|
if epochReq != nil {
|
|
epoch = *epochReq
|
|
}
|
|
// check if the filter (if any) is in the epoch
|
|
if slotFilter != nil && !(epoch*a.beaconChainCfg.SlotsPerEpoch <= *slotFilter && *slotFilter < (epoch+1)*a.beaconChainCfg.SlotsPerEpoch) {
|
|
return nil, beaconhttp.NewEndpointError(http.StatusBadRequest, fmt.Sprintf("slot %d is not in epoch %d", *slotFilter, epoch))
|
|
}
|
|
resp := make([]*committeeResponse, 0, a.beaconChainCfg.SlotsPerEpoch*a.beaconChainCfg.MaxCommitteesPerSlot)
|
|
isFinalized := slot <= a.forkchoiceStore.FinalizedSlot()
|
|
if a.forkchoiceStore.LowestAvaiableSlot() <= slot {
|
|
// non-finality case
|
|
s, cn := a.syncedData.HeadState()
|
|
defer cn()
|
|
if s == nil {
|
|
return nil, beaconhttp.NewEndpointError(http.StatusServiceUnavailable, "node is syncing")
|
|
}
|
|
if epoch > state.Epoch(s)+1 {
|
|
return nil, beaconhttp.NewEndpointError(http.StatusBadRequest, fmt.Sprintf("epoch %d is too far in the future", epoch))
|
|
}
|
|
// get active validator indicies
|
|
committeeCount := s.CommitteeCount(epoch)
|
|
// now start obtaining the committees from the head state
|
|
for currSlot := epoch * a.beaconChainCfg.SlotsPerEpoch; currSlot < (epoch+1)*a.beaconChainCfg.SlotsPerEpoch; currSlot++ {
|
|
if slotFilter != nil && currSlot != *slotFilter {
|
|
continue
|
|
}
|
|
for committeeIndex := uint64(0); committeeIndex < committeeCount; committeeIndex++ {
|
|
if index != nil && committeeIndex != *index {
|
|
continue
|
|
}
|
|
data := &committeeResponse{Index: committeeIndex, Slot: currSlot}
|
|
idxs, err := s.GetBeaconCommitee(currSlot, committeeIndex)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, idx := range idxs {
|
|
data.Validators = append(data.Validators, strconv.FormatUint(idx, 10))
|
|
}
|
|
resp = append(resp, data)
|
|
}
|
|
}
|
|
return newBeaconResponse(resp).withFinalized(isFinalized), nil
|
|
}
|
|
// finality case
|
|
activeIdxs, err := state_accessors.ReadActiveIndicies(tx, epoch*a.beaconChainCfg.SlotsPerEpoch)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
committeesPerSlot := uint64(len(activeIdxs)) / a.beaconChainCfg.SlotsPerEpoch / a.beaconChainCfg.TargetCommitteeSize
|
|
if a.beaconChainCfg.MaxCommitteesPerSlot < committeesPerSlot {
|
|
committeesPerSlot = a.beaconChainCfg.MaxCommitteesPerSlot
|
|
}
|
|
if committeesPerSlot < 1 {
|
|
committeesPerSlot = 1
|
|
}
|
|
|
|
mixPosition := (epoch + a.beaconChainCfg.EpochsPerHistoricalVector - a.beaconChainCfg.MinSeedLookahead - 1) % a.beaconChainCfg.EpochsPerHistoricalVector
|
|
mix, err := a.stateReader.ReadRandaoMixBySlotAndIndex(tx, epoch*a.beaconChainCfg.SlotsPerEpoch, mixPosition)
|
|
if err != nil {
|
|
return nil, beaconhttp.NewEndpointError(http.StatusNotFound, fmt.Sprintf("could not read randao mix: %v", err))
|
|
}
|
|
|
|
for currSlot := epoch * a.beaconChainCfg.SlotsPerEpoch; currSlot < (epoch+1)*a.beaconChainCfg.SlotsPerEpoch; currSlot++ {
|
|
if slotFilter != nil && currSlot != *slotFilter {
|
|
continue
|
|
}
|
|
for committeeIndex := uint64(0); committeeIndex < committeesPerSlot; committeeIndex++ {
|
|
if index != nil && committeeIndex != *index {
|
|
continue
|
|
}
|
|
data := &committeeResponse{Index: committeeIndex, Slot: currSlot}
|
|
index := (currSlot%a.beaconChainCfg.SlotsPerEpoch)*committeesPerSlot + committeeIndex
|
|
committeeCount := committeesPerSlot * a.beaconChainCfg.SlotsPerEpoch
|
|
idxs, err := a.stateReader.ComputeCommittee(mix, activeIdxs, currSlot, committeeCount, index)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, idx := range idxs {
|
|
data.Validators = append(data.Validators, strconv.FormatUint(idx, 10))
|
|
}
|
|
resp = append(resp, data)
|
|
}
|
|
}
|
|
return newBeaconResponse(resp).withFinalized(isFinalized), nil
|
|
}
|