mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-19 00:54:12 +00:00
243 lines
5.6 KiB
Go
243 lines
5.6 KiB
Go
package handler
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
"regexp"
|
|
"strconv"
|
|
|
|
"github.com/go-chi/chi/v5"
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon-lib/types/ssz"
|
|
"github.com/ledgerwatch/erigon/cl/beacon/beaconhttp"
|
|
"github.com/ledgerwatch/erigon/cl/clparams"
|
|
)
|
|
|
|
type apiError struct {
|
|
code int
|
|
err error
|
|
}
|
|
|
|
type beaconResponse struct {
|
|
Data any `json:"data,omitempty"`
|
|
Finalized *bool `json:"finalized,omitempty"`
|
|
Version *clparams.StateVersion `json:"version,omitempty"`
|
|
ExecutionOptimistic *bool `json:"execution_optimistic,omitempty"`
|
|
}
|
|
|
|
func (b *beaconResponse) EncodeSSZ(xs []byte) ([]byte, error) {
|
|
marshaler, ok := b.Data.(ssz.Marshaler)
|
|
if !ok {
|
|
return nil, beaconhttp.NewEndpointError(http.StatusBadRequest, "This endpoint does not support SSZ response")
|
|
}
|
|
encoded, err := marshaler.EncodeSSZ(nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return encoded, nil
|
|
}
|
|
|
|
func (b *beaconResponse) EncodingSizeSSZ() int {
|
|
marshaler, ok := b.Data.(ssz.Marshaler)
|
|
if !ok {
|
|
return 9
|
|
}
|
|
return marshaler.EncodingSizeSSZ()
|
|
}
|
|
|
|
func newBeaconResponse(data any) *beaconResponse {
|
|
return &beaconResponse{
|
|
Data: data,
|
|
}
|
|
}
|
|
|
|
func (r *beaconResponse) withFinalized(finalized bool) (out *beaconResponse) {
|
|
out = new(beaconResponse)
|
|
*out = *r
|
|
out.Finalized = new(bool)
|
|
out.ExecutionOptimistic = new(bool)
|
|
out.Finalized = &finalized
|
|
return out
|
|
}
|
|
|
|
func (r *beaconResponse) withOptimistic(optimistic bool) (out *beaconResponse) {
|
|
out = new(beaconResponse)
|
|
*out = *r
|
|
out.ExecutionOptimistic = new(bool)
|
|
out.ExecutionOptimistic = &optimistic
|
|
return out
|
|
}
|
|
|
|
func (r *beaconResponse) withVersion(version clparams.StateVersion) (out *beaconResponse) {
|
|
out = new(beaconResponse)
|
|
*out = *r
|
|
out.Version = new(clparams.StateVersion)
|
|
out.Version = &version
|
|
return out
|
|
}
|
|
|
|
type chainTag int
|
|
|
|
var (
|
|
Head chainTag = 0
|
|
Finalized chainTag = 1
|
|
Justified chainTag = 2
|
|
Genesis chainTag = 3
|
|
)
|
|
|
|
// Represent either state id or block id
|
|
type segmentID struct {
|
|
tag chainTag
|
|
slot *uint64
|
|
root *libcommon.Hash
|
|
}
|
|
|
|
func (c *segmentID) head() bool {
|
|
return c.tag == Head && c.slot == nil && c.root == nil
|
|
}
|
|
|
|
func (c *segmentID) finalized() bool {
|
|
return c.tag == Finalized
|
|
}
|
|
|
|
func (c *segmentID) justified() bool {
|
|
return c.tag == Justified
|
|
}
|
|
|
|
func (c *segmentID) genesis() bool {
|
|
return c.tag == Genesis
|
|
}
|
|
|
|
func (c *segmentID) getSlot() *uint64 {
|
|
return c.slot
|
|
}
|
|
|
|
func (c *segmentID) getRoot() *libcommon.Hash {
|
|
return c.root
|
|
}
|
|
|
|
func epochFromRequest(r *http.Request) (uint64, error) {
|
|
// Must only be a number
|
|
regex := regexp.MustCompile(`^\d+$`)
|
|
epoch := chi.URLParam(r, "epoch")
|
|
if !regex.MatchString(epoch) {
|
|
return 0, fmt.Errorf("invalid path variable: {epoch}")
|
|
}
|
|
epochMaybe, err := strconv.ParseUint(epoch, 10, 64)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return epochMaybe, nil
|
|
}
|
|
|
|
func stringFromRequest(r *http.Request, name string) (string, error) {
|
|
str := chi.URLParam(r, name)
|
|
if str == "" {
|
|
return "", nil
|
|
}
|
|
return str, nil
|
|
}
|
|
|
|
func blockIdFromRequest(r *http.Request) (*segmentID, error) {
|
|
regex := regexp.MustCompile(`^(?:0x[0-9a-fA-F]{64}|head|finalized|genesis|\d+)$`)
|
|
|
|
blockId := chi.URLParam(r, "block_id")
|
|
if !regex.MatchString(blockId) {
|
|
return nil, fmt.Errorf("invalid path variable: {block_id}")
|
|
}
|
|
|
|
if blockId == "head" {
|
|
return &segmentID{tag: Head}, nil
|
|
}
|
|
if blockId == "finalized" {
|
|
return &segmentID{tag: Finalized}, nil
|
|
}
|
|
if blockId == "genesis" {
|
|
return &segmentID{tag: Genesis}, nil
|
|
}
|
|
slotMaybe, err := strconv.ParseUint(blockId, 10, 64)
|
|
if err == nil {
|
|
return &segmentID{slot: &slotMaybe}, nil
|
|
}
|
|
root := libcommon.HexToHash(blockId)
|
|
return &segmentID{
|
|
root: &root,
|
|
}, nil
|
|
}
|
|
|
|
func stateIdFromRequest(r *http.Request) (*segmentID, error) {
|
|
regex := regexp.MustCompile(`^(?:0x[0-9a-fA-F]{64}|head|finalized|genesis|justified|\d+)$`)
|
|
|
|
stateId := chi.URLParam(r, "state_id")
|
|
if !regex.MatchString(stateId) {
|
|
return nil, fmt.Errorf("invalid path variable: {state_id}")
|
|
}
|
|
|
|
if stateId == "head" {
|
|
return &segmentID{tag: Head}, nil
|
|
}
|
|
if stateId == "finalized" {
|
|
return &segmentID{tag: Finalized}, nil
|
|
}
|
|
if stateId == "genesis" {
|
|
return &segmentID{tag: Genesis}, nil
|
|
}
|
|
if stateId == "justified" {
|
|
return &segmentID{tag: Justified}, nil
|
|
}
|
|
slotMaybe, err := strconv.ParseUint(stateId, 10, 64)
|
|
if err == nil {
|
|
return &segmentID{slot: &slotMaybe}, nil
|
|
}
|
|
root := libcommon.HexToHash(stateId)
|
|
return &segmentID{
|
|
root: &root,
|
|
}, nil
|
|
}
|
|
|
|
func hashFromQueryParams(r *http.Request, name string) (*libcommon.Hash, error) {
|
|
hashStr := r.URL.Query().Get(name)
|
|
if hashStr == "" {
|
|
return nil, nil
|
|
}
|
|
// check if hashstr is an hex string
|
|
if len(hashStr) != 2+2*32 {
|
|
return nil, fmt.Errorf("invalid hash length")
|
|
}
|
|
if hashStr[:2] != "0x" {
|
|
return nil, fmt.Errorf("invalid hash prefix")
|
|
}
|
|
notHex, err := regexp.MatchString("[^0-9A-Fa-f]", hashStr[2:])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if notHex {
|
|
return nil, fmt.Errorf("invalid hash characters")
|
|
}
|
|
|
|
hash := libcommon.HexToHash(hashStr)
|
|
return &hash, nil
|
|
}
|
|
|
|
// uint64FromQueryParams retrieves a number from the query params, in base 10.
|
|
func uint64FromQueryParams(r *http.Request, name string) (*uint64, error) {
|
|
str := r.URL.Query().Get(name)
|
|
if str == "" {
|
|
return nil, nil
|
|
}
|
|
num, err := strconv.ParseUint(str, 10, 64)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &num, nil
|
|
}
|
|
|
|
// decode a list of strings from the query params
|
|
func stringListFromQueryParams(r *http.Request, name string) ([]string, error) {
|
|
str := r.URL.Query().Get(name)
|
|
if str == "" {
|
|
return nil, nil
|
|
}
|
|
return regexp.MustCompile(`\s*,\s*`).Split(str, -1), nil
|
|
}
|