Fix bugs from removing speculative state

This commit is contained in:
Paul Hauner 2019-08-30 00:24:46 +10:00
parent 682081ef07
commit ae114889c1
No known key found for this signature in database
GPG Key ID: 303E4494BB28068C
2 changed files with 45 additions and 4 deletions

View File

@ -90,6 +90,15 @@ impl<'a, T: EthSpec> AsRef<BeaconState<T>> for StateCow<'a, T> {
} }
} }
impl<'a, T: EthSpec> StateCow<'a, T> {
pub fn as_mut_ref(&mut self) -> Option<&mut BeaconState<T>> {
match self {
StateCow::Borrowed(_) => None,
StateCow::Owned(ref mut state) => Some(state),
}
}
}
pub trait BeaconChainTypes: Send + Sync + 'static { pub trait BeaconChainTypes: Send + Sync + 'static {
type Store: store::Store; type Store: store::Store;
type SlotClock: slot_clock::SlotClock; type SlotClock: slot_clock::SlotClock;
@ -447,12 +456,16 @@ impl<T: BeaconChainTypes> BeaconChain<T> {
let epoch = |slot: Slot| slot.epoch(T::EthSpec::slots_per_epoch()); let epoch = |slot: Slot| slot.epoch(T::EthSpec::slots_per_epoch());
let head_state = &self.head().beacon_state; let head_state = &self.head().beacon_state;
let state = if epoch(slot) == epoch(head_state.slot) { let mut state = if epoch(slot) == epoch(head_state.slot) {
StateCow::Borrowed(self.head()) StateCow::Borrowed(self.head())
} else { } else {
self.state_at_slot(slot)? self.state_at_slot(slot)?
}; };
if let Some(state) = state.as_mut_ref() {
state.build_committee_cache(RelativeEpoch::Current, &self.spec)?;
}
state state
.as_ref() .as_ref()
.get_beacon_proposer_index(slot, RelativeEpoch::Current, &self.spec) .get_beacon_proposer_index(slot, RelativeEpoch::Current, &self.spec)
@ -471,12 +484,16 @@ impl<T: BeaconChainTypes> BeaconChain<T> {
let as_epoch = |slot: Slot| slot.epoch(T::EthSpec::slots_per_epoch()); let as_epoch = |slot: Slot| slot.epoch(T::EthSpec::slots_per_epoch());
let head_state = &self.head().beacon_state; let head_state = &self.head().beacon_state;
let state = if epoch == as_epoch(head_state.slot) { let mut state = if epoch == as_epoch(head_state.slot) {
StateCow::Borrowed(self.head()) StateCow::Borrowed(self.head())
} else { } else {
self.state_at_slot(epoch.start_slot(T::EthSpec::slots_per_epoch()))? self.state_at_slot(epoch.start_slot(T::EthSpec::slots_per_epoch()))?
}; };
if let Some(state) = state.as_mut_ref() {
state.build_committee_cache(RelativeEpoch::Current, &self.spec)?;
}
if let Some(attestation_duty) = state if let Some(attestation_duty) = state
.as_ref() .as_ref()
.get_attestation_duties(validator_index, RelativeEpoch::Current)? .get_attestation_duties(validator_index, RelativeEpoch::Current)?

View File

@ -28,9 +28,33 @@ impl<T: BeaconChainTypes> ValidatorService for ValidatorServiceInstance<T> {
let validators = req.get_validators(); let validators = req.get_validators();
trace!(self.log, "RPC request"; "endpoint" => "GetValidatorDuties", "epoch" => req.get_epoch()); trace!(self.log, "RPC request"; "endpoint" => "GetValidatorDuties", "epoch" => req.get_epoch());
let slot = if let Ok(slot) = self.chain.slot() {
slot
} else {
let log_clone = self.log.clone();
let f = sink
.fail(RpcStatus::new(
RpcStatusCode::FailedPrecondition,
Some("No slot for chain".to_string()),
))
.map_err(move |e| warn!(log_clone, "failed to reply {:?}: {:?}", req, e));
return ctx.spawn(f);
};
let state_cow = if let Ok(state) = self.chain.state_at_slot(slot) {
state
} else {
let log_clone = self.log.clone();
let f = sink
.fail(RpcStatus::new(
RpcStatusCode::FailedPrecondition,
Some("No state".to_string()),
))
.map_err(move |e| warn!(log_clone, "failed to reply {:?}: {:?}", req, e));
return ctx.spawn(f);
};
let state = state_cow.as_ref();
let spec = &self.chain.spec; let spec = &self.chain.spec;
// TODO: this whole module is legacy and not maintained well.
let state = &self.chain.head().beacon_state;
let epoch = Epoch::from(req.get_epoch()); let epoch = Epoch::from(req.get_epoch());
let mut resp = GetDutiesResponse::new(); let mut resp = GetDutiesResponse::new();
let resp_validators = resp.mut_active_validators(); let resp_validators = resp.mut_active_validators();