package altair import ( "context" "github.com/prysmaticlabs/prysm/v5/beacon-chain/core/helpers" "github.com/prysmaticlabs/prysm/v5/beacon-chain/core/time" "github.com/prysmaticlabs/prysm/v5/beacon-chain/state" state_native "github.com/prysmaticlabs/prysm/v5/beacon-chain/state/state-native" "github.com/prysmaticlabs/prysm/v5/config/params" ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1/attestation" ) // UpgradeToAltair updates input state to return the version Altair state. // // Spec code: // def upgrade_to_altair(pre: phase0.BeaconState) -> BeaconState: // // epoch = phase0.get_current_epoch(pre) // post = BeaconState( // # Versioning // genesis_time=pre.genesis_time, // genesis_validators_root=pre.genesis_validators_root, // slot=pre.slot, // fork=Fork( // previous_version=pre.fork.current_version, // current_version=ALTAIR_FORK_VERSION, // epoch=epoch, // ), // # History // latest_block_header=pre.latest_block_header, // block_roots=pre.block_roots, // state_roots=pre.state_roots, // historical_roots=pre.historical_roots, // # Eth1 // eth1_data=pre.eth1_data, // eth1_data_votes=pre.eth1_data_votes, // eth1_deposit_index=pre.eth1_deposit_index, // # Registry // validators=pre.validators, // balances=pre.balances, // # Randomness // randao_mixes=pre.randao_mixes, // # Slashings // slashings=pre.slashings, // # Participation // previous_epoch_participation=[ParticipationFlags(0b0000_0000) for _ in range(len(pre.validators))], // current_epoch_participation=[ParticipationFlags(0b0000_0000) for _ in range(len(pre.validators))], // # Finality // justification_bits=pre.justification_bits, // previous_justified_checkpoint=pre.previous_justified_checkpoint, // current_justified_checkpoint=pre.current_justified_checkpoint, // finalized_checkpoint=pre.finalized_checkpoint, // # Inactivity // inactivity_scores=[uint64(0) for _ in range(len(pre.validators))], // ) // # Fill in previous epoch participation from the pre state's pending attestations // translate_participation(post, pre.previous_epoch_attestations) // // # Fill in sync committees // # Note: A duplicate committee is assigned for the current and next committee at the fork boundary // post.current_sync_committee = get_next_sync_committee(post) // post.next_sync_committee = get_next_sync_committee(post) // return post func UpgradeToAltair(ctx context.Context, state state.BeaconState) (state.BeaconState, error) { epoch := time.CurrentEpoch(state) numValidators := state.NumValidators() hrs, err := state.HistoricalRoots() if err != nil { return nil, err } s := ðpb.BeaconStateAltair{ GenesisTime: state.GenesisTime(), GenesisValidatorsRoot: state.GenesisValidatorsRoot(), Slot: state.Slot(), Fork: ðpb.Fork{ PreviousVersion: state.Fork().CurrentVersion, CurrentVersion: params.BeaconConfig().AltairForkVersion, Epoch: epoch, }, LatestBlockHeader: state.LatestBlockHeader(), BlockRoots: state.BlockRoots(), StateRoots: state.StateRoots(), HistoricalRoots: hrs, Eth1Data: state.Eth1Data(), Eth1DataVotes: state.Eth1DataVotes(), Eth1DepositIndex: state.Eth1DepositIndex(), Validators: state.Validators(), Balances: state.Balances(), RandaoMixes: state.RandaoMixes(), Slashings: state.Slashings(), PreviousEpochParticipation: make([]byte, numValidators), CurrentEpochParticipation: make([]byte, numValidators), JustificationBits: state.JustificationBits(), PreviousJustifiedCheckpoint: state.PreviousJustifiedCheckpoint(), CurrentJustifiedCheckpoint: state.CurrentJustifiedCheckpoint(), FinalizedCheckpoint: state.FinalizedCheckpoint(), InactivityScores: make([]uint64, numValidators), } newState, err := state_native.InitializeFromProtoUnsafeAltair(s) if err != nil { return nil, err } prevEpochAtts, err := state.PreviousEpochAttestations() if err != nil { return nil, err } newState, err = TranslateParticipation(ctx, newState, prevEpochAtts) if err != nil { return nil, err } committee, err := NextSyncCommittee(ctx, newState) if err != nil { return nil, err } if err := newState.SetCurrentSyncCommittee(committee); err != nil { return nil, err } if err := newState.SetNextSyncCommittee(committee); err != nil { return nil, err } return newState, nil } // TranslateParticipation translates pending attestations into participation bits, then inserts the bits into beacon state. // This is helper function to convert phase 0 beacon state(pending_attestations) to Altair beacon state(participation_bits). // // Spec code: // def translate_participation(state: BeaconState, pending_attestations: Sequence[phase0.PendingAttestation]) -> None: // // for attestation in pending_attestations: // data = attestation.data // inclusion_delay = attestation.inclusion_delay // # Translate attestation inclusion info to flag indices // participation_flag_indices = get_attestation_participation_flag_indices(state, data, inclusion_delay) // // # Apply flags to all attesting validators // epoch_participation = state.previous_epoch_participation // for index in get_attesting_indices(state, data, attestation.aggregation_bits): // for flag_index in participation_flag_indices: // epoch_participation[index] = add_flag(epoch_participation[index], flag_index) func TranslateParticipation(ctx context.Context, state state.BeaconState, atts []*ethpb.PendingAttestation) (state.BeaconState, error) { epochParticipation, err := state.PreviousEpochParticipation() if err != nil { return nil, err } for _, att := range atts { participatedFlags, err := AttestationParticipationFlagIndices(state, att.Data, att.InclusionDelay) if err != nil { return nil, err } committee, err := helpers.BeaconCommitteeFromState(ctx, state, att.Data.Slot, att.Data.CommitteeIndex) if err != nil { return nil, err } indices, err := attestation.AttestingIndices(att.AggregationBits, committee) if err != nil { return nil, err } cfg := params.BeaconConfig() sourceFlagIndex := cfg.TimelySourceFlagIndex targetFlagIndex := cfg.TimelyTargetFlagIndex headFlagIndex := cfg.TimelyHeadFlagIndex for _, index := range indices { has, err := HasValidatorFlag(epochParticipation[index], sourceFlagIndex) if err != nil { return nil, err } if participatedFlags[sourceFlagIndex] && !has { epochParticipation[index], err = AddValidatorFlag(epochParticipation[index], sourceFlagIndex) if err != nil { return nil, err } } has, err = HasValidatorFlag(epochParticipation[index], targetFlagIndex) if err != nil { return nil, err } if participatedFlags[targetFlagIndex] && !has { epochParticipation[index], err = AddValidatorFlag(epochParticipation[index], targetFlagIndex) if err != nil { return nil, err } } has, err = HasValidatorFlag(epochParticipation[index], headFlagIndex) if err != nil { return nil, err } if participatedFlags[headFlagIndex] && !has { epochParticipation[index], err = AddValidatorFlag(epochParticipation[index], headFlagIndex) if err != nil { return nil, err } } } } if err := state.SetPreviousParticipationBits(epochParticipation); err != nil { return nil, err } return state, nil }