2019-11-26 23:54:46 +00:00
|
|
|
use crate::chunked_vector::{
|
|
|
|
load_variable_list_from_db, load_vector_from_db, BlockRoots, HistoricalRoots, RandaoMixes,
|
|
|
|
StateRoots,
|
|
|
|
};
|
2021-07-09 06:15:32 +00:00
|
|
|
use crate::{get_key_for_col, DBColumn, Error, KeyValueStore, KeyValueStoreOp};
|
|
|
|
use ssz::{Decode, DecodeError, Encode};
|
2019-11-26 23:54:46 +00:00
|
|
|
use ssz_derive::{Decode, Encode};
|
|
|
|
use std::convert::TryInto;
|
2021-07-15 00:52:02 +00:00
|
|
|
use std::sync::Arc;
|
2021-07-09 06:15:32 +00:00
|
|
|
use types::superstruct;
|
2019-11-26 23:54:46 +00:00
|
|
|
use types::*;
|
|
|
|
|
|
|
|
/// Lightweight variant of the `BeaconState` that is stored in the database.
|
|
|
|
///
|
|
|
|
/// Utilises lazy-loading from separate storage for its vector fields.
|
2021-07-09 06:15:32 +00:00
|
|
|
#[superstruct(
|
2022-09-17 09:26:18 +00:00
|
|
|
variants(Base, Altair, Merge, Eip4844),
|
2021-09-08 18:45:22 +00:00
|
|
|
variant_attributes(derive(Debug, PartialEq, Clone, Encode, Decode))
|
2021-07-09 06:15:32 +00:00
|
|
|
)]
|
|
|
|
#[derive(Debug, PartialEq, Clone, Encode)]
|
Implement SSZ union type (#2579)
## Issue Addressed
NA
## Proposed Changes
Implements the "union" type from the SSZ spec for `ssz`, `ssz_derive`, `tree_hash` and `tree_hash_derive` so it may be derived for `enums`:
https://github.com/ethereum/consensus-specs/blob/v1.1.0-beta.3/ssz/simple-serialize.md#union
The union type is required for the merge, since the `Transaction` type is defined as a single-variant union `Union[OpaqueTransaction]`.
### Crate Updates
This PR will (hopefully) cause CI to publish new versions for the following crates:
- `eth2_ssz_derive`: `0.2.1` -> `0.3.0`
- `eth2_ssz`: `0.3.0` -> `0.4.0`
- `eth2_ssz_types`: `0.2.0` -> `0.2.1`
- `tree_hash`: `0.3.0` -> `0.4.0`
- `tree_hash_derive`: `0.3.0` -> `0.4.0`
These these crates depend on each other, I've had to add a workspace-level `[patch]` for these crates. A follow-up PR will need to remove this patch, ones the new versions are published.
### Union Behaviors
We already had SSZ `Encode` and `TreeHash` derive for enums, however it just did a "transparent" pass-through of the inner value. Since the "union" decoding from the spec is in conflict with the transparent method, I've required that all `enum` have exactly one of the following enum-level attributes:
#### SSZ
- `#[ssz(enum_behaviour = "union")]`
- matches the spec used for the merge
- `#[ssz(enum_behaviour = "transparent")]`
- maintains existing functionality
- not supported for `Decode` (never was)
#### TreeHash
- `#[tree_hash(enum_behaviour = "union")]`
- matches the spec used for the merge
- `#[tree_hash(enum_behaviour = "transparent")]`
- maintains existing functionality
This means that we can maintain the existing transparent behaviour, but all existing users will get a compile-time error until they explicitly opt-in to being transparent.
### Legacy Option Encoding
Before this PR, we already had a union-esque encoding for `Option<T>`. However, this was with the *old* SSZ spec where the union selector was 4 bytes. During merge specification, the spec was changed to use 1 byte for the selector.
Whilst the 4-byte `Option` encoding was never used in the spec, we used it in our database. Writing a migrate script for all occurrences of `Option` in the database would be painful, especially since it's used in the `CommitteeCache`. To avoid the migrate script, I added a serde-esque `#[ssz(with = "module")]` field-level attribute to `ssz_derive` so that we can opt into the 4-byte encoding on a field-by-field basis.
The `ssz::legacy::four_byte_impl!` macro allows a one-liner to define the module required for the `#[ssz(with = "module")]` for some `Option<T> where T: Encode + Decode`.
Notably, **I have removed `Encode` and `Decode` impls for `Option`**. I've done this to force a break on downstream users. Like I mentioned, `Option` isn't used in the spec so I don't think it'll be *that* annoying. I think it's nicer than quietly having two different union implementations or quietly breaking the existing `Option` impl.
### Crate Publish Ordering
I've modified the order in which CI publishes crates to ensure that we don't publish a crate without ensuring we already published a crate that it depends upon.
## TODO
- [ ] Queue a follow-up `[patch]`-removing PR.
2021-09-25 05:58:36 +00:00
|
|
|
#[ssz(enum_behaviour = "transparent")]
|
2019-11-26 23:54:46 +00:00
|
|
|
pub struct PartialBeaconState<T>
|
|
|
|
where
|
|
|
|
T: EthSpec,
|
|
|
|
{
|
|
|
|
// Versioning
|
|
|
|
pub genesis_time: u64,
|
2020-04-01 11:03:03 +00:00
|
|
|
pub genesis_validators_root: Hash256,
|
2021-07-09 06:15:32 +00:00
|
|
|
#[superstruct(getter(copy))]
|
2019-11-26 23:54:46 +00:00
|
|
|
pub slot: Slot,
|
|
|
|
pub fork: Fork,
|
|
|
|
|
|
|
|
// History
|
|
|
|
pub latest_block_header: BeaconBlockHeader,
|
|
|
|
|
Implement SSZ union type (#2579)
## Issue Addressed
NA
## Proposed Changes
Implements the "union" type from the SSZ spec for `ssz`, `ssz_derive`, `tree_hash` and `tree_hash_derive` so it may be derived for `enums`:
https://github.com/ethereum/consensus-specs/blob/v1.1.0-beta.3/ssz/simple-serialize.md#union
The union type is required for the merge, since the `Transaction` type is defined as a single-variant union `Union[OpaqueTransaction]`.
### Crate Updates
This PR will (hopefully) cause CI to publish new versions for the following crates:
- `eth2_ssz_derive`: `0.2.1` -> `0.3.0`
- `eth2_ssz`: `0.3.0` -> `0.4.0`
- `eth2_ssz_types`: `0.2.0` -> `0.2.1`
- `tree_hash`: `0.3.0` -> `0.4.0`
- `tree_hash_derive`: `0.3.0` -> `0.4.0`
These these crates depend on each other, I've had to add a workspace-level `[patch]` for these crates. A follow-up PR will need to remove this patch, ones the new versions are published.
### Union Behaviors
We already had SSZ `Encode` and `TreeHash` derive for enums, however it just did a "transparent" pass-through of the inner value. Since the "union" decoding from the spec is in conflict with the transparent method, I've required that all `enum` have exactly one of the following enum-level attributes:
#### SSZ
- `#[ssz(enum_behaviour = "union")]`
- matches the spec used for the merge
- `#[ssz(enum_behaviour = "transparent")]`
- maintains existing functionality
- not supported for `Decode` (never was)
#### TreeHash
- `#[tree_hash(enum_behaviour = "union")]`
- matches the spec used for the merge
- `#[tree_hash(enum_behaviour = "transparent")]`
- maintains existing functionality
This means that we can maintain the existing transparent behaviour, but all existing users will get a compile-time error until they explicitly opt-in to being transparent.
### Legacy Option Encoding
Before this PR, we already had a union-esque encoding for `Option<T>`. However, this was with the *old* SSZ spec where the union selector was 4 bytes. During merge specification, the spec was changed to use 1 byte for the selector.
Whilst the 4-byte `Option` encoding was never used in the spec, we used it in our database. Writing a migrate script for all occurrences of `Option` in the database would be painful, especially since it's used in the `CommitteeCache`. To avoid the migrate script, I added a serde-esque `#[ssz(with = "module")]` field-level attribute to `ssz_derive` so that we can opt into the 4-byte encoding on a field-by-field basis.
The `ssz::legacy::four_byte_impl!` macro allows a one-liner to define the module required for the `#[ssz(with = "module")]` for some `Option<T> where T: Encode + Decode`.
Notably, **I have removed `Encode` and `Decode` impls for `Option`**. I've done this to force a break on downstream users. Like I mentioned, `Option` isn't used in the spec so I don't think it'll be *that* annoying. I think it's nicer than quietly having two different union implementations or quietly breaking the existing `Option` impl.
### Crate Publish Ordering
I've modified the order in which CI publishes crates to ensure that we don't publish a crate without ensuring we already published a crate that it depends upon.
## TODO
- [ ] Queue a follow-up `[patch]`-removing PR.
2021-09-25 05:58:36 +00:00
|
|
|
#[ssz(skip_serializing, skip_deserializing)]
|
2019-11-26 23:54:46 +00:00
|
|
|
pub block_roots: Option<FixedVector<Hash256, T::SlotsPerHistoricalRoot>>,
|
Implement SSZ union type (#2579)
## Issue Addressed
NA
## Proposed Changes
Implements the "union" type from the SSZ spec for `ssz`, `ssz_derive`, `tree_hash` and `tree_hash_derive` so it may be derived for `enums`:
https://github.com/ethereum/consensus-specs/blob/v1.1.0-beta.3/ssz/simple-serialize.md#union
The union type is required for the merge, since the `Transaction` type is defined as a single-variant union `Union[OpaqueTransaction]`.
### Crate Updates
This PR will (hopefully) cause CI to publish new versions for the following crates:
- `eth2_ssz_derive`: `0.2.1` -> `0.3.0`
- `eth2_ssz`: `0.3.0` -> `0.4.0`
- `eth2_ssz_types`: `0.2.0` -> `0.2.1`
- `tree_hash`: `0.3.0` -> `0.4.0`
- `tree_hash_derive`: `0.3.0` -> `0.4.0`
These these crates depend on each other, I've had to add a workspace-level `[patch]` for these crates. A follow-up PR will need to remove this patch, ones the new versions are published.
### Union Behaviors
We already had SSZ `Encode` and `TreeHash` derive for enums, however it just did a "transparent" pass-through of the inner value. Since the "union" decoding from the spec is in conflict with the transparent method, I've required that all `enum` have exactly one of the following enum-level attributes:
#### SSZ
- `#[ssz(enum_behaviour = "union")]`
- matches the spec used for the merge
- `#[ssz(enum_behaviour = "transparent")]`
- maintains existing functionality
- not supported for `Decode` (never was)
#### TreeHash
- `#[tree_hash(enum_behaviour = "union")]`
- matches the spec used for the merge
- `#[tree_hash(enum_behaviour = "transparent")]`
- maintains existing functionality
This means that we can maintain the existing transparent behaviour, but all existing users will get a compile-time error until they explicitly opt-in to being transparent.
### Legacy Option Encoding
Before this PR, we already had a union-esque encoding for `Option<T>`. However, this was with the *old* SSZ spec where the union selector was 4 bytes. During merge specification, the spec was changed to use 1 byte for the selector.
Whilst the 4-byte `Option` encoding was never used in the spec, we used it in our database. Writing a migrate script for all occurrences of `Option` in the database would be painful, especially since it's used in the `CommitteeCache`. To avoid the migrate script, I added a serde-esque `#[ssz(with = "module")]` field-level attribute to `ssz_derive` so that we can opt into the 4-byte encoding on a field-by-field basis.
The `ssz::legacy::four_byte_impl!` macro allows a one-liner to define the module required for the `#[ssz(with = "module")]` for some `Option<T> where T: Encode + Decode`.
Notably, **I have removed `Encode` and `Decode` impls for `Option`**. I've done this to force a break on downstream users. Like I mentioned, `Option` isn't used in the spec so I don't think it'll be *that* annoying. I think it's nicer than quietly having two different union implementations or quietly breaking the existing `Option` impl.
### Crate Publish Ordering
I've modified the order in which CI publishes crates to ensure that we don't publish a crate without ensuring we already published a crate that it depends upon.
## TODO
- [ ] Queue a follow-up `[patch]`-removing PR.
2021-09-25 05:58:36 +00:00
|
|
|
#[ssz(skip_serializing, skip_deserializing)]
|
2019-11-26 23:54:46 +00:00
|
|
|
pub state_roots: Option<FixedVector<Hash256, T::SlotsPerHistoricalRoot>>,
|
|
|
|
|
Implement SSZ union type (#2579)
## Issue Addressed
NA
## Proposed Changes
Implements the "union" type from the SSZ spec for `ssz`, `ssz_derive`, `tree_hash` and `tree_hash_derive` so it may be derived for `enums`:
https://github.com/ethereum/consensus-specs/blob/v1.1.0-beta.3/ssz/simple-serialize.md#union
The union type is required for the merge, since the `Transaction` type is defined as a single-variant union `Union[OpaqueTransaction]`.
### Crate Updates
This PR will (hopefully) cause CI to publish new versions for the following crates:
- `eth2_ssz_derive`: `0.2.1` -> `0.3.0`
- `eth2_ssz`: `0.3.0` -> `0.4.0`
- `eth2_ssz_types`: `0.2.0` -> `0.2.1`
- `tree_hash`: `0.3.0` -> `0.4.0`
- `tree_hash_derive`: `0.3.0` -> `0.4.0`
These these crates depend on each other, I've had to add a workspace-level `[patch]` for these crates. A follow-up PR will need to remove this patch, ones the new versions are published.
### Union Behaviors
We already had SSZ `Encode` and `TreeHash` derive for enums, however it just did a "transparent" pass-through of the inner value. Since the "union" decoding from the spec is in conflict with the transparent method, I've required that all `enum` have exactly one of the following enum-level attributes:
#### SSZ
- `#[ssz(enum_behaviour = "union")]`
- matches the spec used for the merge
- `#[ssz(enum_behaviour = "transparent")]`
- maintains existing functionality
- not supported for `Decode` (never was)
#### TreeHash
- `#[tree_hash(enum_behaviour = "union")]`
- matches the spec used for the merge
- `#[tree_hash(enum_behaviour = "transparent")]`
- maintains existing functionality
This means that we can maintain the existing transparent behaviour, but all existing users will get a compile-time error until they explicitly opt-in to being transparent.
### Legacy Option Encoding
Before this PR, we already had a union-esque encoding for `Option<T>`. However, this was with the *old* SSZ spec where the union selector was 4 bytes. During merge specification, the spec was changed to use 1 byte for the selector.
Whilst the 4-byte `Option` encoding was never used in the spec, we used it in our database. Writing a migrate script for all occurrences of `Option` in the database would be painful, especially since it's used in the `CommitteeCache`. To avoid the migrate script, I added a serde-esque `#[ssz(with = "module")]` field-level attribute to `ssz_derive` so that we can opt into the 4-byte encoding on a field-by-field basis.
The `ssz::legacy::four_byte_impl!` macro allows a one-liner to define the module required for the `#[ssz(with = "module")]` for some `Option<T> where T: Encode + Decode`.
Notably, **I have removed `Encode` and `Decode` impls for `Option`**. I've done this to force a break on downstream users. Like I mentioned, `Option` isn't used in the spec so I don't think it'll be *that* annoying. I think it's nicer than quietly having two different union implementations or quietly breaking the existing `Option` impl.
### Crate Publish Ordering
I've modified the order in which CI publishes crates to ensure that we don't publish a crate without ensuring we already published a crate that it depends upon.
## TODO
- [ ] Queue a follow-up `[patch]`-removing PR.
2021-09-25 05:58:36 +00:00
|
|
|
#[ssz(skip_serializing, skip_deserializing)]
|
2019-11-26 23:54:46 +00:00
|
|
|
pub historical_roots: Option<VariableList<Hash256, T::HistoricalRootsLimit>>,
|
|
|
|
|
|
|
|
// Ethereum 1.0 chain data
|
|
|
|
pub eth1_data: Eth1Data,
|
|
|
|
pub eth1_data_votes: VariableList<Eth1Data, T::SlotsPerEth1VotingPeriod>,
|
|
|
|
pub eth1_deposit_index: u64,
|
|
|
|
|
|
|
|
// Registry
|
|
|
|
pub validators: VariableList<Validator, T::ValidatorRegistryLimit>,
|
|
|
|
pub balances: VariableList<u64, T::ValidatorRegistryLimit>,
|
|
|
|
|
|
|
|
// Shuffling
|
|
|
|
/// Randao value from the current slot, for patching into the per-epoch randao vector.
|
|
|
|
pub latest_randao_value: Hash256,
|
Implement SSZ union type (#2579)
## Issue Addressed
NA
## Proposed Changes
Implements the "union" type from the SSZ spec for `ssz`, `ssz_derive`, `tree_hash` and `tree_hash_derive` so it may be derived for `enums`:
https://github.com/ethereum/consensus-specs/blob/v1.1.0-beta.3/ssz/simple-serialize.md#union
The union type is required for the merge, since the `Transaction` type is defined as a single-variant union `Union[OpaqueTransaction]`.
### Crate Updates
This PR will (hopefully) cause CI to publish new versions for the following crates:
- `eth2_ssz_derive`: `0.2.1` -> `0.3.0`
- `eth2_ssz`: `0.3.0` -> `0.4.0`
- `eth2_ssz_types`: `0.2.0` -> `0.2.1`
- `tree_hash`: `0.3.0` -> `0.4.0`
- `tree_hash_derive`: `0.3.0` -> `0.4.0`
These these crates depend on each other, I've had to add a workspace-level `[patch]` for these crates. A follow-up PR will need to remove this patch, ones the new versions are published.
### Union Behaviors
We already had SSZ `Encode` and `TreeHash` derive for enums, however it just did a "transparent" pass-through of the inner value. Since the "union" decoding from the spec is in conflict with the transparent method, I've required that all `enum` have exactly one of the following enum-level attributes:
#### SSZ
- `#[ssz(enum_behaviour = "union")]`
- matches the spec used for the merge
- `#[ssz(enum_behaviour = "transparent")]`
- maintains existing functionality
- not supported for `Decode` (never was)
#### TreeHash
- `#[tree_hash(enum_behaviour = "union")]`
- matches the spec used for the merge
- `#[tree_hash(enum_behaviour = "transparent")]`
- maintains existing functionality
This means that we can maintain the existing transparent behaviour, but all existing users will get a compile-time error until they explicitly opt-in to being transparent.
### Legacy Option Encoding
Before this PR, we already had a union-esque encoding for `Option<T>`. However, this was with the *old* SSZ spec where the union selector was 4 bytes. During merge specification, the spec was changed to use 1 byte for the selector.
Whilst the 4-byte `Option` encoding was never used in the spec, we used it in our database. Writing a migrate script for all occurrences of `Option` in the database would be painful, especially since it's used in the `CommitteeCache`. To avoid the migrate script, I added a serde-esque `#[ssz(with = "module")]` field-level attribute to `ssz_derive` so that we can opt into the 4-byte encoding on a field-by-field basis.
The `ssz::legacy::four_byte_impl!` macro allows a one-liner to define the module required for the `#[ssz(with = "module")]` for some `Option<T> where T: Encode + Decode`.
Notably, **I have removed `Encode` and `Decode` impls for `Option`**. I've done this to force a break on downstream users. Like I mentioned, `Option` isn't used in the spec so I don't think it'll be *that* annoying. I think it's nicer than quietly having two different union implementations or quietly breaking the existing `Option` impl.
### Crate Publish Ordering
I've modified the order in which CI publishes crates to ensure that we don't publish a crate without ensuring we already published a crate that it depends upon.
## TODO
- [ ] Queue a follow-up `[patch]`-removing PR.
2021-09-25 05:58:36 +00:00
|
|
|
#[ssz(skip_serializing, skip_deserializing)]
|
2019-11-26 23:54:46 +00:00
|
|
|
pub randao_mixes: Option<FixedVector<Hash256, T::EpochsPerHistoricalVector>>,
|
|
|
|
|
|
|
|
// Slashings
|
|
|
|
slashings: FixedVector<u64, T::EpochsPerSlashingsVector>,
|
|
|
|
|
2021-07-09 06:15:32 +00:00
|
|
|
// Attestations (genesis fork only)
|
|
|
|
#[superstruct(only(Base))]
|
2019-11-26 23:54:46 +00:00
|
|
|
pub previous_epoch_attestations: VariableList<PendingAttestation<T>, T::MaxPendingAttestations>,
|
2021-07-09 06:15:32 +00:00
|
|
|
#[superstruct(only(Base))]
|
2019-11-26 23:54:46 +00:00
|
|
|
pub current_epoch_attestations: VariableList<PendingAttestation<T>, T::MaxPendingAttestations>,
|
|
|
|
|
2021-07-09 06:15:32 +00:00
|
|
|
// Participation (Altair and later)
|
2021-09-08 18:45:22 +00:00
|
|
|
#[superstruct(only(Altair, Merge))]
|
2021-07-09 06:15:32 +00:00
|
|
|
pub previous_epoch_participation: VariableList<ParticipationFlags, T::ValidatorRegistryLimit>,
|
2021-09-08 18:45:22 +00:00
|
|
|
#[superstruct(only(Altair, Merge))]
|
2021-07-09 06:15:32 +00:00
|
|
|
pub current_epoch_participation: VariableList<ParticipationFlags, T::ValidatorRegistryLimit>,
|
|
|
|
|
2019-11-26 23:54:46 +00:00
|
|
|
// Finality
|
|
|
|
pub justification_bits: BitVector<T::JustificationBitsLength>,
|
|
|
|
pub previous_justified_checkpoint: Checkpoint,
|
|
|
|
pub current_justified_checkpoint: Checkpoint,
|
|
|
|
pub finalized_checkpoint: Checkpoint,
|
2021-07-09 06:15:32 +00:00
|
|
|
|
|
|
|
// Inactivity
|
2021-09-08 18:45:22 +00:00
|
|
|
#[superstruct(only(Altair, Merge))]
|
2021-07-09 06:15:32 +00:00
|
|
|
pub inactivity_scores: VariableList<u64, T::ValidatorRegistryLimit>,
|
|
|
|
|
|
|
|
// Light-client sync committees
|
2021-09-08 18:45:22 +00:00
|
|
|
#[superstruct(only(Altair, Merge))]
|
2021-07-15 00:52:02 +00:00
|
|
|
pub current_sync_committee: Arc<SyncCommittee<T>>,
|
2021-09-08 18:45:22 +00:00
|
|
|
#[superstruct(only(Altair, Merge))]
|
2021-07-15 00:52:02 +00:00
|
|
|
pub next_sync_committee: Arc<SyncCommittee<T>>,
|
2021-09-08 18:45:22 +00:00
|
|
|
|
|
|
|
// Execution
|
|
|
|
#[superstruct(only(Merge))]
|
|
|
|
pub latest_execution_payload_header: ExecutionPayloadHeader<T>,
|
2019-11-26 23:54:46 +00:00
|
|
|
}
|
|
|
|
|
2021-07-09 06:15:32 +00:00
|
|
|
/// Implement the conversion function from BeaconState -> PartialBeaconState.
|
|
|
|
macro_rules! impl_from_state_forgetful {
|
|
|
|
($s:ident, $outer:ident, $variant_name:ident, $struct_name:ident, [$($extra_fields:ident),*]) => {
|
|
|
|
PartialBeaconState::$variant_name($struct_name {
|
|
|
|
// Versioning
|
|
|
|
genesis_time: $s.genesis_time,
|
|
|
|
genesis_validators_root: $s.genesis_validators_root,
|
|
|
|
slot: $s.slot,
|
|
|
|
fork: $s.fork,
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// History
|
2021-07-09 06:15:32 +00:00
|
|
|
latest_block_header: $s.latest_block_header.clone(),
|
2019-11-26 23:54:46 +00:00
|
|
|
block_roots: None,
|
|
|
|
state_roots: None,
|
|
|
|
historical_roots: None,
|
|
|
|
|
|
|
|
// Eth1
|
2021-07-09 06:15:32 +00:00
|
|
|
eth1_data: $s.eth1_data.clone(),
|
|
|
|
eth1_data_votes: $s.eth1_data_votes.clone(),
|
|
|
|
eth1_deposit_index: $s.eth1_deposit_index,
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// Validator registry
|
2021-07-09 06:15:32 +00:00
|
|
|
validators: $s.validators.clone(),
|
|
|
|
balances: $s.balances.clone(),
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// Shuffling
|
2021-07-09 06:15:32 +00:00
|
|
|
latest_randao_value: *$outer
|
|
|
|
.get_randao_mix($outer.current_epoch())
|
2019-11-26 23:54:46 +00:00
|
|
|
.expect("randao at current epoch is OK"),
|
|
|
|
randao_mixes: None,
|
|
|
|
|
|
|
|
// Slashings
|
2021-07-09 06:15:32 +00:00
|
|
|
slashings: $s.slashings.clone(),
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// Finality
|
2021-07-09 06:15:32 +00:00
|
|
|
justification_bits: $s.justification_bits.clone(),
|
|
|
|
previous_justified_checkpoint: $s.previous_justified_checkpoint,
|
|
|
|
current_justified_checkpoint: $s.current_justified_checkpoint,
|
|
|
|
finalized_checkpoint: $s.finalized_checkpoint,
|
|
|
|
|
|
|
|
// Variant-specific fields
|
|
|
|
$(
|
|
|
|
$extra_fields: $s.$extra_fields.clone()
|
|
|
|
),*
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: EthSpec> PartialBeaconState<T> {
|
|
|
|
/// Convert a `BeaconState` to a `PartialBeaconState`, while dropping the optional fields.
|
|
|
|
pub fn from_state_forgetful(outer: &BeaconState<T>) -> Self {
|
|
|
|
match outer {
|
|
|
|
BeaconState::Base(s) => impl_from_state_forgetful!(
|
|
|
|
s,
|
|
|
|
outer,
|
|
|
|
Base,
|
|
|
|
PartialBeaconStateBase,
|
|
|
|
[previous_epoch_attestations, current_epoch_attestations]
|
|
|
|
),
|
|
|
|
BeaconState::Altair(s) => impl_from_state_forgetful!(
|
|
|
|
s,
|
|
|
|
outer,
|
|
|
|
Altair,
|
|
|
|
PartialBeaconStateAltair,
|
|
|
|
[
|
|
|
|
previous_epoch_participation,
|
|
|
|
current_epoch_participation,
|
|
|
|
current_sync_committee,
|
|
|
|
next_sync_committee,
|
|
|
|
inactivity_scores
|
|
|
|
]
|
|
|
|
),
|
2021-09-08 18:45:22 +00:00
|
|
|
BeaconState::Merge(s) => impl_from_state_forgetful!(
|
|
|
|
s,
|
|
|
|
outer,
|
|
|
|
Merge,
|
|
|
|
PartialBeaconStateMerge,
|
|
|
|
[
|
|
|
|
previous_epoch_participation,
|
|
|
|
current_epoch_participation,
|
|
|
|
current_sync_committee,
|
|
|
|
next_sync_committee,
|
|
|
|
inactivity_scores,
|
|
|
|
latest_execution_payload_header
|
|
|
|
]
|
|
|
|
),
|
2019-11-26 23:54:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-09 06:15:32 +00:00
|
|
|
/// SSZ decode.
|
|
|
|
pub fn from_ssz_bytes(bytes: &[u8], spec: &ChainSpec) -> Result<Self, ssz::DecodeError> {
|
|
|
|
// Slot is after genesis_time (u64) and genesis_validators_root (Hash256).
|
|
|
|
let slot_offset = <u64 as Decode>::ssz_fixed_len() + <Hash256 as Decode>::ssz_fixed_len();
|
|
|
|
let slot_len = <Slot as Decode>::ssz_fixed_len();
|
|
|
|
let slot_bytes = bytes.get(slot_offset..slot_offset + slot_len).ok_or(
|
|
|
|
DecodeError::InvalidByteLength {
|
|
|
|
len: bytes.len(),
|
|
|
|
expected: slot_offset + slot_len,
|
|
|
|
},
|
|
|
|
)?;
|
|
|
|
|
|
|
|
let slot = Slot::from_ssz_bytes(slot_bytes)?;
|
2021-10-28 01:18:04 +00:00
|
|
|
let fork_at_slot = spec.fork_name_at_slot::<T>(slot);
|
|
|
|
|
|
|
|
Ok(map_fork_name!(
|
|
|
|
fork_at_slot,
|
|
|
|
Self,
|
|
|
|
<_>::from_ssz_bytes(bytes)?
|
|
|
|
))
|
2021-07-09 06:15:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Prepare the partial state for storage in the KV database.
|
|
|
|
pub fn as_kv_store_op(&self, state_root: Hash256) -> KeyValueStoreOp {
|
|
|
|
let db_key = get_key_for_col(DBColumn::BeaconState.into(), state_root.as_bytes());
|
|
|
|
KeyValueStoreOp::PutKeyValue(db_key, self.as_ssz_bytes())
|
|
|
|
}
|
|
|
|
|
2020-05-31 22:13:49 +00:00
|
|
|
pub fn load_block_roots<S: KeyValueStore<T>>(
|
2019-12-06 07:52:11 +00:00
|
|
|
&mut self,
|
|
|
|
store: &S,
|
|
|
|
spec: &ChainSpec,
|
|
|
|
) -> Result<(), Error> {
|
2021-07-09 06:15:32 +00:00
|
|
|
if self.block_roots().is_none() {
|
|
|
|
*self.block_roots_mut() = Some(load_vector_from_db::<BlockRoots, T, _>(
|
|
|
|
store,
|
|
|
|
self.slot(),
|
|
|
|
spec,
|
2019-11-26 23:54:46 +00:00
|
|
|
)?);
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-05-31 22:13:49 +00:00
|
|
|
pub fn load_state_roots<S: KeyValueStore<T>>(
|
2019-12-06 07:52:11 +00:00
|
|
|
&mut self,
|
|
|
|
store: &S,
|
|
|
|
spec: &ChainSpec,
|
|
|
|
) -> Result<(), Error> {
|
2021-07-09 06:15:32 +00:00
|
|
|
if self.state_roots().is_none() {
|
|
|
|
*self.state_roots_mut() = Some(load_vector_from_db::<StateRoots, T, _>(
|
|
|
|
store,
|
|
|
|
self.slot(),
|
|
|
|
spec,
|
2019-11-26 23:54:46 +00:00
|
|
|
)?);
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-05-31 22:13:49 +00:00
|
|
|
pub fn load_historical_roots<S: KeyValueStore<T>>(
|
2019-11-26 23:54:46 +00:00
|
|
|
&mut self,
|
|
|
|
store: &S,
|
|
|
|
spec: &ChainSpec,
|
|
|
|
) -> Result<(), Error> {
|
2021-07-09 06:15:32 +00:00
|
|
|
if self.historical_roots().is_none() {
|
|
|
|
*self.historical_roots_mut() = Some(
|
|
|
|
load_variable_list_from_db::<HistoricalRoots, T, _>(store, self.slot(), spec)?,
|
|
|
|
);
|
2019-11-26 23:54:46 +00:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-05-31 22:13:49 +00:00
|
|
|
pub fn load_randao_mixes<S: KeyValueStore<T>>(
|
2019-11-26 23:54:46 +00:00
|
|
|
&mut self,
|
|
|
|
store: &S,
|
|
|
|
spec: &ChainSpec,
|
|
|
|
) -> Result<(), Error> {
|
2021-07-09 06:15:32 +00:00
|
|
|
if self.randao_mixes().is_none() {
|
2019-11-26 23:54:46 +00:00
|
|
|
// Load the per-epoch values from the database
|
|
|
|
let mut randao_mixes =
|
2021-07-09 06:15:32 +00:00
|
|
|
load_vector_from_db::<RandaoMixes, T, _>(store, self.slot(), spec)?;
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// Patch the value for the current slot into the index for the current epoch
|
2021-07-09 06:15:32 +00:00
|
|
|
let current_epoch = self.slot().epoch(T::slots_per_epoch());
|
2019-11-26 23:54:46 +00:00
|
|
|
let len = randao_mixes.len();
|
2021-07-09 06:15:32 +00:00
|
|
|
randao_mixes[current_epoch.as_usize() % len] = *self.latest_randao_value();
|
2019-11-26 23:54:46 +00:00
|
|
|
|
2021-07-09 06:15:32 +00:00
|
|
|
*self.randao_mixes_mut() = Some(randao_mixes)
|
2019-11-26 23:54:46 +00:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-09 06:15:32 +00:00
|
|
|
/// Implement the conversion from PartialBeaconState -> BeaconState.
|
|
|
|
macro_rules! impl_try_into_beacon_state {
|
|
|
|
($inner:ident, $variant_name:ident, $struct_name:ident, [$($extra_fields:ident),*]) => {
|
|
|
|
BeaconState::$variant_name($struct_name {
|
|
|
|
// Versioning
|
|
|
|
genesis_time: $inner.genesis_time,
|
|
|
|
genesis_validators_root: $inner.genesis_validators_root,
|
|
|
|
slot: $inner.slot,
|
|
|
|
fork: $inner.fork,
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// History
|
2021-07-09 06:15:32 +00:00
|
|
|
latest_block_header: $inner.latest_block_header,
|
|
|
|
block_roots: unpack_field($inner.block_roots)?,
|
|
|
|
state_roots: unpack_field($inner.state_roots)?,
|
|
|
|
historical_roots: unpack_field($inner.historical_roots)?,
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// Eth1
|
2021-07-09 06:15:32 +00:00
|
|
|
eth1_data: $inner.eth1_data,
|
|
|
|
eth1_data_votes: $inner.eth1_data_votes,
|
|
|
|
eth1_deposit_index: $inner.eth1_deposit_index,
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// Validator registry
|
2021-07-09 06:15:32 +00:00
|
|
|
validators: $inner.validators,
|
|
|
|
balances: $inner.balances,
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// Shuffling
|
2021-07-09 06:15:32 +00:00
|
|
|
randao_mixes: unpack_field($inner.randao_mixes)?,
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// Slashings
|
2021-07-09 06:15:32 +00:00
|
|
|
slashings: $inner.slashings,
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// Finality
|
2021-07-09 06:15:32 +00:00
|
|
|
justification_bits: $inner.justification_bits,
|
|
|
|
previous_justified_checkpoint: $inner.previous_justified_checkpoint,
|
|
|
|
current_justified_checkpoint: $inner.current_justified_checkpoint,
|
|
|
|
finalized_checkpoint: $inner.finalized_checkpoint,
|
2019-11-26 23:54:46 +00:00
|
|
|
|
|
|
|
// Caching
|
2021-09-03 07:50:43 +00:00
|
|
|
total_active_balance: <_>::default(),
|
2019-11-26 23:54:46 +00:00
|
|
|
committee_caches: <_>::default(),
|
|
|
|
pubkey_cache: <_>::default(),
|
|
|
|
exit_cache: <_>::default(),
|
|
|
|
tree_hash_cache: <_>::default(),
|
2021-07-09 06:15:32 +00:00
|
|
|
|
|
|
|
// Variant-specific fields
|
|
|
|
$(
|
|
|
|
$extra_fields: $inner.$extra_fields
|
|
|
|
),*
|
2019-11-26 23:54:46 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2021-07-09 06:15:32 +00:00
|
|
|
|
|
|
|
fn unpack_field<T>(x: Option<T>) -> Result<T, Error> {
|
|
|
|
x.ok_or(Error::PartialBeaconStateError)
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<E: EthSpec> TryInto<BeaconState<E>> for PartialBeaconState<E> {
|
|
|
|
type Error = Error;
|
|
|
|
|
|
|
|
fn try_into(self) -> Result<BeaconState<E>, Error> {
|
|
|
|
let state = match self {
|
|
|
|
PartialBeaconState::Base(inner) => impl_try_into_beacon_state!(
|
|
|
|
inner,
|
|
|
|
Base,
|
|
|
|
BeaconStateBase,
|
|
|
|
[previous_epoch_attestations, current_epoch_attestations]
|
|
|
|
),
|
|
|
|
PartialBeaconState::Altair(inner) => impl_try_into_beacon_state!(
|
|
|
|
inner,
|
|
|
|
Altair,
|
|
|
|
BeaconStateAltair,
|
|
|
|
[
|
|
|
|
previous_epoch_participation,
|
|
|
|
current_epoch_participation,
|
|
|
|
current_sync_committee,
|
|
|
|
next_sync_committee,
|
|
|
|
inactivity_scores
|
|
|
|
]
|
|
|
|
),
|
2021-09-08 18:45:22 +00:00
|
|
|
PartialBeaconState::Merge(inner) => impl_try_into_beacon_state!(
|
|
|
|
inner,
|
|
|
|
Merge,
|
|
|
|
BeaconStateMerge,
|
|
|
|
[
|
|
|
|
previous_epoch_participation,
|
|
|
|
current_epoch_participation,
|
|
|
|
current_sync_committee,
|
|
|
|
next_sync_committee,
|
|
|
|
inactivity_scores,
|
|
|
|
latest_execution_payload_header
|
|
|
|
]
|
|
|
|
),
|
2021-07-09 06:15:32 +00:00
|
|
|
};
|
|
|
|
Ok(state)
|
|
|
|
}
|
|
|
|
}
|