More efficient ancestor head retrieval for GetAttestationData (#5669)

* Use `BlockRootAtSlot` to look up historical head root

* Update test

* Typo
This commit is contained in:
terence tsao 2020-04-28 17:44:06 -07:00 committed by GitHub
parent 44611e0fb2
commit 5636cd3ed8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 18 additions and 17 deletions

View File

@ -78,27 +78,27 @@ func (vs *Server) GetAttestationData(ctx context.Context, req *ethpb.Attestation
return nil, status.Errorf(codes.Internal, "Could not retrieve head root: %v", err) return nil, status.Errorf(codes.Internal, "Could not retrieve head root: %v", err)
} }
// In the case that we receive an attestation request after a newer state/block has been // In the case that we receive an attestation request after a newer state/block has been processed.
// processed, we walk up the chain until state.Slot <= req.Slot to prevent producing an if headState.Slot() > req.Slot {
// attestation that violates processing constraints. headRoot, err = helpers.BlockRootAtSlot(headState, req.Slot)
fetchState := vs.BeaconDB.State
if featureconfig.Get().NewStateMgmt {
fetchState = vs.StateGen.StateByRoot
}
for headState.Slot() > req.Slot {
if ctx.Err() != nil {
return nil, status.Errorf(codes.Aborted, ctx.Err().Error())
}
parent := headState.ParentRoot()
headRoot = parent[:]
headState, err = fetchState(ctx, parent)
if err != nil { if err != nil {
return nil, status.Error(codes.Internal, err.Error()) return nil, status.Errorf(codes.Internal, "Could not get historical head root: %v", err)
}
if featureconfig.Get().NewStateMgmt {
headState, err = vs.StateGen.StateByRoot(ctx, bytesutil.ToBytes32(headRoot))
if err != nil {
return nil, status.Errorf(codes.Internal, "Could not get historical head state: %v", err)
}
} else {
headState, err = vs.BeaconDB.State(ctx, bytesutil.ToBytes32(headRoot))
if err != nil {
return nil, status.Errorf(codes.Internal, "Could not get historical head state: %v", err)
}
}
} }
if headState == nil { if headState == nil {
return nil, status.Error(codes.Internal, "Failed to lookup parent state from head.") return nil, status.Error(codes.Internal, "Failed to lookup parent state from head.")
} }
}
if helpers.CurrentEpoch(headState) < helpers.SlotToEpoch(req.Slot) { if helpers.CurrentEpoch(headState) < helpers.SlotToEpoch(req.Slot) {
headState, err = state.ProcessSlots(ctx, headState, helpers.StartSlot(helpers.SlotToEpoch(req.Slot))) headState, err = state.ProcessSlots(ctx, headState, helpers.StartSlot(helpers.SlotToEpoch(req.Slot)))

View File

@ -488,6 +488,7 @@ func TestServer_GetAttestationData_HeadStateSlotGreaterThanRequestSlot(t *testin
blockRoots[1] = blockRoot[:] blockRoots[1] = blockRoot[:]
blockRoots[1*params.BeaconConfig().SlotsPerEpoch] = targetRoot[:] blockRoots[1*params.BeaconConfig().SlotsPerEpoch] = targetRoot[:]
blockRoots[2*params.BeaconConfig().SlotsPerEpoch] = justifiedRoot[:] blockRoots[2*params.BeaconConfig().SlotsPerEpoch] = justifiedRoot[:]
blockRoots[3*params.BeaconConfig().SlotsPerEpoch] = blockRoot2[:]
if err := beaconState.SetBlockRoots(blockRoots); err != nil { if err := beaconState.SetBlockRoots(blockRoots); err != nil {
t.Fatal(err) t.Fatal(err)
} }