prysm-pulse/testing/endtoend/evaluators/api_middleware.go
Radosław Kapka 79445d2bf6
HTTP Beacon API: /eth/v1/validator/duties (#12810)
* impl

* protos

* remove apimiddleware from e2e

* register endpoint

* get epoch from vars

* tests

* test fixes

* remove unused function

* GetProposerDuties

* proto

* fix compilation

* register

* tests

* GetSyncCommitteeDuties

* protos

* register

* tests in progress

* complete tests

* check altair epoch

* more testing

* create variable

---------

Co-authored-by: james-prysm <90280386+james-prysm@users.noreply.github.com>
2023-09-12 13:57:04 +00:00

204 lines
6.5 KiB
Go

package evaluators
import (
"context"
"encoding/json"
"fmt"
"net/http"
"strconv"
"strings"
"github.com/ethereum/go-ethereum/common/hexutil"
"github.com/pkg/errors"
"github.com/prysmaticlabs/prysm/v4/proto/eth/service"
ethpbv1 "github.com/prysmaticlabs/prysm/v4/proto/eth/v1"
ethpbv2 "github.com/prysmaticlabs/prysm/v4/proto/eth/v2"
"github.com/prysmaticlabs/prysm/v4/testing/endtoend/helpers"
"github.com/prysmaticlabs/prysm/v4/testing/endtoend/params"
"github.com/prysmaticlabs/prysm/v4/testing/endtoend/policies"
e2etypes "github.com/prysmaticlabs/prysm/v4/testing/endtoend/types"
"google.golang.org/grpc"
)
type validatorJson struct {
PublicKey string `json:"pubkey"`
WithdrawalCredentials string `json:"withdrawal_credentials"`
EffectiveBalance string `json:"effective_balance"`
Slashed bool `json:"slashed"`
ActivationEligibilityEpoch string `json:"activation_eligibility_epoch"`
ActivationEpoch string `json:"activation_epoch"`
ExitEpoch string `json:"exit_epoch"`
WithdrawableEpoch string `json:"withdrawable_epoch"`
}
type validatorContainerJson struct {
Index string `json:"index"`
Balance string `json:"balance"`
Status string `json:"status"`
Validator *validatorJson `json:"validator"`
}
// APIMiddlewareVerifyIntegrity tests our API Middleware for the official Ethereum API.
// This ensures our API Middleware returns good data compared to gRPC.
var APIMiddlewareVerifyIntegrity = e2etypes.Evaluator{
Name: "api_middleware_verify_integrity_epoch_%d",
Policy: policies.OnEpoch(helpers.AltairE2EForkEpoch),
Evaluation: apiMiddlewareVerify,
}
const (
v1MiddlewarePathTemplate = "http://localhost:%d/eth/v1"
)
func apiMiddlewareVerify(_ *e2etypes.EvaluationContext, conns ...*grpc.ClientConn) error {
for beaconNodeIdx, conn := range conns {
if err := runAPIComparisonFunctions(
beaconNodeIdx,
conn,
withCompareValidatorsEth,
withCompareSyncCommittee,
); err != nil {
return err
}
}
return nil
}
func withCompareValidatorsEth(beaconNodeIdx int, conn *grpc.ClientConn) error {
type stateValidatorsResponseJson struct {
Data []*validatorContainerJson `json:"data"`
}
ctx := context.Background()
beaconClient := service.NewBeaconChainClient(conn)
resp, err := beaconClient.ListValidators(ctx, &ethpbv1.StateValidatorsRequest{
StateId: []byte("head"),
Status: []ethpbv1.ValidatorStatus{ethpbv1.ValidatorStatus_EXITED},
})
if err != nil {
return err
}
respJSON := &stateValidatorsResponseJson{}
if err := doMiddlewareJSONGetRequestV1(
"/beacon/states/head/validators?status=exited",
beaconNodeIdx,
respJSON,
); err != nil {
return err
}
if len(respJSON.Data) != len(resp.Data) {
return fmt.Errorf(
"API Middleware number of validators %d does not match gRPC %d",
len(respJSON.Data),
len(resp.Data),
)
}
resp, err = beaconClient.ListValidators(ctx, &ethpbv1.StateValidatorsRequest{
StateId: []byte("head"),
Id: [][]byte{[]byte("100"), []byte("200")},
})
if err != nil {
return err
}
if err := doMiddlewareJSONGetRequestV1(
"/beacon/states/head/validators?id=100&id=200",
beaconNodeIdx,
respJSON,
); err != nil {
return err
}
if len(respJSON.Data) != len(resp.Data) {
return fmt.Errorf(
"API Middleware number of validators %d does not match gRPC %d",
len(respJSON.Data),
len(resp.Data),
)
}
if err = assertValidator(respJSON.Data[0], resp.Data[0]); err != nil {
return errors.Wrapf(err, "incorrect validator data returned from the API request")
}
if err = assertValidator(respJSON.Data[1], resp.Data[1]); err != nil {
return errors.Wrapf(err, "incorrect validator data returned from the API request")
}
return nil
}
func assertValidator(jsonVal *validatorContainerJson, val *ethpbv1.ValidatorContainer) error {
if jsonVal == nil {
return errors.New("validator is nil")
}
if jsonVal.Index != "100" && jsonVal.Index != "200" {
return fmt.Errorf("unexpected validator index '%s'", jsonVal.Index)
}
if jsonVal.Balance != strconv.FormatUint(val.Balance, 10) {
return buildFieldError("Balance", strconv.FormatUint(val.Balance, 10), jsonVal.Balance)
}
if jsonVal.Status != strings.ToLower(val.Status.String()) {
return buildFieldError("Status", strings.ToLower(val.Status.String()), jsonVal.Status)
}
if jsonVal.Validator == nil {
return errors.New("validator is nil")
}
if jsonVal.Validator.PublicKey != hexutil.Encode(val.Validator.Pubkey) {
return buildFieldError("PublicKey", hexutil.Encode(val.Validator.Pubkey), jsonVal.Validator.PublicKey)
}
if jsonVal.Validator.Slashed != val.Validator.Slashed {
return buildFieldError("Slashed", strconv.FormatBool(val.Validator.Slashed), strconv.FormatBool(jsonVal.Validator.Slashed))
}
return nil
}
func withCompareSyncCommittee(beaconNodeIdx int, conn *grpc.ClientConn) error {
type syncCommitteeValidatorsJson struct {
Validators []string `json:"validators"`
ValidatorAggregates [][]string `json:"validator_aggregates"`
}
type syncCommitteesResponseJson struct {
Data *syncCommitteeValidatorsJson `json:"data"`
}
ctx := context.Background()
beaconClient := service.NewBeaconChainClient(conn)
resp, err := beaconClient.ListSyncCommittees(ctx, &ethpbv2.StateSyncCommitteesRequest{
StateId: []byte("head"),
})
if err != nil {
return err
}
respJSON := &syncCommitteesResponseJson{}
if err := doMiddlewareJSONGetRequestV1(
"/beacon/states/head/sync_committees",
beaconNodeIdx,
respJSON,
); err != nil {
return err
}
if len(respJSON.Data.Validators) != len(resp.Data.Validators) {
return fmt.Errorf(
"API Middleware number of validators %d does not match gRPC %d",
len(respJSON.Data.Validators),
len(resp.Data.Validators),
)
}
if len(respJSON.Data.ValidatorAggregates) != len(resp.Data.ValidatorAggregates) {
return fmt.Errorf(
"API Middleware number of validator aggregates %d does not match gRPC %d",
len(respJSON.Data.ValidatorAggregates),
len(resp.Data.ValidatorAggregates),
)
}
return nil
}
func doMiddlewareJSONGetRequestV1(requestPath string, beaconNodeIdx int, dst interface{}) error {
basePath := fmt.Sprintf(v1MiddlewarePathTemplate, params.TestParams.Ports.PrysmBeaconNodeGatewayPort+beaconNodeIdx)
httpResp, err := http.Get(
basePath + requestPath,
)
if err != nil {
return err
}
return json.NewDecoder(httpResp.Body).Decode(&dst)
}
func buildFieldError(field, expected, actual string) error {
return fmt.Errorf("value of '%s' was expected to be '%s' but was '%s'", field, expected, actual)
}