package beacon import ( "bytes" "context" "encoding/json" "fmt" "io" "net/http" "testing" "github.com/prysmaticlabs/prysm/v5/api/client" "github.com/prysmaticlabs/prysm/v5/beacon-chain/state" "github.com/prysmaticlabs/prysm/v5/consensus-types/blocks" blocktest "github.com/prysmaticlabs/prysm/v5/consensus-types/blocks/testing" "github.com/prysmaticlabs/prysm/v5/network/forks" ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/v5/testing/util" "github.com/prysmaticlabs/prysm/v5/time/slots" "github.com/prysmaticlabs/prysm/v5/config/params" "github.com/prysmaticlabs/prysm/v5/consensus-types/primitives" "github.com/prysmaticlabs/prysm/v5/encoding/ssz/detect" "github.com/prysmaticlabs/prysm/v5/runtime/version" "github.com/pkg/errors" "github.com/prysmaticlabs/prysm/v5/testing/require" ) type testRT struct { rt func(*http.Request) (*http.Response, error) } func (rt *testRT) RoundTrip(req *http.Request) (*http.Response, error) { if rt.rt != nil { return rt.rt(req) } return nil, errors.New("RoundTripper not implemented") } var _ http.RoundTripper = &testRT{} func marshalToEnvelope(val interface{}) ([]byte, error) { raw, err := json.Marshal(val) if err != nil { return nil, errors.Wrap(err, "error marshaling value to place in data envelope") } env := struct { Data json.RawMessage `json:"data"` }{ Data: raw, } return json.Marshal(env) } func TestMarshalToEnvelope(t *testing.T) { d := struct { Version string `json:"version"` }{ Version: "Prysm/v2.0.5 (linux amd64)", } encoded, err := marshalToEnvelope(d) require.NoError(t, err) expected := `{"data":{"version":"Prysm/v2.0.5 (linux amd64)"}}` require.Equal(t, expected, string(encoded)) } func TestFallbackVersionCheck(t *testing.T) { trans := &testRT{rt: func(req *http.Request) (*http.Response, error) { res := &http.Response{Request: req} switch req.URL.Path { case getNodeVersionPath: res.StatusCode = http.StatusOK b := bytes.NewBuffer(nil) d := struct { Version string `json:"version"` }{ Version: "Prysm/v2.0.5 (linux amd64)", } encoded, err := marshalToEnvelope(d) require.NoError(t, err) b.Write(encoded) res.Body = io.NopCloser(b) case getWeakSubjectivityPath: res.StatusCode = http.StatusNotFound } return res, nil }} c, err := NewClient("http://localhost:3500", client.WithRoundTripper(trans)) require.NoError(t, err) ctx := context.Background() _, err = ComputeWeakSubjectivityCheckpoint(ctx, c) require.ErrorIs(t, err, errUnsupportedPrysmCheckpointVersion) } func TestFname(t *testing.T) { vu := &detect.VersionedUnmarshaler{ Config: params.MainnetConfig(), Fork: version.Phase0, } slot := primitives.Slot(23) prefix := "block" var root [32]byte copy(root[:], []byte{0x23, 0x23, 0x23}) expected := "block_mainnet_phase0_23-0x2323230000000000000000000000000000000000000000000000000000000000.ssz" actual := fname(prefix, vu, slot, root) require.Equal(t, expected, actual) vu.Config = params.MinimalSpecConfig() vu.Fork = version.Altair slot = 17 prefix = "state" copy(root[29:], []byte{0x17, 0x17, 0x17}) expected = "state_minimal_altair_17-0x2323230000000000000000000000000000000000000000000000000000171717.ssz" actual = fname(prefix, vu, slot, root) require.Equal(t, expected, actual) } func TestDownloadWeakSubjectivityCheckpoint(t *testing.T) { ctx := context.Background() cfg := params.MainnetConfig().Copy() epoch := cfg.AltairForkEpoch - 1 // set up checkpoint state, using the epoch that will be computed as the ws checkpoint state based on the head state wSlot, err := slots.EpochStart(epoch) require.NoError(t, err) wst, err := util.NewBeaconState() require.NoError(t, err) fork, err := forkForEpoch(cfg, epoch) require.NoError(t, err) require.NoError(t, wst.SetFork(fork)) // set up checkpoint block b, err := blocks.NewSignedBeaconBlock(util.NewBeaconBlock()) require.NoError(t, err) b, err = blocktest.SetBlockParentRoot(b, cfg.ZeroHash) require.NoError(t, err) b, err = blocktest.SetBlockSlot(b, wSlot) require.NoError(t, err) b, err = blocktest.SetProposerIndex(b, 0) require.NoError(t, err) // set up state header pointing at checkpoint block - this is how the block is downloaded by root header, err := b.Header() require.NoError(t, err) require.NoError(t, wst.SetLatestBlockHeader(header.Header)) // order of operations can be confusing here: // - when computing the state root, make sure block header is complete, EXCEPT the state root should be zero-value // - before computing the block root (to match the request route), the block should include the state root // *computed from the state with a header that does not have a state root set yet* wRoot, err := wst.HashTreeRoot(ctx) require.NoError(t, err) b, err = blocktest.SetBlockStateRoot(b, wRoot) require.NoError(t, err) serBlock, err := b.MarshalSSZ() require.NoError(t, err) bRoot, err := b.Block().HashTreeRoot() require.NoError(t, err) wsSerialized, err := wst.MarshalSSZ() require.NoError(t, err) expectedWSD := WeakSubjectivityData{ BlockRoot: bRoot, StateRoot: wRoot, Epoch: epoch, } trans := &testRT{rt: func(req *http.Request) (*http.Response, error) { res := &http.Response{Request: req} switch req.URL.Path { case getWeakSubjectivityPath: res.StatusCode = http.StatusOK cp := struct { Epoch string `json:"epoch"` Root string `json:"root"` }{ Epoch: fmt.Sprintf("%d", slots.ToEpoch(b.Block().Slot())), Root: fmt.Sprintf("%#x", bRoot), } wsr := struct { Checkpoint interface{} `json:"ws_checkpoint"` StateRoot string `json:"state_root"` }{ Checkpoint: cp, StateRoot: fmt.Sprintf("%#x", wRoot), } rb, err := marshalToEnvelope(wsr) require.NoError(t, err) res.Body = io.NopCloser(bytes.NewBuffer(rb)) case renderGetStatePath(IdFromSlot(wSlot)): res.StatusCode = http.StatusOK res.Body = io.NopCloser(bytes.NewBuffer(wsSerialized)) case renderGetBlockPath(IdFromRoot(bRoot)): res.StatusCode = http.StatusOK res.Body = io.NopCloser(bytes.NewBuffer(serBlock)) } return res, nil }} c, err := NewClient("http://localhost:3500", client.WithRoundTripper(trans)) require.NoError(t, err) wsd, err := ComputeWeakSubjectivityCheckpoint(ctx, c) require.NoError(t, err) require.Equal(t, expectedWSD.Epoch, wsd.Epoch) require.Equal(t, expectedWSD.StateRoot, wsd.StateRoot) require.Equal(t, expectedWSD.BlockRoot, wsd.BlockRoot) } // runs computeBackwardsCompatible directly // and via ComputeWeakSubjectivityCheckpoint with a round tripper that triggers the backwards compatible code path func TestDownloadBackwardsCompatibleCombined(t *testing.T) { ctx := context.Background() cfg := params.MainnetConfig() st, expectedEpoch := defaultTestHeadState(t, cfg) serialized, err := st.MarshalSSZ() require.NoError(t, err) // set up checkpoint state, using the epoch that will be computed as the ws checkpoint state based on the head state wSlot, err := slots.EpochStart(expectedEpoch) require.NoError(t, err) wst, err := util.NewBeaconState() require.NoError(t, err) fork, err := forkForEpoch(cfg, cfg.GenesisEpoch) require.NoError(t, err) require.NoError(t, wst.SetFork(fork)) // set up checkpoint block b, err := blocks.NewSignedBeaconBlock(util.NewBeaconBlock()) require.NoError(t, err) b, err = blocktest.SetBlockParentRoot(b, cfg.ZeroHash) require.NoError(t, err) b, err = blocktest.SetBlockSlot(b, wSlot) require.NoError(t, err) b, err = blocktest.SetProposerIndex(b, 0) require.NoError(t, err) // set up state header pointing at checkpoint block - this is how the block is downloaded by root header, err := b.Header() require.NoError(t, err) require.NoError(t, wst.SetLatestBlockHeader(header.Header)) // order of operations can be confusing here: // - when computing the state root, make sure block header is complete, EXCEPT the state root should be zero-value // - before computing the block root (to match the request route), the block should include the state root // *computed from the state with a header that does not have a state root set yet* wRoot, err := wst.HashTreeRoot(ctx) require.NoError(t, err) b, err = blocktest.SetBlockStateRoot(b, wRoot) require.NoError(t, err) serBlock, err := b.MarshalSSZ() require.NoError(t, err) bRoot, err := b.Block().HashTreeRoot() require.NoError(t, err) wsSerialized, err := wst.MarshalSSZ() require.NoError(t, err) trans := &testRT{rt: func(req *http.Request) (*http.Response, error) { res := &http.Response{Request: req} switch req.URL.Path { case getNodeVersionPath: res.StatusCode = http.StatusOK b := bytes.NewBuffer(nil) d := struct { Version string `json:"version"` }{ Version: "Lighthouse/v0.1.5 (Linux x86_64)", } encoded, err := marshalToEnvelope(d) require.NoError(t, err) b.Write(encoded) res.Body = io.NopCloser(b) case getWeakSubjectivityPath: res.StatusCode = http.StatusNotFound case renderGetStatePath(IdHead): res.StatusCode = http.StatusOK res.Body = io.NopCloser(bytes.NewBuffer(serialized)) case renderGetStatePath(IdFromSlot(wSlot)): res.StatusCode = http.StatusOK res.Body = io.NopCloser(bytes.NewBuffer(wsSerialized)) case renderGetBlockPath(IdFromRoot(bRoot)): res.StatusCode = http.StatusOK res.Body = io.NopCloser(bytes.NewBuffer(serBlock)) } return res, nil }} c, err := NewClient("http://localhost:3500", client.WithRoundTripper(trans)) require.NoError(t, err) wsPub, err := ComputeWeakSubjectivityCheckpoint(ctx, c) require.NoError(t, err) wsPriv, err := computeBackwardsCompatible(ctx, c) require.NoError(t, err) require.DeepEqual(t, wsPriv, wsPub) } func TestGetWeakSubjectivityEpochFromHead(t *testing.T) { st, expectedEpoch := defaultTestHeadState(t, params.MainnetConfig()) serialized, err := st.MarshalSSZ() require.NoError(t, err) trans := &testRT{rt: func(req *http.Request) (*http.Response, error) { res := &http.Response{Request: req} if req.URL.Path == renderGetStatePath(IdHead) { res.StatusCode = http.StatusOK res.Body = io.NopCloser(bytes.NewBuffer(serialized)) } return res, nil }} c, err := NewClient("http://localhost:3500", client.WithRoundTripper(trans)) require.NoError(t, err) actualEpoch, err := getWeakSubjectivityEpochFromHead(context.Background(), c) require.NoError(t, err) require.Equal(t, expectedEpoch, actualEpoch) } func forkForEpoch(cfg *params.BeaconChainConfig, epoch primitives.Epoch) (*ethpb.Fork, error) { os := forks.NewOrderedSchedule(cfg) currentVersion, err := os.VersionForEpoch(epoch) if err != nil { return nil, err } prevVersion, err := os.Previous(currentVersion) if err != nil { if !errors.Is(err, forks.ErrNoPreviousVersion) { return nil, err } // use same version for both in the case of genesis prevVersion = currentVersion } forkEpoch := cfg.ForkVersionSchedule[currentVersion] return ðpb.Fork{ PreviousVersion: prevVersion[:], CurrentVersion: currentVersion[:], Epoch: forkEpoch, }, nil } func defaultTestHeadState(t *testing.T, cfg *params.BeaconChainConfig) (state.BeaconState, primitives.Epoch) { st, err := util.NewBeaconStateAltair() require.NoError(t, err) fork, err := forkForEpoch(cfg, cfg.AltairForkEpoch) require.NoError(t, err) require.NoError(t, st.SetFork(fork)) slot, err := slots.EpochStart(cfg.AltairForkEpoch) require.NoError(t, err) require.NoError(t, st.SetSlot(slot)) var validatorCount, avgBalance uint64 = 100, 35 require.NoError(t, populateValidators(cfg, st, validatorCount, avgBalance)) require.NoError(t, st.SetFinalizedCheckpoint(ðpb.Checkpoint{ Epoch: fork.Epoch - 10, Root: make([]byte, 32), })) // to see the math for this, look at helpers.LatestWeakSubjectivityEpoch // and for the values use mainnet config values, the validatorCount and avgBalance above, and altair fork epoch expectedEpoch := slots.ToEpoch(st.Slot()) - 224 return st, expectedEpoch } // TODO(10429): refactor beacon state options in testing/util to take a state.BeaconState so this can become an option func populateValidators(cfg *params.BeaconChainConfig, st state.BeaconState, valCount, avgBalance uint64) error { validators := make([]*ethpb.Validator, valCount) balances := make([]uint64, len(validators)) for i := uint64(0); i < valCount; i++ { validators[i] = ðpb.Validator{ PublicKey: make([]byte, cfg.BLSPubkeyLength), WithdrawalCredentials: make([]byte, 32), EffectiveBalance: avgBalance * 1e9, ExitEpoch: cfg.FarFutureEpoch, } balances[i] = validators[i].EffectiveBalance } if err := st.SetValidators(validators); err != nil { return err } return st.SetBalances(balances) } func TestDownloadFinalizedData(t *testing.T) { ctx := context.Background() cfg := params.MainnetConfig().Copy() // avoid the altair zone because genesis tests are easier to set up epoch := cfg.AltairForkEpoch - 1 // set up checkpoint state, using the epoch that will be computed as the ws checkpoint state based on the head state slot, err := slots.EpochStart(epoch) require.NoError(t, err) st, err := util.NewBeaconState() require.NoError(t, err) fork, err := forkForEpoch(cfg, epoch) require.NoError(t, err) require.NoError(t, st.SetFork(fork)) require.NoError(t, st.SetSlot(slot)) // set up checkpoint block b, err := blocks.NewSignedBeaconBlock(util.NewBeaconBlock()) require.NoError(t, err) b, err = blocktest.SetBlockParentRoot(b, cfg.ZeroHash) require.NoError(t, err) b, err = blocktest.SetBlockSlot(b, slot) require.NoError(t, err) b, err = blocktest.SetProposerIndex(b, 0) require.NoError(t, err) // set up state header pointing at checkpoint block - this is how the block is downloaded by root header, err := b.Header() require.NoError(t, err) require.NoError(t, st.SetLatestBlockHeader(header.Header)) // order of operations can be confusing here: // - when computing the state root, make sure block header is complete, EXCEPT the state root should be zero-value // - before computing the block root (to match the request route), the block should include the state root // *computed from the state with a header that does not have a state root set yet* sr, err := st.HashTreeRoot(ctx) require.NoError(t, err) b, err = blocktest.SetBlockStateRoot(b, sr) require.NoError(t, err) mb, err := b.MarshalSSZ() require.NoError(t, err) br, err := b.Block().HashTreeRoot() require.NoError(t, err) ms, err := st.MarshalSSZ() require.NoError(t, err) trans := &testRT{rt: func(req *http.Request) (*http.Response, error) { res := &http.Response{Request: req} switch req.URL.Path { case renderGetStatePath(IdFinalized): res.StatusCode = http.StatusOK res.Body = io.NopCloser(bytes.NewBuffer(ms)) case renderGetBlockPath(IdFromSlot(b.Block().Slot())): res.StatusCode = http.StatusOK res.Body = io.NopCloser(bytes.NewBuffer(mb)) default: res.StatusCode = http.StatusInternalServerError res.Body = io.NopCloser(bytes.NewBufferString("")) } return res, nil }} c, err := NewClient("http://localhost:3500", client.WithRoundTripper(trans)) require.NoError(t, err) // sanity check before we go through checkpoint // make sure we can download the state and unmarshal it with the VersionedUnmarshaler sb, err := c.GetState(ctx, IdFinalized) require.NoError(t, err) require.Equal(t, true, bytes.Equal(sb, ms)) vu, err := detect.FromState(sb) require.NoError(t, err) us, err := vu.UnmarshalBeaconState(sb) require.NoError(t, err) ushtr, err := us.HashTreeRoot(ctx) require.NoError(t, err) require.Equal(t, sr, ushtr) expected := &OriginData{ sb: ms, bb: mb, br: br, sr: sr, } od, err := DownloadFinalizedData(ctx, c) require.NoError(t, err) require.Equal(t, true, bytes.Equal(expected.sb, od.sb)) require.Equal(t, true, bytes.Equal(expected.bb, od.bb)) require.Equal(t, expected.br, od.br) require.Equal(t, expected.sr, od.sr) }