mirror of
https://github.com/status-im/nimbus-eth2.git
synced 2025-01-19 02:52:58 +00:00
Implement all sync committee duties in the validator client (#3583)
Other changes: * logtrace can now verify sync committee messages and contributions * Many unnecessary use of pairs() have been removed for consistency * Map 40x BN response codes to BeaconNodeStatus.Incompatible in the VC
This commit is contained in:
parent
6d11ad6ce1
commit
a2ba34f686
@ -80,7 +80,7 @@ func checkMissing*(quarantine: var Quarantine): seq[FetchRecord] =
|
|||||||
quarantine.missing.del(k)
|
quarantine.missing.del(k)
|
||||||
|
|
||||||
# simple (simplistic?) exponential backoff for retries..
|
# simple (simplistic?) exponential backoff for retries..
|
||||||
for k, v in quarantine.missing.pairs():
|
for k, v in quarantine.missing:
|
||||||
if countOnes(v.tries.uint64) == 1:
|
if countOnes(v.tries.uint64) == 1:
|
||||||
result.add(FetchRecord(root: k))
|
result.add(FetchRecord(root: k))
|
||||||
|
|
||||||
@ -164,7 +164,7 @@ func addUnviable*(quarantine: var Quarantine, root: Eth2Digest) =
|
|||||||
func cleanupOrphans(quarantine: var Quarantine, finalizedSlot: Slot) =
|
func cleanupOrphans(quarantine: var Quarantine, finalizedSlot: Slot) =
|
||||||
var toDel: seq[(Eth2Digest, ValidatorSig)]
|
var toDel: seq[(Eth2Digest, ValidatorSig)]
|
||||||
|
|
||||||
for k, v in quarantine.orphans.pairs():
|
for k, v in quarantine.orphans:
|
||||||
if not isViableOrphan(finalizedSlot, v):
|
if not isViableOrphan(finalizedSlot, v):
|
||||||
toDel.add k
|
toDel.add k
|
||||||
|
|
||||||
|
@ -85,7 +85,7 @@ iterator get_attesting_indices*(epochRef: EpochRef,
|
|||||||
trace "get_attesting_indices: inconsistent aggregation and committee length"
|
trace "get_attesting_indices: inconsistent aggregation and committee length"
|
||||||
else:
|
else:
|
||||||
for index_in_committee, validator_index in get_beacon_committee(
|
for index_in_committee, validator_index in get_beacon_committee(
|
||||||
epochRef, slot, committee_index).pairs():
|
epochRef, slot, committee_index):
|
||||||
if bits[index_in_committee]:
|
if bits[index_in_committee]:
|
||||||
yield validator_index
|
yield validator_index
|
||||||
|
|
||||||
|
@ -586,7 +586,7 @@ when isMainModule:
|
|||||||
|
|
||||||
doAssert err.isOk, "compute_deltas finished with error: " & $err
|
doAssert err.isOk, "compute_deltas finished with error: " & $err
|
||||||
|
|
||||||
for i, delta in deltas.pairs:
|
for i, delta in deltas:
|
||||||
if i == 0:
|
if i == 0:
|
||||||
doAssert delta == Delta(Balance * validator_count), "The 0th root should have a delta"
|
doAssert delta == Delta(Balance * validator_count), "The 0th root should have a delta"
|
||||||
else:
|
else:
|
||||||
@ -625,7 +625,7 @@ when isMainModule:
|
|||||||
|
|
||||||
doAssert err.isOk, "compute_deltas finished with error: " & $err
|
doAssert err.isOk, "compute_deltas finished with error: " & $err
|
||||||
|
|
||||||
for i, delta in deltas.pairs:
|
for i, delta in deltas:
|
||||||
doAssert delta == Delta(Balance), "Each root should have a delta"
|
doAssert delta == Delta(Balance), "Each root should have a delta"
|
||||||
|
|
||||||
for vote in votes:
|
for vote in votes:
|
||||||
@ -663,7 +663,7 @@ when isMainModule:
|
|||||||
|
|
||||||
doAssert err.isOk, "compute_deltas finished with error: " & $err
|
doAssert err.isOk, "compute_deltas finished with error: " & $err
|
||||||
|
|
||||||
for i, delta in deltas.pairs:
|
for i, delta in deltas:
|
||||||
if i == 0:
|
if i == 0:
|
||||||
doAssert delta == -TotalDeltas, "0th root should have a negative delta"
|
doAssert delta == -TotalDeltas, "0th root should have a negative delta"
|
||||||
elif i == 1:
|
elif i == 1:
|
||||||
@ -750,7 +750,7 @@ when isMainModule:
|
|||||||
|
|
||||||
doAssert err.isOk, "compute_deltas finished with error: " & $err
|
doAssert err.isOk, "compute_deltas finished with error: " & $err
|
||||||
|
|
||||||
for i, delta in deltas.pairs:
|
for i, delta in deltas:
|
||||||
if i == 0:
|
if i == 0:
|
||||||
doAssert delta == -TotalOldDeltas, "0th root should have a negative delta"
|
doAssert delta == -TotalOldDeltas, "0th root should have a negative delta"
|
||||||
elif i == 1:
|
elif i == 1:
|
||||||
|
@ -455,7 +455,7 @@ proc scheduleContributionChecks*(
|
|||||||
proofFut = batchCrypto.withBatch("scheduleContributionAndProofChecks.selection_proof"):
|
proofFut = batchCrypto.withBatch("scheduleContributionAndProofChecks.selection_proof"):
|
||||||
sync_committee_selection_proof_set(
|
sync_committee_selection_proof_set(
|
||||||
fork, genesis_validators_root, contribution.slot,
|
fork, genesis_validators_root, contribution.slot,
|
||||||
contribution.subcommittee_index, aggregatorKey, proofSig)
|
subcommitteeIdx, aggregatorKey, proofSig)
|
||||||
contributionFut = batchCrypto.withBatch("scheduleContributionAndProofChecks.contribution"):
|
contributionFut = batchCrypto.withBatch("scheduleContributionAndProofChecks.contribution"):
|
||||||
sync_committee_message_signature_set(
|
sync_committee_message_signature_set(
|
||||||
fork, genesis_validators_root, contribution.slot,
|
fork, genesis_validators_root, contribution.slot,
|
||||||
|
@ -1319,7 +1319,7 @@ proc trimConnections(node: Eth2Node, count: int) =
|
|||||||
currentVal.count + 1
|
currentVal.count + 1
|
||||||
)
|
)
|
||||||
|
|
||||||
for peerId, gScore in gossipScores.pairs:
|
for peerId, gScore in gossipScores:
|
||||||
scores[peerId] =
|
scores[peerId] =
|
||||||
scores.getOrDefault(peerId) + (gScore.sum div gScore.count)
|
scores.getOrDefault(peerId) + (gScore.sum div gScore.count)
|
||||||
|
|
||||||
|
@ -302,8 +302,10 @@ proc installApiHandlers*(node: SigningNode) =
|
|||||||
let
|
let
|
||||||
forkInfo = request.forkInfo.get()
|
forkInfo = request.forkInfo.get()
|
||||||
msg = request.syncAggregatorSelectionData
|
msg = request.syncAggregatorSelectionData
|
||||||
|
subcommittee = SyncSubcommitteeIndex.init(msg.subcommittee_index).valueOr:
|
||||||
|
return errorResponse(Http400, InvalidSubCommitteeIndexValueError)
|
||||||
cooked = get_sync_committee_selection_proof(forkInfo.fork,
|
cooked = get_sync_committee_selection_proof(forkInfo.fork,
|
||||||
forkInfo.genesis_validators_root, msg.slot, msg.subcommittee_index,
|
forkInfo.genesis_validators_root, msg.slot, subcommittee,
|
||||||
validator.data.privateKey)
|
validator.data.privateKey)
|
||||||
signature = cooked.toValidatorSig().toHex()
|
signature = cooked.toValidatorSig().toHex()
|
||||||
signatureResponse(Http200, signature)
|
signatureResponse(Http200, signature)
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
|
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
|
||||||
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
||||||
import validator_client/[common, fallback_service, duties_service,
|
import validator_client/[common, fallback_service, duties_service,
|
||||||
attestation_service, fork_service]
|
attestation_service, fork_service, sync_committee_service]
|
||||||
|
|
||||||
proc initGenesis(vc: ValidatorClientRef): Future[RestGenesis] {.async.} =
|
proc initGenesis(vc: ValidatorClientRef): Future[RestGenesis] {.async.} =
|
||||||
info "Initializing genesis", nodes_count = len(vc.beaconNodes)
|
info "Initializing genesis", nodes_count = len(vc.beaconNodes)
|
||||||
@ -126,6 +126,7 @@ proc asyncInit(vc: ValidatorClientRef) {.async.} =
|
|||||||
vc.forkService = await ForkServiceRef.init(vc)
|
vc.forkService = await ForkServiceRef.init(vc)
|
||||||
vc.dutiesService = await DutiesServiceRef.init(vc)
|
vc.dutiesService = await DutiesServiceRef.init(vc)
|
||||||
vc.attestationService = await AttestationServiceRef.init(vc)
|
vc.attestationService = await AttestationServiceRef.init(vc)
|
||||||
|
vc.syncCommitteeService = await SyncCommitteeServiceRef.init(vc)
|
||||||
|
|
||||||
proc onSlotStart(vc: ValidatorClientRef, wallTime: BeaconTime,
|
proc onSlotStart(vc: ValidatorClientRef, wallTime: BeaconTime,
|
||||||
lastSlot: Slot) {.async.} =
|
lastSlot: Slot) {.async.} =
|
||||||
@ -159,6 +160,7 @@ proc asyncRun(vc: ValidatorClientRef) {.async.} =
|
|||||||
vc.forkService.start()
|
vc.forkService.start()
|
||||||
vc.dutiesService.start()
|
vc.dutiesService.start()
|
||||||
vc.attestationService.start()
|
vc.attestationService.start()
|
||||||
|
vc.syncCommitteeService.start()
|
||||||
|
|
||||||
await runSlotLoop(vc, vc.beaconClock.now(), onSlotStart)
|
await runSlotLoop(vc, vc.beaconClock.now(), onSlotStart)
|
||||||
|
|
||||||
|
@ -272,8 +272,7 @@ proc installBeaconApiHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
# return empty response.
|
# return empty response.
|
||||||
if len(validatorIds) == 0:
|
if len(validatorIds) == 0:
|
||||||
# There is no indices, so we going to filter all the validators.
|
# There is no indices, so we going to filter all the validators.
|
||||||
for index, validator in getStateField(state,
|
for index, validator in getStateField(state, validators):
|
||||||
validators).pairs():
|
|
||||||
let
|
let
|
||||||
balance = getStateField(state, balances).asSeq()[index]
|
balance = getStateField(state, balances).asSeq()[index]
|
||||||
status =
|
status =
|
||||||
@ -447,7 +446,7 @@ proc installBeaconApiHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
if len(validatorIds) == 0:
|
if len(validatorIds) == 0:
|
||||||
# There is no indices, so we going to return balances of all
|
# There is no indices, so we going to return balances of all
|
||||||
# known validators.
|
# known validators.
|
||||||
for index, balance in getStateField(state, balances).pairs():
|
for index, balance in getStateField(state, balances):
|
||||||
res.add(RestValidatorBalance.init(ValidatorIndex(index),
|
res.add(RestValidatorBalance.init(ValidatorIndex(index),
|
||||||
balance))
|
balance))
|
||||||
else:
|
else:
|
||||||
@ -905,7 +904,7 @@ proc installBeaconApiHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
block:
|
block:
|
||||||
var res: seq[RestAttestationsFailure]
|
var res: seq[RestAttestationsFailure]
|
||||||
await allFutures(pending)
|
await allFutures(pending)
|
||||||
for index, future in pending.pairs():
|
for index, future in pending:
|
||||||
if future.done():
|
if future.done():
|
||||||
let fres = future.read()
|
let fres = future.read()
|
||||||
if fres.isErr():
|
if fres.isErr():
|
||||||
@ -1008,7 +1007,7 @@ proc installBeaconApiHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
let failures =
|
let failures =
|
||||||
block:
|
block:
|
||||||
var res: seq[RestAttestationsFailure]
|
var res: seq[RestAttestationsFailure]
|
||||||
for index, item in results.pairs():
|
for index, item in results:
|
||||||
if item.isErr():
|
if item.isErr():
|
||||||
res.add(RestAttestationsFailure(index: uint64(index),
|
res.add(RestAttestationsFailure(index: uint64(index),
|
||||||
message: $item.error()))
|
message: $item.error()))
|
||||||
|
@ -144,7 +144,7 @@ proc installKeymanagerHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
|
|
||||||
var response: PostKeystoresResponse
|
var response: PostKeystoresResponse
|
||||||
|
|
||||||
for index, item in request.keystores.pairs():
|
for index, item in request.keystores:
|
||||||
let res = importKeystore(node.attachedValidators[], node.network.rng[],
|
let res = importKeystore(node.attachedValidators[], node.network.rng[],
|
||||||
node.config, item, request.passwords[index])
|
node.config, item, request.passwords[index])
|
||||||
if res.isErr():
|
if res.isErr():
|
||||||
@ -189,7 +189,7 @@ proc installKeymanagerHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
|
|
||||||
response.slashing_protection.metadata = nodeSPDIR.metadata
|
response.slashing_protection.metadata = nodeSPDIR.metadata
|
||||||
|
|
||||||
for index, key in keys.pairs():
|
for index, key in keys:
|
||||||
let
|
let
|
||||||
res = removeValidator(node.attachedValidators[], node.config, key,
|
res = removeValidator(node.attachedValidators[], node.config, key,
|
||||||
KeystoreKind.Local)
|
KeystoreKind.Local)
|
||||||
@ -221,7 +221,7 @@ proc installKeymanagerHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
if value.status == $KeystoreStatus.notFound:
|
if value.status == $KeystoreStatus.notFound:
|
||||||
value.status = $KeystoreStatus.notActive
|
value.status = $KeystoreStatus.notActive
|
||||||
|
|
||||||
for index, key in keys.pairs():
|
for index, key in keys:
|
||||||
response.data.add(keysAndDeleteStatus[key.blob.PubKey0x.PubKeyBytes])
|
response.data.add(keysAndDeleteStatus[key.blob.PubKey0x.PubKeyBytes])
|
||||||
|
|
||||||
return RestApiResponse.jsonResponsePlain(response)
|
return RestApiResponse.jsonResponsePlain(response)
|
||||||
@ -254,7 +254,7 @@ proc installKeymanagerHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
|
|
||||||
var response: PostKeystoresResponse
|
var response: PostKeystoresResponse
|
||||||
|
|
||||||
for index, key in keys.pairs():
|
for index, key in keys:
|
||||||
let
|
let
|
||||||
remoteInfo = RemoteSignerInfo(
|
remoteInfo = RemoteSignerInfo(
|
||||||
url: key.url,
|
url: key.url,
|
||||||
@ -322,7 +322,7 @@ proc installKeymanagerHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
|
|
||||||
var response: PostKeystoresResponse
|
var response: PostKeystoresResponse
|
||||||
|
|
||||||
for index, key in keys.pairs():
|
for index, key in keys:
|
||||||
let keystore = RemoteKeystore(
|
let keystore = RemoteKeystore(
|
||||||
version: 2'u64,
|
version: 2'u64,
|
||||||
remoteType: RemoteSignerType.Web3Signer,
|
remoteType: RemoteSignerType.Web3Signer,
|
||||||
@ -352,7 +352,7 @@ proc installKeymanagerHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
dres.get.pubkeys
|
dres.get.pubkeys
|
||||||
|
|
||||||
var response: DeleteRemoteKeystoresResponse
|
var response: DeleteRemoteKeystoresResponse
|
||||||
for index, key in keys.pairs():
|
for index, key in keys:
|
||||||
let status = node.removeValidator(key)
|
let status = node.removeValidator(key)
|
||||||
response.data.add(status)
|
response.data.add(status)
|
||||||
|
|
||||||
|
@ -241,7 +241,7 @@ func keysToIndices*(cacheTable: var Table[ValidatorPubKey, ValidatorIndex],
|
|||||||
var keyset =
|
var keyset =
|
||||||
block:
|
block:
|
||||||
var res: Table[ValidatorPubKey, int]
|
var res: Table[ValidatorPubKey, int]
|
||||||
for inputIndex, pubkey in keys.pairs():
|
for inputIndex, pubkey in keys:
|
||||||
# Try to search in cache first.
|
# Try to search in cache first.
|
||||||
cacheTable.withValue(pubkey, vindex):
|
cacheTable.withValue(pubkey, vindex):
|
||||||
if uint64(vindex[]) < totalValidatorsInState:
|
if uint64(vindex[]) < totalValidatorsInState:
|
||||||
@ -250,8 +250,7 @@ func keysToIndices*(cacheTable: var Table[ValidatorPubKey, ValidatorIndex],
|
|||||||
res[pubkey] = inputIndex
|
res[pubkey] = inputIndex
|
||||||
res
|
res
|
||||||
if len(keyset) > 0:
|
if len(keyset) > 0:
|
||||||
for validatorIndex, validator in getStateField(forkedState,
|
for validatorIndex, validator in getStateField(forkedState, validators):
|
||||||
validators).pairs():
|
|
||||||
keyset.withValue(validator.pubkey, listIndex):
|
keyset.withValue(validator.pubkey, listIndex):
|
||||||
# Store pair (pubkey, index) into cache table.
|
# Store pair (pubkey, index) into cache table.
|
||||||
cacheTable[validator.pubkey] = ValidatorIndex(validatorIndex)
|
cacheTable[validator.pubkey] = ValidatorIndex(validatorIndex)
|
||||||
|
@ -631,8 +631,6 @@ proc installValidatorApiHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
|
|
||||||
subs
|
subs
|
||||||
|
|
||||||
# TODO because we subscribe to all sync committee subnets, we don not need
|
|
||||||
# to remember which ones are requested by validator clients
|
|
||||||
return RestApiResponse.jsonMsgResponse(SyncCommitteeSubscriptionSuccess)
|
return RestApiResponse.jsonMsgResponse(SyncCommitteeSubscriptionSuccess)
|
||||||
|
|
||||||
# https://ethereum.github.io/beacon-APIs/#/Validator/produceSyncCommitteeContribution
|
# https://ethereum.github.io/beacon-APIs/#/Validator/produceSyncCommitteeContribution
|
||||||
@ -717,7 +715,7 @@ proc installValidatorApiHandlers*(router: var RestRouter, node: BeaconNode) =
|
|||||||
block:
|
block:
|
||||||
var res: seq[RestAttestationsFailure]
|
var res: seq[RestAttestationsFailure]
|
||||||
await allFutures(pending)
|
await allFutures(pending)
|
||||||
for index, future in pending.pairs():
|
for index, future in pending:
|
||||||
if future.done():
|
if future.done():
|
||||||
let fres = future.read()
|
let fres = future.read()
|
||||||
if fres.isErr():
|
if fres.isErr():
|
||||||
|
@ -230,7 +230,7 @@ proc installBeaconApiHandlers*(rpcServer: RpcServer, node: BeaconNode) {.
|
|||||||
vquery = vqres.get()
|
vquery = vqres.get()
|
||||||
|
|
||||||
if validatorIds.isNone():
|
if validatorIds.isNone():
|
||||||
for index, validator in getStateField(state, validators).pairs():
|
for index, validator in getStateField(state, validators):
|
||||||
let sres = validator.getStatus(current_epoch)
|
let sres = validator.getStatus(current_epoch)
|
||||||
if sres.isOk:
|
if sres.isOk:
|
||||||
let vstatus = sres.get()
|
let vstatus = sres.get()
|
||||||
@ -257,7 +257,7 @@ proc installBeaconApiHandlers*(rpcServer: RpcServer, node: BeaconNode) {.
|
|||||||
status: vstatus,
|
status: vstatus,
|
||||||
balance: getStateField(state, balances).asSeq()[index]))
|
balance: getStateField(state, balances).asSeq()[index]))
|
||||||
|
|
||||||
for index, validator in getStateField(state, validators).pairs():
|
for index, validator in getStateField(state, validators):
|
||||||
if validator.pubkey in vquery.keyset:
|
if validator.pubkey in vquery.keyset:
|
||||||
let sres = validator.getStatus(current_epoch)
|
let sres = validator.getStatus(current_epoch)
|
||||||
if sres.isOk:
|
if sres.isOk:
|
||||||
@ -292,7 +292,7 @@ proc installBeaconApiHandlers*(rpcServer: RpcServer, node: BeaconNode) {.
|
|||||||
else:
|
else:
|
||||||
raise newException(CatchableError, "Incorrect validator's state")
|
raise newException(CatchableError, "Incorrect validator's state")
|
||||||
else:
|
else:
|
||||||
for index, validator in getStateField(state, validators).pairs():
|
for index, validator in getStateField(state, validators):
|
||||||
if validator.pubkey in vquery.keyset:
|
if validator.pubkey in vquery.keyset:
|
||||||
let sres = validator.getStatus(current_epoch)
|
let sres = validator.getStatus(current_epoch)
|
||||||
if sres.isOk:
|
if sres.isOk:
|
||||||
@ -308,7 +308,7 @@ proc installBeaconApiHandlers*(rpcServer: RpcServer, node: BeaconNode) {.
|
|||||||
var res: seq[RpcBalance]
|
var res: seq[RpcBalance]
|
||||||
withStateForStateId(stateId):
|
withStateForStateId(stateId):
|
||||||
if validatorsId.isNone():
|
if validatorsId.isNone():
|
||||||
for index, value in getStateField(state, balances).pairs():
|
for index, value in getStateField(state, balances):
|
||||||
let balance = (index: uint64(index), balance: value)
|
let balance = (index: uint64(index), balance: value)
|
||||||
res.add(balance)
|
res.add(balance)
|
||||||
else:
|
else:
|
||||||
@ -325,7 +325,7 @@ proc installBeaconApiHandlers*(rpcServer: RpcServer, node: BeaconNode) {.
|
|||||||
balance: getStateField(state, balances).asSeq()[index])
|
balance: getStateField(state, balances).asSeq()[index])
|
||||||
res.add(balance)
|
res.add(balance)
|
||||||
|
|
||||||
for index, validator in getStateField(state, validators).pairs():
|
for index, validator in getStateField(state, validators):
|
||||||
if validator.pubkey in vquery.keyset:
|
if validator.pubkey in vquery.keyset:
|
||||||
let balance = (index: uint64(index),
|
let balance = (index: uint64(index),
|
||||||
balance: getStateField(state, balances).asSeq()[index])
|
balance: getStateField(state, balances).asSeq()[index])
|
||||||
|
@ -450,7 +450,7 @@ func get_attesting_indices*(state: ForkyBeaconState,
|
|||||||
trace "get_attesting_indices: invalid attestation data"
|
trace "get_attesting_indices: invalid attestation data"
|
||||||
else:
|
else:
|
||||||
for index_in_committee, validator_index in get_beacon_committee(
|
for index_in_committee, validator_index in get_beacon_committee(
|
||||||
state, data.slot, committee_index.get(), cache).pairs():
|
state, data.slot, committee_index.get(), cache):
|
||||||
if bits[index_in_committee]:
|
if bits[index_in_committee]:
|
||||||
res.add validator_index
|
res.add validator_index
|
||||||
|
|
||||||
|
@ -547,7 +547,8 @@ template `[]`*(arr: array[SYNC_COMMITTEE_SIZE, auto] | seq;
|
|||||||
idx: IndexInSyncCommittee): auto =
|
idx: IndexInSyncCommittee): auto =
|
||||||
arr[int idx]
|
arr[int idx]
|
||||||
|
|
||||||
makeLimitedU64(SyncSubcommitteeIndex, SYNC_COMMITTEE_SUBNET_COUNT)
|
makeLimitedU8(SyncSubcommitteeIndex, SYNC_COMMITTEE_SUBNET_COUNT)
|
||||||
|
makeLimitedU16(IndexInSyncCommittee, SYNC_COMMITTEE_SIZE)
|
||||||
|
|
||||||
func shortLog*(v: SomeBeaconBlock): auto =
|
func shortLog*(v: SomeBeaconBlock): auto =
|
||||||
(
|
(
|
||||||
|
@ -535,9 +535,10 @@ func getImmutableValidatorData*(validator: Validator): ImmutableValidatorData2 =
|
|||||||
pubkey: cookedKey.get(),
|
pubkey: cookedKey.get(),
|
||||||
withdrawal_credentials: validator.withdrawal_credentials)
|
withdrawal_credentials: validator.withdrawal_credentials)
|
||||||
|
|
||||||
template makeLimitedU64*(T: untyped, limit: uint64) =
|
template makeLimitedUInt*(T: untyped, limit: SomeUnsignedInt) =
|
||||||
# A "tigher" type is often used for T, but for the range check to be effective
|
# A "tigher" type is often used for T, but for the range check to be effective
|
||||||
# it must make sense..
|
# it must make sense..
|
||||||
|
type L = typeof limit
|
||||||
|
|
||||||
static: doAssert limit <= distinctBase(T).high()
|
static: doAssert limit <= distinctBase(T).high()
|
||||||
# Many `uint64` values in the spec have a more limited range of valid values
|
# Many `uint64` values in the spec have a more limited range of valid values
|
||||||
@ -582,6 +583,15 @@ template makeLimitedU64*(T: untyped, limit: uint64) =
|
|||||||
template toSszType(x: T): uint64 =
|
template toSszType(x: T): uint64 =
|
||||||
{.error: "Limited types should not be used with SSZ (abi differences)".}
|
{.error: "Limited types should not be used with SSZ (abi differences)".}
|
||||||
|
|
||||||
|
template makeLimitedU64*(T: untyped, limit: uint64) =
|
||||||
|
makeLimitedUInt(T, limit)
|
||||||
|
|
||||||
|
template makeLimitedU8*(T: untyped, limit: uint8) =
|
||||||
|
makeLimitedUInt(T, limit)
|
||||||
|
|
||||||
|
template makeLimitedU16*(T: type, limit: uint16) =
|
||||||
|
makeLimitedUInt(T, limit)
|
||||||
|
|
||||||
makeLimitedU64(CommitteeIndex, MAX_COMMITTEES_PER_SLOT)
|
makeLimitedU64(CommitteeIndex, MAX_COMMITTEES_PER_SLOT)
|
||||||
makeLimitedU64(SubnetId, ATTESTATION_SUBNET_COUNT)
|
makeLimitedU64(SubnetId, ATTESTATION_SUBNET_COUNT)
|
||||||
|
|
||||||
|
@ -43,7 +43,7 @@ type
|
|||||||
message*: string
|
message*: string
|
||||||
stacktraces*: Option[seq[string]]
|
stacktraces*: Option[seq[string]]
|
||||||
|
|
||||||
RestAttestationError* = object
|
RestDutyError* = object
|
||||||
code*: uint64
|
code*: uint64
|
||||||
message*: string
|
message*: string
|
||||||
failures*: seq[RestFailureItem]
|
failures*: seq[RestFailureItem]
|
||||||
@ -82,7 +82,7 @@ type
|
|||||||
GetStateV2Response |
|
GetStateV2Response |
|
||||||
GetStateForkResponse |
|
GetStateForkResponse |
|
||||||
ProduceBlockResponseV2 |
|
ProduceBlockResponseV2 |
|
||||||
RestAttestationError |
|
RestDutyError |
|
||||||
RestValidator |
|
RestValidator |
|
||||||
RestGenericError |
|
RestGenericError |
|
||||||
Web3SignerErrorResponse |
|
Web3SignerErrorResponse |
|
||||||
|
@ -543,7 +543,6 @@ type
|
|||||||
GetPoolProposerSlashingsResponse* = DataEnclosedObject[seq[ProposerSlashing]]
|
GetPoolProposerSlashingsResponse* = DataEnclosedObject[seq[ProposerSlashing]]
|
||||||
GetPoolVoluntaryExitsResponse* = DataEnclosedObject[seq[SignedVoluntaryExit]]
|
GetPoolVoluntaryExitsResponse* = DataEnclosedObject[seq[SignedVoluntaryExit]]
|
||||||
GetProposerDutiesResponse* = DataRootEnclosedObject[seq[RestProposerDuty]]
|
GetProposerDutiesResponse* = DataRootEnclosedObject[seq[RestProposerDuty]]
|
||||||
GetSyncCommitteeDutiesResponse* = DataEnclosedObject[seq[RestSyncCommitteeDuty]]
|
|
||||||
GetSpecResponse* = DataEnclosedObject[RestSpec]
|
GetSpecResponse* = DataEnclosedObject[RestSpec]
|
||||||
GetSpecVCResponse* = DataEnclosedObject[RestSpecVC]
|
GetSpecVCResponse* = DataEnclosedObject[RestSpecVC]
|
||||||
GetStateFinalityCheckpointsResponse* = DataEnclosedObject[RestBeaconStatesFinalityCheckpoints]
|
GetStateFinalityCheckpointsResponse* = DataEnclosedObject[RestBeaconStatesFinalityCheckpoints]
|
||||||
@ -552,12 +551,14 @@ type
|
|||||||
GetStateValidatorBalancesResponse* = DataEnclosedObject[seq[RestValidatorBalance]]
|
GetStateValidatorBalancesResponse* = DataEnclosedObject[seq[RestValidatorBalance]]
|
||||||
GetStateValidatorResponse* = DataEnclosedObject[RestValidator]
|
GetStateValidatorResponse* = DataEnclosedObject[RestValidator]
|
||||||
GetStateValidatorsResponse* = DataEnclosedObject[seq[RestValidator]]
|
GetStateValidatorsResponse* = DataEnclosedObject[seq[RestValidator]]
|
||||||
|
GetSyncCommitteeDutiesResponse* = DataRootEnclosedObject[seq[RestSyncCommitteeDuty]]
|
||||||
GetSyncingStatusResponse* = DataEnclosedObject[RestSyncInfo]
|
GetSyncingStatusResponse* = DataEnclosedObject[RestSyncInfo]
|
||||||
GetVersionResponse* = DataEnclosedObject[RestNodeVersion]
|
GetVersionResponse* = DataEnclosedObject[RestNodeVersion]
|
||||||
GetEpochSyncCommitteesResponse* = DataEnclosedObject[RestEpochSyncCommittee]
|
GetEpochSyncCommitteesResponse* = DataEnclosedObject[RestEpochSyncCommittee]
|
||||||
ProduceAttestationDataResponse* = DataEnclosedObject[AttestationData]
|
ProduceAttestationDataResponse* = DataEnclosedObject[AttestationData]
|
||||||
ProduceBlockResponse* = DataEnclosedObject[phase0.BeaconBlock]
|
ProduceBlockResponse* = DataEnclosedObject[phase0.BeaconBlock]
|
||||||
ProduceBlockResponseV2* = ForkedBeaconBlock
|
ProduceBlockResponseV2* = ForkedBeaconBlock
|
||||||
|
ProduceSyncCommitteeContributionResponse* = DataEnclosedObject[SyncCommitteeContribution]
|
||||||
|
|
||||||
func `==`*(a, b: RestValidatorIndex): bool =
|
func `==`*(a, b: RestValidatorIndex): bool =
|
||||||
uint64(a) == uint64(b)
|
uint64(a) == uint64(b)
|
||||||
@ -749,3 +750,54 @@ func init*(t: typedesc[Web3SignerRequest], fork: Fork,
|
|||||||
signingRoot: signingRoot,
|
signingRoot: signingRoot,
|
||||||
syncCommitteeContributionAndProof: data
|
syncCommitteeContributionAndProof: data
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func init*(t: typedesc[RestSyncCommitteeMessage],
|
||||||
|
slot: Slot,
|
||||||
|
beacon_block_root: Eth2Digest,
|
||||||
|
validator_index: uint64,
|
||||||
|
signature: ValidatorSig): RestSyncCommitteeMessage =
|
||||||
|
RestSyncCommitteeMessage(
|
||||||
|
slot: slot,
|
||||||
|
beacon_block_root: beacon_block_root,
|
||||||
|
validator_index: validator_index,
|
||||||
|
signature: signature
|
||||||
|
)
|
||||||
|
|
||||||
|
func init*(t: typedesc[RestSyncCommitteeContribution],
|
||||||
|
slot: Slot,
|
||||||
|
beacon_block_root: Eth2Digest,
|
||||||
|
subcommittee_index: uint64,
|
||||||
|
aggregation_bits: SyncCommitteeAggregationBits,
|
||||||
|
signature: ValidatorSig): RestSyncCommitteeContribution =
|
||||||
|
RestSyncCommitteeContribution(
|
||||||
|
slot: slot,
|
||||||
|
beacon_block_root: beacon_block_root,
|
||||||
|
subcommittee_index: subcommittee_index,
|
||||||
|
aggregation_bits: aggregation_bits,
|
||||||
|
signature: signature)
|
||||||
|
|
||||||
|
func init*(t: typedesc[RestContributionAndProof],
|
||||||
|
aggregator_index: uint64,
|
||||||
|
selection_proof: ValidatorSig,
|
||||||
|
contribution: SyncCommitteeContribution): RestContributionAndProof =
|
||||||
|
RestContributionAndProof(
|
||||||
|
aggregator_index: aggregator_index,
|
||||||
|
selection_proof: selection_proof,
|
||||||
|
contribution: RestSyncCommitteeContribution.init(
|
||||||
|
contribution.slot,
|
||||||
|
contribution.beacon_block_root,
|
||||||
|
contribution.subcommittee_index,
|
||||||
|
contribution.aggregation_bits,
|
||||||
|
contribution.signature
|
||||||
|
))
|
||||||
|
|
||||||
|
func init*(t: typedesc[RestSignedContributionAndProof],
|
||||||
|
message: ContributionAndProof,
|
||||||
|
signature: ValidatorSig): RestSignedContributionAndProof =
|
||||||
|
RestSignedContributionAndProof(
|
||||||
|
message: RestContributionAndProof.init(
|
||||||
|
message.aggregator_index,
|
||||||
|
message.selection_proof,
|
||||||
|
message.contribution
|
||||||
|
),
|
||||||
|
signature: signature)
|
||||||
|
@ -76,7 +76,8 @@ proc prepareSyncCommitteeSubnets*(body: seq[RestSyncCommitteeSubscription]): Res
|
|||||||
|
|
||||||
proc produceSyncCommitteeContribution*(slot: Slot,
|
proc produceSyncCommitteeContribution*(slot: Slot,
|
||||||
subcommittee_index: SyncSubcommitteeIndex,
|
subcommittee_index: SyncSubcommitteeIndex,
|
||||||
beacon_block_root: Eth2Digest): RestPlainResponse {.
|
beacon_block_root: Eth2Digest
|
||||||
|
): RestResponse[ProduceSyncCommitteeContributionResponse] {.
|
||||||
rest, endpoint: "/eth/v1/validator/sync_committee_contribution",
|
rest, endpoint: "/eth/v1/validator/sync_committee_contribution",
|
||||||
meth: MethodGet.}
|
meth: MethodGet.}
|
||||||
## https://ethereum.github.io/beacon-APIs/#/Validator/produceSyncCommitteeContribution
|
## https://ethereum.github.io/beacon-APIs/#/Validator/produceSyncCommitteeContribution
|
||||||
|
@ -307,7 +307,7 @@ func assign*(tgt: var ForkedHashedBeaconState, src: ForkedHashedBeaconState) =
|
|||||||
template getStateField*(x: ForkedHashedBeaconState, y: untyped): untyped =
|
template getStateField*(x: ForkedHashedBeaconState, y: untyped): untyped =
|
||||||
# The use of `unsafeAddr` avoids excessive copying in certain situations, e.g.,
|
# The use of `unsafeAddr` avoids excessive copying in certain situations, e.g.,
|
||||||
# ```
|
# ```
|
||||||
# for index, validator in getStateField(stateData.data, validators).pairs():
|
# for index, validator in getStateField(stateData.data, validators):
|
||||||
# ```
|
# ```
|
||||||
# Without `unsafeAddr`, the `validators` list would be copied to a temporary variable.
|
# Without `unsafeAddr`, the `validators` list would be copied to a temporary variable.
|
||||||
(case x.kind
|
(case x.kind
|
||||||
|
@ -268,18 +268,18 @@ proc verify_sync_committee_signature*(
|
|||||||
# https://github.com/ethereum/consensus-specs/blob/v1.1.10/specs/altair/validator.md#aggregation-selection
|
# https://github.com/ethereum/consensus-specs/blob/v1.1.10/specs/altair/validator.md#aggregation-selection
|
||||||
func compute_sync_committee_selection_proof_signing_root*(
|
func compute_sync_committee_selection_proof_signing_root*(
|
||||||
fork: Fork, genesis_validators_root: Eth2Digest,
|
fork: Fork, genesis_validators_root: Eth2Digest,
|
||||||
slot: Slot, subcommittee_index: uint64): Eth2Digest =
|
slot: Slot, subcommittee_index: SyncSubcommitteeIndex): Eth2Digest =
|
||||||
let
|
let
|
||||||
domain = get_domain(fork, DOMAIN_SYNC_COMMITTEE_SELECTION_PROOF,
|
domain = get_domain(fork, DOMAIN_SYNC_COMMITTEE_SELECTION_PROOF,
|
||||||
slot.epoch, genesis_validators_root)
|
slot.epoch, genesis_validators_root)
|
||||||
signing_data = SyncAggregatorSelectionData(
|
signing_data = SyncAggregatorSelectionData(
|
||||||
slot: slot,
|
slot: slot,
|
||||||
subcommittee_index: subcommittee_index)
|
subcommittee_index: uint64 subcommittee_index)
|
||||||
compute_signing_root(signing_data, domain)
|
compute_signing_root(signing_data, domain)
|
||||||
|
|
||||||
func get_sync_committee_selection_proof*(
|
func get_sync_committee_selection_proof*(
|
||||||
fork: Fork, genesis_validators_root: Eth2Digest,
|
fork: Fork, genesis_validators_root: Eth2Digest,
|
||||||
slot: Slot, subcommittee_index: uint64,
|
slot: Slot, subcommittee_index: SyncSubcommitteeIndex,
|
||||||
privkey: ValidatorPrivKey): CookedSig =
|
privkey: ValidatorPrivKey): CookedSig =
|
||||||
let signing_root = compute_sync_committee_selection_proof_signing_root(
|
let signing_root = compute_sync_committee_selection_proof_signing_root(
|
||||||
fork, genesis_validators_root, slot, subcommittee_index)
|
fork, genesis_validators_root, slot, subcommittee_index)
|
||||||
@ -288,7 +288,7 @@ func get_sync_committee_selection_proof*(
|
|||||||
|
|
||||||
proc verify_sync_committee_selection_proof*(
|
proc verify_sync_committee_selection_proof*(
|
||||||
fork: Fork, genesis_validators_root: Eth2Digest,
|
fork: Fork, genesis_validators_root: Eth2Digest,
|
||||||
slot: Slot, subcommittee_index: uint64,
|
slot: Slot, subcommittee_index: SyncSubcommitteeIndex,
|
||||||
pubkey: ValidatorPubKey | CookedPubKey, signature: SomeSig): bool =
|
pubkey: ValidatorPubKey | CookedPubKey, signature: SomeSig): bool =
|
||||||
withTrust(signature):
|
withTrust(signature):
|
||||||
let signing_root = compute_sync_committee_selection_proof_signing_root(
|
let signing_root = compute_sync_committee_selection_proof_signing_root(
|
||||||
|
@ -199,7 +199,7 @@ proc sync_committee_message_signature_set*(
|
|||||||
# See also: verify_sync_committee_selection_proof
|
# See also: verify_sync_committee_selection_proof
|
||||||
proc sync_committee_selection_proof_set*(
|
proc sync_committee_selection_proof_set*(
|
||||||
fork: Fork, genesis_validators_root: Eth2Digest,
|
fork: Fork, genesis_validators_root: Eth2Digest,
|
||||||
slot: Slot, subcommittee_index: uint64,
|
slot: Slot, subcommittee_index: SyncSubcommitteeIndex,
|
||||||
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
pubkey: CookedPubKey, signature: CookedSig): SignatureSet =
|
||||||
let signing_root = compute_sync_committee_selection_proof_signing_root(
|
let signing_root = compute_sync_committee_selection_proof_signing_root(
|
||||||
fork, genesis_validators_root, slot, subcommittee_index)
|
fork, genesis_validators_root, slot, subcommittee_index)
|
||||||
|
@ -209,7 +209,7 @@ template onceToAll*(vc: ValidatorClientRef, responseType: typedesc,
|
|||||||
except CancelledError:
|
except CancelledError:
|
||||||
ApiOperation.Interrupt
|
ApiOperation.Interrupt
|
||||||
|
|
||||||
for idx, node {.inject.} in onlineNodes.pairs():
|
for idx, node {.inject.} in onlineNodes:
|
||||||
it = node.client
|
it = node.client
|
||||||
let apiResponse {.inject.} =
|
let apiResponse {.inject.} =
|
||||||
block:
|
block:
|
||||||
@ -342,16 +342,17 @@ template firstSuccessTimeout*(vc: ValidatorClientRef, respType: typedesc,
|
|||||||
if exitNow:
|
if exitNow:
|
||||||
break
|
break
|
||||||
|
|
||||||
let offlineMask = {RestBeaconNodeStatus.Offline,
|
let unusableModeMask = {RestBeaconNodeStatus.Offline,
|
||||||
RestBeaconNodeStatus.NotSynced,
|
RestBeaconNodeStatus.NotSynced,
|
||||||
RestBeaconNodeStatus.Uninitalized}
|
RestBeaconNodeStatus.Uninitalized,
|
||||||
let offlineNodes = vc.beaconNodes.filterIt(it.status in offlineMask)
|
RestBeaconNodeStatus.Incompatible}
|
||||||
let onlineNodesCount = len(vc.beaconNodes) - len(offlineNodes)
|
let unusableNodes = vc.beaconNodes.filterIt(it.status in unusableModeMask)
|
||||||
|
let onlineNodesCount = len(vc.beaconNodes) - len(unusableNodes)
|
||||||
|
|
||||||
warn "No working beacon nodes available, refreshing nodes status",
|
warn "No working beacon nodes available, refreshing nodes status",
|
||||||
online_nodes = onlineNodesCount, offline_nodes = len(offlineNodes)
|
online_nodes = onlineNodesCount, unusable_nodes = len(unusableNodes)
|
||||||
|
|
||||||
var checkFut = vc.checkNodes(offlineMask)
|
var checkFut = vc.checkNodes(unusableModeMask)
|
||||||
|
|
||||||
let checkOp =
|
let checkOp =
|
||||||
block:
|
block:
|
||||||
@ -410,8 +411,7 @@ template firstSuccessTimeout*(vc: ValidatorClientRef, respType: typedesc,
|
|||||||
break
|
break
|
||||||
|
|
||||||
proc getProposerDuties*(vc: ValidatorClientRef,
|
proc getProposerDuties*(vc: ValidatorClientRef,
|
||||||
epoch: Epoch): Future[GetProposerDutiesResponse] {.
|
epoch: Epoch): Future[GetProposerDutiesResponse] {.async.} =
|
||||||
async.} =
|
|
||||||
logScope: request = "getProposerDuties"
|
logScope: request = "getProposerDuties"
|
||||||
vc.firstSuccessTimeout(RestResponse[GetProposerDutiesResponse], SlotDuration,
|
vc.firstSuccessTimeout(RestResponse[GetProposerDutiesResponse], SlotDuration,
|
||||||
getProposerDuties(it, epoch)):
|
getProposerDuties(it, epoch)):
|
||||||
@ -428,7 +428,7 @@ proc getProposerDuties*(vc: ValidatorClientRef,
|
|||||||
of 400:
|
of 400:
|
||||||
debug "Received invalid request response",
|
debug "Received invalid request response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Incompatible
|
||||||
of 500:
|
of 500:
|
||||||
debug "Received internal error response",
|
debug "Received internal error response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
@ -444,9 +444,10 @@ proc getProposerDuties*(vc: ValidatorClientRef,
|
|||||||
|
|
||||||
raise newException(ValidatorApiError, "Unable to retrieve proposer duties")
|
raise newException(ValidatorApiError, "Unable to retrieve proposer duties")
|
||||||
|
|
||||||
proc getAttesterDuties*(vc: ValidatorClientRef, epoch: Epoch,
|
proc getAttesterDuties*(
|
||||||
validators: seq[ValidatorIndex]
|
vc: ValidatorClientRef,
|
||||||
): Future[GetAttesterDutiesResponse] {.async.} =
|
epoch: Epoch,
|
||||||
|
validators: seq[ValidatorIndex]): Future[GetAttesterDutiesResponse] {.async.} =
|
||||||
logScope: request = "getAttesterDuties"
|
logScope: request = "getAttesterDuties"
|
||||||
vc.firstSuccessTimeout(RestResponse[GetAttesterDutiesResponse], SlotDuration,
|
vc.firstSuccessTimeout(RestResponse[GetAttesterDutiesResponse], SlotDuration,
|
||||||
getAttesterDuties(it, epoch, validators)):
|
getAttesterDuties(it, epoch, validators)):
|
||||||
@ -463,7 +464,7 @@ proc getAttesterDuties*(vc: ValidatorClientRef, epoch: Epoch,
|
|||||||
of 400:
|
of 400:
|
||||||
debug "Received invalid request response",
|
debug "Received invalid request response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Incompatible
|
||||||
of 500:
|
of 500:
|
||||||
debug "Received internal error response",
|
debug "Received internal error response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
@ -479,6 +480,42 @@ proc getAttesterDuties*(vc: ValidatorClientRef, epoch: Epoch,
|
|||||||
|
|
||||||
raise newException(ValidatorApiError, "Unable to retrieve attester duties")
|
raise newException(ValidatorApiError, "Unable to retrieve attester duties")
|
||||||
|
|
||||||
|
proc getSyncCommitteeDuties*(
|
||||||
|
vc: ValidatorClientRef,
|
||||||
|
epoch: Epoch,
|
||||||
|
validators: seq[ValidatorIndex]): Future[GetSyncCommitteeDutiesResponse] {.async.} =
|
||||||
|
logScope: request = "getSyncCommitteeDuties"
|
||||||
|
vc.firstSuccessTimeout(RestResponse[GetSyncCommitteeDutiesResponse], SlotDuration,
|
||||||
|
getSyncCommitteeDuties(it, epoch, validators)):
|
||||||
|
if apiResponse.isErr():
|
||||||
|
debug "Unable to retrieve sync committee duties", endpoint = node,
|
||||||
|
error = apiResponse.error()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
else:
|
||||||
|
let response = apiResponse.get()
|
||||||
|
case response.status
|
||||||
|
of 200:
|
||||||
|
debug "Received successful response", endpoint = node
|
||||||
|
return response.data
|
||||||
|
of 400:
|
||||||
|
debug "Received invalid request response",
|
||||||
|
response_code = response.status, endpoint = node
|
||||||
|
RestBeaconNodeStatus.Incompatible
|
||||||
|
of 500:
|
||||||
|
debug "Received internal error response",
|
||||||
|
response_code = response.status, endpoint = node
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
of 503:
|
||||||
|
debug "Received not synced error response",
|
||||||
|
response_code = response.status, endpoint = node
|
||||||
|
RestBeaconNodeStatus.NotSynced
|
||||||
|
else:
|
||||||
|
debug "Received unexpected error response",
|
||||||
|
response_code = response.status, endpoint = node
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
|
||||||
|
raise newException(ValidatorApiError, "Unable to retrieve sync committee duties")
|
||||||
|
|
||||||
proc getForkSchedule*(vc: ValidatorClientRef): Future[seq[Fork]] {.async.} =
|
proc getForkSchedule*(vc: ValidatorClientRef): Future[seq[Fork]] {.async.} =
|
||||||
logScope: request = "getForkSchedule"
|
logScope: request = "getForkSchedule"
|
||||||
vc.firstSuccessTimeout(RestResponse[GetForkScheduleResponse], SlotDuration,
|
vc.firstSuccessTimeout(RestResponse[GetForkScheduleResponse], SlotDuration,
|
||||||
@ -521,7 +558,7 @@ proc getHeadStateFork*(vc: ValidatorClientRef): Future[Fork] {.async.} =
|
|||||||
of 400, 404:
|
of 400, 404:
|
||||||
debug "Received invalid request response",
|
debug "Received invalid request response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Incompatible
|
||||||
of 500:
|
of 500:
|
||||||
debug "Received internal error response",
|
debug "Received internal error response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
@ -533,9 +570,39 @@ proc getHeadStateFork*(vc: ValidatorClientRef): Future[Fork] {.async.} =
|
|||||||
|
|
||||||
raise newException(ValidatorApiError, "Unable to retrieve head state's fork")
|
raise newException(ValidatorApiError, "Unable to retrieve head state's fork")
|
||||||
|
|
||||||
proc getValidators*(vc: ValidatorClientRef,
|
proc getHeadBlockRoot*(vc: ValidatorClientRef): Future[RestRoot] {.async.} =
|
||||||
id: seq[ValidatorIdent]): Future[seq[RestValidator]] {.
|
logScope: request = "getHeadBlockRoot"
|
||||||
async.} =
|
let blockIdent = BlockIdent.init(BlockIdentType.Head)
|
||||||
|
vc.firstSuccessTimeout(RestResponse[GetBlockRootResponse], SlotDuration,
|
||||||
|
getBlockRoot(it, blockIdent)):
|
||||||
|
if apiResponse.isErr():
|
||||||
|
debug "Unable to retrieve head block's root", endpoint = node,
|
||||||
|
error = apiResponse.error()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
else:
|
||||||
|
let response = apiResponse.get()
|
||||||
|
case response.status
|
||||||
|
of 200:
|
||||||
|
debug "Received successful response", endpoint = node
|
||||||
|
return response.data.data
|
||||||
|
of 400, 404:
|
||||||
|
debug "Received invalid request response",
|
||||||
|
response_code = response.status, endpoint = node
|
||||||
|
RestBeaconNodeStatus.Incompatible
|
||||||
|
of 500:
|
||||||
|
debug "Received internal error response",
|
||||||
|
response_code = response.status, endpoint = node
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
else:
|
||||||
|
debug "Received unexpected error response",
|
||||||
|
response_code = response.status, endpoint = node
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
|
||||||
|
raise newException(ValidatorApiError, "Unable to retrieve head block's root")
|
||||||
|
|
||||||
|
proc getValidators*(
|
||||||
|
vc: ValidatorClientRef,
|
||||||
|
id: seq[ValidatorIdent]): Future[seq[RestValidator]] {.async.} =
|
||||||
logScope: request = "getStateValidators"
|
logScope: request = "getStateValidators"
|
||||||
let stateIdent = StateIdent.init(StateIdentType.Head)
|
let stateIdent = StateIdent.init(StateIdentType.Head)
|
||||||
vc.firstSuccessTimeout(RestResponse[GetStateValidatorsResponse], SlotDuration,
|
vc.firstSuccessTimeout(RestResponse[GetStateValidatorsResponse], SlotDuration,
|
||||||
@ -553,7 +620,7 @@ proc getValidators*(vc: ValidatorClientRef,
|
|||||||
of 400, 404:
|
of 400, 404:
|
||||||
debug "Received invalid request response",
|
debug "Received invalid request response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Incompatible
|
||||||
of 500:
|
of 500:
|
||||||
debug "Received internal error response",
|
debug "Received internal error response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
@ -566,9 +633,10 @@ proc getValidators*(vc: ValidatorClientRef,
|
|||||||
raise newException(ValidatorApiError,
|
raise newException(ValidatorApiError,
|
||||||
"Unable to retrieve head state's validator information")
|
"Unable to retrieve head state's validator information")
|
||||||
|
|
||||||
proc produceAttestationData*(vc: ValidatorClientRef, slot: Slot,
|
proc produceAttestationData*(
|
||||||
committee_index: CommitteeIndex
|
vc: ValidatorClientRef,
|
||||||
): Future[AttestationData] {.async.} =
|
slot: Slot,
|
||||||
|
committee_index: CommitteeIndex): Future[AttestationData] {.async.} =
|
||||||
logScope: request = "produceAttestationData"
|
logScope: request = "produceAttestationData"
|
||||||
vc.firstSuccessTimeout(RestResponse[ProduceAttestationDataResponse],
|
vc.firstSuccessTimeout(RestResponse[ProduceAttestationDataResponse],
|
||||||
OneThirdDuration,
|
OneThirdDuration,
|
||||||
@ -586,7 +654,7 @@ proc produceAttestationData*(vc: ValidatorClientRef, slot: Slot,
|
|||||||
of 400:
|
of 400:
|
||||||
debug "Received invalid request response",
|
debug "Received invalid request response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Incompatible
|
||||||
of 500:
|
of 500:
|
||||||
debug "Received internal error response",
|
debug "Received internal error response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
@ -602,8 +670,8 @@ proc produceAttestationData*(vc: ValidatorClientRef, slot: Slot,
|
|||||||
|
|
||||||
raise newException(ValidatorApiError, "Unable to retrieve attestation data")
|
raise newException(ValidatorApiError, "Unable to retrieve attestation data")
|
||||||
|
|
||||||
proc getAttestationErrorMessage(response: RestPlainResponse): string =
|
proc getDutyErrorMessage(response: RestPlainResponse): string =
|
||||||
let res = decodeBytes(RestAttestationError, response.data,
|
let res = decodeBytes(RestDutyError, response.data,
|
||||||
response.contentType)
|
response.contentType)
|
||||||
if res.isOk():
|
if res.isOk():
|
||||||
let errorObj = res.get()
|
let errorObj = res.get()
|
||||||
@ -626,8 +694,7 @@ proc getGenericErrorMessage(response: RestPlainResponse): string =
|
|||||||
"Unable to decode error response: [" & $res.error() & "]"
|
"Unable to decode error response: [" & $res.error() & "]"
|
||||||
|
|
||||||
proc submitPoolAttestations*(vc: ValidatorClientRef,
|
proc submitPoolAttestations*(vc: ValidatorClientRef,
|
||||||
data: seq[Attestation]): Future[bool] {.
|
data: seq[Attestation]): Future[bool] {.async.} =
|
||||||
async.} =
|
|
||||||
logScope: request = "submitPoolAttestations"
|
logScope: request = "submitPoolAttestations"
|
||||||
vc.firstSuccessTimeout(RestPlainResponse, SlotDuration,
|
vc.firstSuccessTimeout(RestPlainResponse, SlotDuration,
|
||||||
submitPoolAttestations(it, data)):
|
submitPoolAttestations(it, data)):
|
||||||
@ -644,24 +711,62 @@ proc submitPoolAttestations*(vc: ValidatorClientRef,
|
|||||||
of 400:
|
of 400:
|
||||||
debug "Received invalid request response",
|
debug "Received invalid request response",
|
||||||
response_code = response.status, endpoint = node,
|
response_code = response.status, endpoint = node,
|
||||||
response_error = response.getAttestationErrorMessage()
|
response_error = response.getDutyErrorMessage()
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Incompatible
|
||||||
of 500:
|
of 500:
|
||||||
debug "Received internal error response",
|
debug "Received internal error response",
|
||||||
response_code = response.status, endpoint = node,
|
response_code = response.status, endpoint = node,
|
||||||
response_error = response.getAttestationErrorMessage()
|
response_error = response.getDutyErrorMessage()
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Offline
|
||||||
else:
|
else:
|
||||||
debug "Received unexpected error response",
|
debug "Received unexpected error response",
|
||||||
response_code = response.status, endpoint = node,
|
response_code = response.status, endpoint = node,
|
||||||
response_error = response.getAttestationErrorMessage()
|
response_error = response.getDutyErrorMessage()
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Offline
|
||||||
|
|
||||||
raise newException(ValidatorApiError, "Unable to submit attestation")
|
raise newException(ValidatorApiError, "Unable to submit attestation")
|
||||||
|
|
||||||
|
proc submitPoolSyncCommitteeSignature*(vc: ValidatorClientRef,
|
||||||
|
data: SyncCommitteeMessage): Future[bool] {.async.} =
|
||||||
|
logScope: request = "submitPoolSyncCommitteeSignatures"
|
||||||
|
let restData = RestSyncCommitteeMessage.init(
|
||||||
|
data.slot,
|
||||||
|
data.beacon_block_root,
|
||||||
|
data.validator_index,
|
||||||
|
data.signature
|
||||||
|
)
|
||||||
|
vc.firstSuccessTimeout(RestPlainResponse, SlotDuration,
|
||||||
|
submitPoolSyncCommitteeSignatures(it, @[restData])):
|
||||||
|
if apiResponse.isErr():
|
||||||
|
debug "Unable to submit sync committee message", endpoint = node,
|
||||||
|
error = apiResponse.error()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
else:
|
||||||
|
let response = apiResponse.get()
|
||||||
|
case response.status
|
||||||
|
of 200:
|
||||||
|
debug "Sync committee message was successfully published", endpoint = node
|
||||||
|
return true
|
||||||
|
of 400:
|
||||||
|
debug "Received invalid request response",
|
||||||
|
response_code = response.status, endpoint = node,
|
||||||
|
response_error = response.getDutyErrorMessage()
|
||||||
|
RestBeaconNodeStatus.Incompatible
|
||||||
|
of 500:
|
||||||
|
debug "Received internal error response",
|
||||||
|
response_code = response.status, endpoint = node,
|
||||||
|
response_error = response.getDutyErrorMessage()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
else:
|
||||||
|
debug "Received unexpected error response",
|
||||||
|
response_code = response.status, endpoint = node,
|
||||||
|
response_error = response.getDutyErrorMessage()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
|
||||||
|
raise newException(ValidatorApiError, "Unable to submit sync committee message")
|
||||||
|
|
||||||
proc getAggregatedAttestation*(vc: ValidatorClientRef, slot: Slot,
|
proc getAggregatedAttestation*(vc: ValidatorClientRef, slot: Slot,
|
||||||
root: Eth2Digest): Future[Attestation] {.
|
root: Eth2Digest): Future[Attestation] {.async.} =
|
||||||
async.} =
|
|
||||||
logScope: request = "getAggregatedAttestation"
|
logScope: request = "getAggregatedAttestation"
|
||||||
vc.firstSuccessTimeout(RestResponse[GetAggregatedAttestationResponse],
|
vc.firstSuccessTimeout(RestResponse[GetAggregatedAttestationResponse],
|
||||||
OneThirdDuration,
|
OneThirdDuration,
|
||||||
@ -679,7 +784,7 @@ proc getAggregatedAttestation*(vc: ValidatorClientRef, slot: Slot,
|
|||||||
of 400:
|
of 400:
|
||||||
debug "Received invalid request response",
|
debug "Received invalid request response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Incompatible
|
||||||
of 500:
|
of 500:
|
||||||
debug "Received internal error response",
|
debug "Received internal error response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
@ -692,9 +797,48 @@ proc getAggregatedAttestation*(vc: ValidatorClientRef, slot: Slot,
|
|||||||
raise newException(ValidatorApiError,
|
raise newException(ValidatorApiError,
|
||||||
"Unable to retrieve aggregated attestation data")
|
"Unable to retrieve aggregated attestation data")
|
||||||
|
|
||||||
proc publishAggregateAndProofs*(vc: ValidatorClientRef,
|
proc produceSyncCommitteeContribution*(
|
||||||
data: seq[SignedAggregateAndProof]): Future[bool] {.
|
vc: ValidatorClientRef,
|
||||||
async.} =
|
slot: Slot,
|
||||||
|
subcommitteeIndex: SyncSubcommitteeIndex,
|
||||||
|
root: Eth2Digest): Future[SyncCommitteeContribution] {.async.} =
|
||||||
|
logScope: request = "produceSyncCommitteeContribution"
|
||||||
|
vc.firstSuccessTimeout(RestResponse[ProduceSyncCommitteeContributionResponse],
|
||||||
|
OneThirdDuration,
|
||||||
|
produceSyncCommitteeContribution(it,
|
||||||
|
slot,
|
||||||
|
subcommitteeIndex,
|
||||||
|
root)):
|
||||||
|
if apiResponse.isErr():
|
||||||
|
debug "Unable to retrieve sync committee contribution data",
|
||||||
|
endpoint = node,
|
||||||
|
error = apiResponse.error()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
else:
|
||||||
|
let response = apiResponse.get()
|
||||||
|
case response.status:
|
||||||
|
of 200:
|
||||||
|
debug "Received successful response", endpoint = node
|
||||||
|
return response.data.data
|
||||||
|
of 400:
|
||||||
|
debug "Received invalid request response",
|
||||||
|
response_code = response.status, endpoint = node
|
||||||
|
RestBeaconNodeStatus.Incompatible
|
||||||
|
of 500:
|
||||||
|
debug "Received internal error response",
|
||||||
|
response_code = response.status, endpoint = node
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
else:
|
||||||
|
debug "Received unexpected error response",
|
||||||
|
response_code = response.status, endpoint = node
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
|
||||||
|
raise newException(ValidatorApiError,
|
||||||
|
"Unable to retrieve sync committee contribution data")
|
||||||
|
|
||||||
|
proc publishAggregateAndProofs*(
|
||||||
|
vc: ValidatorClientRef,
|
||||||
|
data: seq[SignedAggregateAndProof]): Future[bool] {.async.} =
|
||||||
logScope: request = "publishAggregateAndProofs"
|
logScope: request = "publishAggregateAndProofs"
|
||||||
vc.firstSuccessTimeout(RestPlainResponse, SlotDuration,
|
vc.firstSuccessTimeout(RestPlainResponse, SlotDuration,
|
||||||
publishAggregateAndProofs(it, data)):
|
publishAggregateAndProofs(it, data)):
|
||||||
@ -712,7 +856,7 @@ proc publishAggregateAndProofs*(vc: ValidatorClientRef,
|
|||||||
debug "Received invalid request response",
|
debug "Received invalid request response",
|
||||||
response_code = response.status, endpoint = node,
|
response_code = response.status, endpoint = node,
|
||||||
response_error = response.getGenericErrorMessage()
|
response_error = response.getGenericErrorMessage()
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Incompatible
|
||||||
of 500:
|
of 500:
|
||||||
debug "Received internal error response",
|
debug "Received internal error response",
|
||||||
response_code = response.status, endpoint = node,
|
response_code = response.status, endpoint = node,
|
||||||
@ -727,10 +871,46 @@ proc publishAggregateAndProofs*(vc: ValidatorClientRef,
|
|||||||
raise newException(ValidatorApiError,
|
raise newException(ValidatorApiError,
|
||||||
"Unable to publish aggregate and proofs")
|
"Unable to publish aggregate and proofs")
|
||||||
|
|
||||||
proc produceBlockV2*(vc: ValidatorClientRef, slot: Slot,
|
proc publishContributionAndProofs*(
|
||||||
randao_reveal: ValidatorSig,
|
vc: ValidatorClientRef,
|
||||||
graffiti: GraffitiBytes): Future[ProduceBlockResponseV2] {.
|
data: seq[RestSignedContributionAndProof]): Future[bool] {.async.} =
|
||||||
async.} =
|
logScope: request = "publishContributionAndProofs"
|
||||||
|
vc.firstSuccessTimeout(RestPlainResponse, SlotDuration,
|
||||||
|
publishContributionAndProofs(it, data)):
|
||||||
|
if apiResponse.isErr():
|
||||||
|
debug "Unable to publish contribution and proofs", endpoint = node,
|
||||||
|
error = apiResponse.error()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
else:
|
||||||
|
let response = apiResponse.get()
|
||||||
|
case response.status:
|
||||||
|
of 200:
|
||||||
|
debug "Contribution and proofs were successfully published", endpoint = node
|
||||||
|
return true
|
||||||
|
of 400:
|
||||||
|
debug "Received invalid request response",
|
||||||
|
response_code = response.status, endpoint = node,
|
||||||
|
response_error = response.getGenericErrorMessage()
|
||||||
|
RestBeaconNodeStatus.Incompatible
|
||||||
|
of 500:
|
||||||
|
debug "Received internal error response",
|
||||||
|
response_code = response.status, endpoint = node,
|
||||||
|
response_error = response.getGenericErrorMessage()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
else:
|
||||||
|
debug "Received unexpected error response",
|
||||||
|
response_code = response.status, endpoint = node,
|
||||||
|
response_error = response.getGenericErrorMessage()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
|
||||||
|
raise newException(ValidatorApiError,
|
||||||
|
"Unable to publish contribution and proofs")
|
||||||
|
|
||||||
|
proc produceBlockV2*(
|
||||||
|
vc: ValidatorClientRef,
|
||||||
|
slot: Slot,
|
||||||
|
randao_reveal: ValidatorSig,
|
||||||
|
graffiti: GraffitiBytes): Future[ProduceBlockResponseV2] {.async.} =
|
||||||
logScope: request = "produceBlockV2"
|
logScope: request = "produceBlockV2"
|
||||||
vc.firstSuccessTimeout(RestResponse[ProduceBlockResponseV2],
|
vc.firstSuccessTimeout(RestResponse[ProduceBlockResponseV2],
|
||||||
SlotDuration,
|
SlotDuration,
|
||||||
@ -748,7 +928,7 @@ proc produceBlockV2*(vc: ValidatorClientRef, slot: Slot,
|
|||||||
of 400:
|
of 400:
|
||||||
debug "Received invalid request response",
|
debug "Received invalid request response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Incompatible
|
||||||
of 500:
|
of 500:
|
||||||
debug "Received internal error response",
|
debug "Received internal error response",
|
||||||
response_code = response.status, endpoint = node
|
response_code = response.status, endpoint = node
|
||||||
@ -794,7 +974,7 @@ proc publishBlock*(vc: ValidatorClientRef,
|
|||||||
debug "Received invalid request response",
|
debug "Received invalid request response",
|
||||||
response_code = response.status, endpoint = node,
|
response_code = response.status, endpoint = node,
|
||||||
response_error = response.getGenericErrorMessage()
|
response_error = response.getGenericErrorMessage()
|
||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Incompatible
|
||||||
of 500:
|
of 500:
|
||||||
debug "Received internal error response",
|
debug "Received internal error response",
|
||||||
response_code = response.status, endpoint = node,
|
response_code = response.status, endpoint = node,
|
||||||
@ -813,9 +993,9 @@ proc publishBlock*(vc: ValidatorClientRef,
|
|||||||
|
|
||||||
raise newException(ValidatorApiError, "Unable to publish block")
|
raise newException(ValidatorApiError, "Unable to publish block")
|
||||||
|
|
||||||
proc prepareBeaconCommitteeSubnet*(vc: ValidatorClientRef,
|
proc prepareBeaconCommitteeSubnet*(
|
||||||
data: seq[RestCommitteeSubscription]
|
vc: ValidatorClientRef,
|
||||||
): Future[bool] {.async.} =
|
data: seq[RestCommitteeSubscription]): Future[bool] {.async.} =
|
||||||
logScope: request = "prepareBeaconCommitteeSubnet"
|
logScope: request = "prepareBeaconCommitteeSubnet"
|
||||||
vc.firstSuccessTimeout(RestPlainResponse, OneThirdDuration,
|
vc.firstSuccessTimeout(RestPlainResponse, OneThirdDuration,
|
||||||
prepareBeaconCommitteeSubnet(it, data)):
|
prepareBeaconCommitteeSubnet(it, data)):
|
||||||
@ -851,3 +1031,44 @@ proc prepareBeaconCommitteeSubnet*(vc: ValidatorClientRef,
|
|||||||
RestBeaconNodeStatus.Offline
|
RestBeaconNodeStatus.Offline
|
||||||
|
|
||||||
raise newException(ValidatorApiError, "Unable to prepare committee subnet")
|
raise newException(ValidatorApiError, "Unable to prepare committee subnet")
|
||||||
|
|
||||||
|
proc prepareSyncCommitteeSubnets*(
|
||||||
|
vc: ValidatorClientRef,
|
||||||
|
data: seq[RestSyncCommitteeSubscription]): Future[bool] {.async.} =
|
||||||
|
logScope: request = "prepareSyncCommitteeSubnet"
|
||||||
|
vc.firstSuccessTimeout(RestPlainResponse, OneThirdDuration,
|
||||||
|
prepareSyncCommitteeSubnets(it, data)):
|
||||||
|
if apiResponse.isErr():
|
||||||
|
debug "Unable to prepare sync committee subnet", endpoint = node,
|
||||||
|
error = apiResponse.error()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
else:
|
||||||
|
let response = apiResponse.get()
|
||||||
|
case response.status
|
||||||
|
of 200:
|
||||||
|
debug "Sync committee subnet was successfully prepared",
|
||||||
|
endpoint = node
|
||||||
|
return true
|
||||||
|
of 400:
|
||||||
|
debug "Received invalid request response",
|
||||||
|
response_code = response.status, endpoint = node,
|
||||||
|
response_error = response.getGenericErrorMessage()
|
||||||
|
return false
|
||||||
|
of 500:
|
||||||
|
debug "Received internal error response",
|
||||||
|
response_code = response.status, endpoint = node,
|
||||||
|
response_error = response.getGenericErrorMessage()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
of 503:
|
||||||
|
debug "Received not synced error response",
|
||||||
|
response_code = response.status, endpoint = node,
|
||||||
|
response_error = response.getGenericErrorMessage()
|
||||||
|
RestBeaconNodeStatus.NotSynced
|
||||||
|
else:
|
||||||
|
debug "Received unexpected error response",
|
||||||
|
response_code = response.status, endpoint = node,
|
||||||
|
response_error = response.getGenericErrorMessage()
|
||||||
|
RestBeaconNodeStatus.Offline
|
||||||
|
|
||||||
|
raise newException(ValidatorApiError,
|
||||||
|
"Unable to prepare sync committee subnet")
|
||||||
|
@ -353,7 +353,7 @@ proc spawnAttestationTasks(service: AttestationServiceRef,
|
|||||||
for item in attesters:
|
for item in attesters:
|
||||||
res.mgetOrPut(item.data.committee_index, default).add(item)
|
res.mgetOrPut(item.data.committee_index, default).add(item)
|
||||||
res
|
res
|
||||||
for index, duties in dutiesByCommittee.pairs():
|
for index, duties in dutiesByCommittee:
|
||||||
if len(duties) > 0:
|
if len(duties) > 0:
|
||||||
asyncSpawn service.publishAttestationsAndAggregates(slot, index, duties)
|
asyncSpawn service.publishAttestationsAndAggregates(slot, index, duties)
|
||||||
|
|
||||||
|
@ -10,7 +10,8 @@ import
|
|||||||
validator],
|
validator],
|
||||||
../spec/eth2_apis/[eth2_rest_serialization, rest_beacon_client],
|
../spec/eth2_apis/[eth2_rest_serialization, rest_beacon_client],
|
||||||
../validators/[keystore_management, validator_pool, slashing_protection],
|
../validators/[keystore_management, validator_pool, slashing_protection],
|
||||||
".."/[conf, beacon_clock, version, nimbus_binary_common]
|
".."/[conf, beacon_clock, version, nimbus_binary_common],
|
||||||
|
../spec/eth2_apis/eth2_rest_serialization
|
||||||
|
|
||||||
export os, sets, sequtils, sequtils, chronos, presto, chronicles, confutils,
|
export os, sets, sequtils, sequtils, chronos, presto, chronicles, confutils,
|
||||||
nimbus_binary_common, version, conf, options, tables, results, base10,
|
nimbus_binary_common, version, conf, options, tables, results, base10,
|
||||||
@ -51,12 +52,28 @@ type
|
|||||||
|
|
||||||
BlockServiceRef* = ref object of ClientServiceRef
|
BlockServiceRef* = ref object of ClientServiceRef
|
||||||
|
|
||||||
|
SyncCommitteeServiceRef* = ref object of ClientServiceRef
|
||||||
|
|
||||||
DutyAndProof* = object
|
DutyAndProof* = object
|
||||||
epoch*: Epoch
|
epoch*: Epoch
|
||||||
dependentRoot*: Eth2Digest
|
dependentRoot*: Eth2Digest
|
||||||
data*: RestAttesterDuty
|
data*: RestAttesterDuty
|
||||||
slotSig*: Option[ValidatorSig]
|
slotSig*: Option[ValidatorSig]
|
||||||
|
|
||||||
|
SyncCommitteeDuty* = object
|
||||||
|
pubkey*: ValidatorPubKey
|
||||||
|
validator_index*: ValidatorIndex
|
||||||
|
validator_sync_committee_index*: IndexInSyncCommittee
|
||||||
|
|
||||||
|
SyncDutyAndProof* = object
|
||||||
|
epoch*: Epoch
|
||||||
|
data*: SyncCommitteeDuty
|
||||||
|
slotSig*: Option[ValidatorSig]
|
||||||
|
|
||||||
|
SyncCommitteeSubscriptionInfo* = object
|
||||||
|
validator_index*: ValidatorIndex
|
||||||
|
validator_sync_committee_indices*: seq[IndexInSyncCommittee]
|
||||||
|
|
||||||
ProposerTask* = object
|
ProposerTask* = object
|
||||||
duty*: RestProposerDuty
|
duty*: RestProposerDuty
|
||||||
future*: Future[void]
|
future*: Future[void]
|
||||||
@ -78,12 +95,16 @@ type
|
|||||||
EpochDuties* = object
|
EpochDuties* = object
|
||||||
duties*: Table[Epoch, DutyAndProof]
|
duties*: Table[Epoch, DutyAndProof]
|
||||||
|
|
||||||
|
EpochSyncDuties* = object
|
||||||
|
duties*: Table[Epoch, SyncDutyAndProof]
|
||||||
|
|
||||||
RestBeaconNodeStatus* {.pure.} = enum
|
RestBeaconNodeStatus* {.pure.} = enum
|
||||||
Uninitalized, Offline, Incompatible, NotSynced, Online
|
Uninitalized, Offline, Incompatible, NotSynced, Online
|
||||||
|
|
||||||
BeaconNodeServerRef* = ref BeaconNodeServer
|
BeaconNodeServerRef* = ref BeaconNodeServer
|
||||||
|
|
||||||
AttesterMap* = Table[ValidatorPubKey, EpochDuties]
|
AttesterMap* = Table[ValidatorPubKey, EpochDuties]
|
||||||
|
SyncCommitteeDutiesMap* = Table[ValidatorPubKey, EpochSyncDuties]
|
||||||
ProposerMap* = Table[Epoch, ProposedData]
|
ProposerMap* = Table[Epoch, ProposedData]
|
||||||
|
|
||||||
ValidatorClient* = object
|
ValidatorClient* = object
|
||||||
@ -96,6 +117,7 @@ type
|
|||||||
dutiesService*: DutiesServiceRef
|
dutiesService*: DutiesServiceRef
|
||||||
attestationService*: AttestationServiceRef
|
attestationService*: AttestationServiceRef
|
||||||
blockService*: BlockServiceRef
|
blockService*: BlockServiceRef
|
||||||
|
syncCommitteeService*: SyncCommitteeServiceRef
|
||||||
runSlotLoop*: Future[void]
|
runSlotLoop*: Future[void]
|
||||||
beaconClock*: BeaconClock
|
beaconClock*: BeaconClock
|
||||||
attachedValidators*: ValidatorPool
|
attachedValidators*: ValidatorPool
|
||||||
@ -103,6 +125,7 @@ type
|
|||||||
forksAvailable*: AsyncEvent
|
forksAvailable*: AsyncEvent
|
||||||
attesters*: AttesterMap
|
attesters*: AttesterMap
|
||||||
proposers*: ProposerMap
|
proposers*: ProposerMap
|
||||||
|
syncCommitteeDuties*: SyncCommitteeDutiesMap
|
||||||
beaconGenesis*: RestGenesis
|
beaconGenesis*: RestGenesis
|
||||||
proposerTasks*: Table[Slot, seq[ProposerTask]]
|
proposerTasks*: Table[Slot, seq[ProposerTask]]
|
||||||
|
|
||||||
@ -113,6 +136,7 @@ type
|
|||||||
|
|
||||||
const
|
const
|
||||||
DefaultDutyAndProof* = DutyAndProof(epoch: Epoch(0xFFFF_FFFF_FFFF_FFFF'u64))
|
DefaultDutyAndProof* = DutyAndProof(epoch: Epoch(0xFFFF_FFFF_FFFF_FFFF'u64))
|
||||||
|
DefaultSyncDutyAndProof* = SyncDutyAndProof(epoch: Epoch(0xFFFF_FFFF_FFFF_FFFF'u64))
|
||||||
SlotDuration* = int64(SECONDS_PER_SLOT).seconds
|
SlotDuration* = int64(SECONDS_PER_SLOT).seconds
|
||||||
OneThirdDuration* = int64(SECONDS_PER_SLOT).seconds div INTERVALS_PER_SLOT
|
OneThirdDuration* = int64(SECONDS_PER_SLOT).seconds div INTERVALS_PER_SLOT
|
||||||
|
|
||||||
@ -146,6 +170,9 @@ proc stop*(csr: ClientServiceRef) {.async.} =
|
|||||||
proc isDefault*(dap: DutyAndProof): bool =
|
proc isDefault*(dap: DutyAndProof): bool =
|
||||||
dap.epoch == Epoch(0xFFFF_FFFF_FFFF_FFFF'u64)
|
dap.epoch == Epoch(0xFFFF_FFFF_FFFF_FFFF'u64)
|
||||||
|
|
||||||
|
proc isDefault*(sdap: SyncDutyAndProof): bool =
|
||||||
|
sdap.epoch == Epoch(0xFFFF_FFFF_FFFF_FFFF'u64)
|
||||||
|
|
||||||
proc isDefault*(prd: ProposedData): bool =
|
proc isDefault*(prd: ProposedData): bool =
|
||||||
prd.epoch == Epoch(0xFFFF_FFFF_FFFF_FFFF'u64)
|
prd.epoch == Epoch(0xFFFF_FFFF_FFFF_FFFF'u64)
|
||||||
|
|
||||||
@ -155,6 +182,11 @@ proc init*(t: typedesc[DutyAndProof], epoch: Epoch, dependentRoot: Eth2Digest,
|
|||||||
DutyAndProof(epoch: epoch, dependentRoot: dependentRoot, data: duty,
|
DutyAndProof(epoch: epoch, dependentRoot: dependentRoot, data: duty,
|
||||||
slotSig: slotSig)
|
slotSig: slotSig)
|
||||||
|
|
||||||
|
proc init*(t: typedesc[SyncDutyAndProof], epoch: Epoch,
|
||||||
|
duty: SyncCommitteeDuty,
|
||||||
|
slotSig: Option[ValidatorSig]): SyncDutyAndProof =
|
||||||
|
SyncDutyAndProof(epoch: epoch, data: duty, slotSig: slotSig)
|
||||||
|
|
||||||
proc init*(t: typedesc[ProposedData], epoch: Epoch, dependentRoot: Eth2Digest,
|
proc init*(t: typedesc[ProposedData], epoch: Epoch, dependentRoot: Eth2Digest,
|
||||||
data: openArray[ProposerTask]): ProposedData =
|
data: openArray[ProposerTask]): ProposedData =
|
||||||
ProposedData(epoch: epoch, dependentRoot: dependentRoot, duties: @data)
|
ProposedData(epoch: epoch, dependentRoot: dependentRoot, duties: @data)
|
||||||
@ -174,22 +206,45 @@ proc getCurrentSlot*(vc: ValidatorClientRef): Option[Slot] =
|
|||||||
|
|
||||||
proc getAttesterDutiesForSlot*(vc: ValidatorClientRef,
|
proc getAttesterDutiesForSlot*(vc: ValidatorClientRef,
|
||||||
slot: Slot): seq[DutyAndProof] =
|
slot: Slot): seq[DutyAndProof] =
|
||||||
## Returns all `DutyAndPrrof` for the given `slot`.
|
## Returns all `DutyAndProof` for the given `slot`.
|
||||||
var res: seq[DutyAndProof]
|
var res: seq[DutyAndProof]
|
||||||
let epoch = slot.epoch()
|
let epoch = slot.epoch()
|
||||||
for key, item in vc.attesters.pairs():
|
for key, item in vc.attesters:
|
||||||
let duty = item.duties.getOrDefault(epoch, DefaultDutyAndProof)
|
let duty = item.duties.getOrDefault(epoch, DefaultDutyAndProof)
|
||||||
if not(duty.isDefault()):
|
if not(duty.isDefault()):
|
||||||
if duty.data.slot == slot:
|
if duty.data.slot == slot:
|
||||||
res.add(duty)
|
res.add(duty)
|
||||||
res
|
res
|
||||||
|
|
||||||
|
proc getSyncCommitteeDutiesForSlot*(vc: ValidatorClientRef,
|
||||||
|
slot: Slot): seq[SyncDutyAndProof] =
|
||||||
|
## Returns all `SyncDutyAndProof` for the given `slot`.
|
||||||
|
var res: seq[SyncDutyAndProof]
|
||||||
|
let epoch = slot.epoch()
|
||||||
|
for key, item in mpairs(vc.syncCommitteeDuties):
|
||||||
|
item.duties.withValue(epoch, duty):
|
||||||
|
res.add(duty[])
|
||||||
|
res
|
||||||
|
|
||||||
|
proc removeOldSyncPeriodDuties*(vc: ValidatorClientRef,
|
||||||
|
slot: Slot) =
|
||||||
|
if slot.is_sync_committee_period:
|
||||||
|
let epoch = slot.epoch()
|
||||||
|
var prunedDuties = SyncCommitteeDutiesMap()
|
||||||
|
for key, item in vc.syncCommitteeDuties:
|
||||||
|
var curPeriodDuties = EpochSyncDuties()
|
||||||
|
for epochKey, epochDuty in item.duties:
|
||||||
|
if epochKey >= epoch:
|
||||||
|
curPeriodDuties.duties[epochKey] = epochDuty
|
||||||
|
prunedDuties[key] = curPeriodDuties
|
||||||
|
vc.syncCommitteeDuties = prunedDuties
|
||||||
|
|
||||||
proc getDurationToNextAttestation*(vc: ValidatorClientRef,
|
proc getDurationToNextAttestation*(vc: ValidatorClientRef,
|
||||||
slot: Slot): string =
|
slot: Slot): string =
|
||||||
var minSlot = FAR_FUTURE_SLOT
|
var minSlot = FAR_FUTURE_SLOT
|
||||||
let currentEpoch = slot.epoch()
|
let currentEpoch = slot.epoch()
|
||||||
for epoch in [currentEpoch, currentEpoch + 1'u64]:
|
for epoch in [currentEpoch, currentEpoch + 1'u64]:
|
||||||
for key, item in vc.attesters.pairs():
|
for key, item in vc.attesters:
|
||||||
let duty = item.duties.getOrDefault(epoch, DefaultDutyAndProof)
|
let duty = item.duties.getOrDefault(epoch, DefaultDutyAndProof)
|
||||||
if not(duty.isDefault()):
|
if not(duty.isDefault()):
|
||||||
let dutySlotTime = duty.data.slot
|
let dutySlotTime = duty.data.slot
|
||||||
@ -222,11 +277,31 @@ proc getDurationToNextBlock*(vc: ValidatorClientRef, slot: Slot): string =
|
|||||||
|
|
||||||
iterator attesterDutiesForEpoch*(vc: ValidatorClientRef,
|
iterator attesterDutiesForEpoch*(vc: ValidatorClientRef,
|
||||||
epoch: Epoch): DutyAndProof =
|
epoch: Epoch): DutyAndProof =
|
||||||
for key, item in vc.attesters.pairs():
|
for key, item in vc.attesters:
|
||||||
let epochDuties = item.duties.getOrDefault(epoch)
|
let epochDuties = item.duties.getOrDefault(epoch)
|
||||||
if not(isDefault(epochDuties)):
|
if not(isDefault(epochDuties)):
|
||||||
yield epochDuties
|
yield epochDuties
|
||||||
|
|
||||||
|
proc syncMembersSubscriptionInfoForEpoch*(
|
||||||
|
vc: ValidatorClientRef,
|
||||||
|
epoch: Epoch): seq[SyncCommitteeSubscriptionInfo] =
|
||||||
|
var res: seq[SyncCommitteeSubscriptionInfo]
|
||||||
|
for key, item in mpairs(vc.syncCommitteeDuties):
|
||||||
|
var cur: SyncCommitteeSubscriptionInfo
|
||||||
|
var initialized = false
|
||||||
|
|
||||||
|
item.duties.withValue(epoch, epochDuties):
|
||||||
|
if not initialized:
|
||||||
|
cur.validator_index = epochDuties.data.validator_index
|
||||||
|
initialized = true
|
||||||
|
cur.validator_sync_committee_indices.add(
|
||||||
|
epochDuties.data.validator_sync_committee_index)
|
||||||
|
|
||||||
|
if initialized:
|
||||||
|
res.add cur
|
||||||
|
|
||||||
|
res
|
||||||
|
|
||||||
proc getDelay*(vc: ValidatorClientRef, deadline: BeaconTime): TimeDiff =
|
proc getDelay*(vc: ValidatorClientRef, deadline: BeaconTime): TimeDiff =
|
||||||
vc.beaconClock.now() - deadline
|
vc.beaconClock.now() - deadline
|
||||||
|
|
||||||
@ -253,3 +328,6 @@ proc forkAtEpoch*(vc: ValidatorClientRef, epoch: Epoch): Fork =
|
|||||||
else:
|
else:
|
||||||
break
|
break
|
||||||
res
|
res
|
||||||
|
|
||||||
|
proc getSubcommitteeIndex*(syncCommitteeIndex: IndexInSyncCommittee): SyncSubcommitteeIndex =
|
||||||
|
SyncSubcommitteeIndex(uint16(syncCommitteeIndex) div SYNC_SUBCOMMITTEE_SIZE)
|
||||||
|
@ -6,13 +6,14 @@ logScope: service = "duties_service"
|
|||||||
|
|
||||||
type
|
type
|
||||||
DutiesServiceLoop* = enum
|
DutiesServiceLoop* = enum
|
||||||
AttesterLoop, ProposerLoop, IndicesLoop
|
AttesterLoop, ProposerLoop, IndicesLoop, SyncCommitteeLoop
|
||||||
|
|
||||||
chronicles.formatIt(DutiesServiceLoop):
|
chronicles.formatIt(DutiesServiceLoop):
|
||||||
case it
|
case it
|
||||||
of AttesterLoop: "attester_loop"
|
of AttesterLoop: "attester_loop"
|
||||||
of ProposerLoop: "proposer_loop"
|
of ProposerLoop: "proposer_loop"
|
||||||
of IndicesLoop: "index_loop"
|
of IndicesLoop: "index_loop"
|
||||||
|
of SyncCommitteeLoop: "sync_committee_loop"
|
||||||
|
|
||||||
proc checkDuty(duty: RestAttesterDuty): bool =
|
proc checkDuty(duty: RestAttesterDuty): bool =
|
||||||
(duty.committee_length <= MAX_VALIDATORS_PER_COMMITTEE) and
|
(duty.committee_length <= MAX_VALIDATORS_PER_COMMITTEE) and
|
||||||
@ -20,6 +21,9 @@ proc checkDuty(duty: RestAttesterDuty): bool =
|
|||||||
(uint64(duty.validator_committee_index) <= duty.committee_length) and
|
(uint64(duty.validator_committee_index) <= duty.committee_length) and
|
||||||
(uint64(duty.validator_index) <= VALIDATOR_REGISTRY_LIMIT)
|
(uint64(duty.validator_index) <= VALIDATOR_REGISTRY_LIMIT)
|
||||||
|
|
||||||
|
proc checkSyncDuty(duty: RestSyncCommitteeDuty): bool =
|
||||||
|
uint64(duty.validator_index) <= VALIDATOR_REGISTRY_LIMIT
|
||||||
|
|
||||||
proc pollForValidatorIndices*(vc: ValidatorClientRef) {.async.} =
|
proc pollForValidatorIndices*(vc: ValidatorClientRef) {.async.} =
|
||||||
let validatorIdents =
|
let validatorIdents =
|
||||||
block:
|
block:
|
||||||
@ -161,7 +165,7 @@ proc pollForAttesterDuties*(vc: ValidatorClientRef,
|
|||||||
|
|
||||||
await allFutures(pending)
|
await allFutures(pending)
|
||||||
|
|
||||||
for index, fut in pending.pairs():
|
for index, fut in pending:
|
||||||
let item = addOrReplaceItems[index]
|
let item = addOrReplaceItems[index]
|
||||||
let dap =
|
let dap =
|
||||||
if fut.done():
|
if fut.done():
|
||||||
@ -185,11 +189,109 @@ proc pollForAttesterDuties*(vc: ValidatorClientRef,
|
|||||||
|
|
||||||
return len(addOrReplaceItems)
|
return len(addOrReplaceItems)
|
||||||
|
|
||||||
|
proc pollForSyncCommitteeDuties*(vc: ValidatorClientRef,
|
||||||
|
epoch: Epoch): Future[int] {.async.} =
|
||||||
|
let validatorIndices = toSeq(vc.attachedValidators.indices())
|
||||||
|
var
|
||||||
|
filteredDuties: seq[RestSyncCommitteeDuty]
|
||||||
|
offset = 0
|
||||||
|
remainingItems = len(validatorIndices)
|
||||||
|
|
||||||
|
while offset < len(validatorIndices):
|
||||||
|
let
|
||||||
|
arraySize = min(MaximumValidatorIds, remainingItems)
|
||||||
|
indices = validatorIndices[offset ..< (offset + arraySize)]
|
||||||
|
|
||||||
|
res =
|
||||||
|
try:
|
||||||
|
await vc.getSyncCommitteeDuties(epoch, indices)
|
||||||
|
except ValidatorApiError:
|
||||||
|
error "Unable to get sync committee duties", epoch = epoch
|
||||||
|
return 0
|
||||||
|
except CatchableError as exc:
|
||||||
|
error "Unexpected error occurred while getting sync committee duties",
|
||||||
|
epoch = epoch, err_name = exc.name, err_msg = exc.msg
|
||||||
|
return 0
|
||||||
|
|
||||||
|
for item in res.data:
|
||||||
|
if checkSyncDuty(item) and (item.pubkey in vc.attachedValidators):
|
||||||
|
filteredDuties.add(item)
|
||||||
|
|
||||||
|
offset += arraySize
|
||||||
|
remainingItems -= arraySize
|
||||||
|
|
||||||
|
let
|
||||||
|
relevantDuties =
|
||||||
|
block:
|
||||||
|
var res: seq[SyncCommitteeDuty]
|
||||||
|
for duty in filteredDuties:
|
||||||
|
for validatorSyncCommitteeIndex in duty.validator_sync_committee_indices:
|
||||||
|
res.add(SyncCommitteeDuty(
|
||||||
|
pubkey: duty.pubkey,
|
||||||
|
validator_index: duty.validator_index,
|
||||||
|
validator_sync_committee_index: validatorSyncCommitteeIndex))
|
||||||
|
res
|
||||||
|
|
||||||
|
fork = vc.forkAtEpoch(epoch)
|
||||||
|
|
||||||
|
genesisRoot = vc.beaconGenesis.genesis_validators_root
|
||||||
|
|
||||||
|
let addOrReplaceItems =
|
||||||
|
block:
|
||||||
|
var res: seq[tuple[epoch: Epoch, duty: SyncCommitteeDuty]]
|
||||||
|
for duty in relevantDuties:
|
||||||
|
let map = vc.syncCommitteeDuties.getOrDefault(duty.pubkey)
|
||||||
|
let epochDuty = map.duties.getOrDefault(epoch, DefaultSyncDutyAndProof)
|
||||||
|
info "Received new sync committee duty", duty, epoch
|
||||||
|
res.add((epoch, duty))
|
||||||
|
res
|
||||||
|
|
||||||
|
if len(addOrReplaceItems) > 0:
|
||||||
|
var pending: seq[Future[SignatureResult]]
|
||||||
|
var validators: seq[AttachedValidator]
|
||||||
|
let sres = vc.getCurrentSlot()
|
||||||
|
if sres.isSome():
|
||||||
|
for item in addOrReplaceItems:
|
||||||
|
let validator = vc.attachedValidators.getValidator(item.duty.pubkey)
|
||||||
|
let future = validator.getSyncCommitteeSelectionProof(
|
||||||
|
fork,
|
||||||
|
genesisRoot,
|
||||||
|
sres.get(),
|
||||||
|
getSubcommitteeIndex(item.duty.validator_sync_committee_index))
|
||||||
|
pending.add(future)
|
||||||
|
validators.add(validator)
|
||||||
|
|
||||||
|
await allFutures(pending)
|
||||||
|
|
||||||
|
for index, fut in pending:
|
||||||
|
let item = addOrReplaceItems[index]
|
||||||
|
let dap =
|
||||||
|
if fut.done():
|
||||||
|
let sigRes = fut.read()
|
||||||
|
if sigRes.isErr():
|
||||||
|
error "Unable to create slot signature using remote signer",
|
||||||
|
validator = shortLog(validators[index]),
|
||||||
|
error_msg = sigRes.error()
|
||||||
|
SyncDutyAndProof.init(item.epoch, item.duty,
|
||||||
|
none[ValidatorSig]())
|
||||||
|
else:
|
||||||
|
SyncDutyAndProof.init(item.epoch, item.duty,
|
||||||
|
some(sigRes.get()))
|
||||||
|
else:
|
||||||
|
SyncDutyAndProof.init(item.epoch, item.duty,
|
||||||
|
none[ValidatorSig]())
|
||||||
|
|
||||||
|
var validatorDuties = vc.syncCommitteeDuties.getOrDefault(item.duty.pubkey)
|
||||||
|
validatorDuties.duties[item.epoch] = dap
|
||||||
|
vc.syncCommitteeDuties[item.duty.pubkey] = validatorDuties
|
||||||
|
|
||||||
|
return len(addOrReplaceItems)
|
||||||
|
|
||||||
proc pruneAttesterDuties(vc: ValidatorClientRef, epoch: Epoch) =
|
proc pruneAttesterDuties(vc: ValidatorClientRef, epoch: Epoch) =
|
||||||
var attesters: AttesterMap
|
var attesters: AttesterMap
|
||||||
for key, item in vc.attesters.pairs():
|
for key, item in vc.attesters:
|
||||||
var v = EpochDuties()
|
var v = EpochDuties()
|
||||||
for epochKey, epochDuty in item.duties.pairs():
|
for epochKey, epochDuty in item.duties:
|
||||||
if (epochKey + HISTORICAL_DUTIES_EPOCHS) >= epoch:
|
if (epochKey + HISTORICAL_DUTIES_EPOCHS) >= epoch:
|
||||||
v.duties[epochKey] = epochDuty
|
v.duties[epochKey] = epochDuty
|
||||||
else:
|
else:
|
||||||
@ -251,9 +353,45 @@ proc pollForAttesterDuties*(vc: ValidatorClientRef) {.async.} =
|
|||||||
|
|
||||||
vc.pruneAttesterDuties(currentEpoch)
|
vc.pruneAttesterDuties(currentEpoch)
|
||||||
|
|
||||||
|
proc pollForSyncCommitteeDuties* (vc: ValidatorClientRef) {.async.} =
|
||||||
|
let sres = vc.getCurrentSlot()
|
||||||
|
if sres.isSome():
|
||||||
|
let
|
||||||
|
currentSlot = sres.get()
|
||||||
|
currentEpoch = currentSlot.epoch()
|
||||||
|
nextEpoch = currentEpoch + 1'u64
|
||||||
|
|
||||||
|
if vc.attachedValidators.count() != 0:
|
||||||
|
var counts: array[2, tuple[epoch: Epoch, count: int]]
|
||||||
|
counts[0] = (currentEpoch, await vc.pollForSyncCommitteeDuties(currentEpoch))
|
||||||
|
counts[1] = (nextEpoch, await vc.pollForSyncCommitteeDuties(nextEpoch))
|
||||||
|
|
||||||
|
if (counts[0].count == 0) and (counts[1].count == 0):
|
||||||
|
debug "No new sync committee member's duties received", slot = currentSlot
|
||||||
|
|
||||||
|
let subscriptions =
|
||||||
|
block:
|
||||||
|
var res: seq[RestSyncCommitteeSubscription]
|
||||||
|
for item in counts:
|
||||||
|
if item.count > 0:
|
||||||
|
let subscriptionsInfo = vc.syncMembersSubscriptionInfoForEpoch(item.epoch)
|
||||||
|
for subInfo in subscriptionsInfo:
|
||||||
|
let sub = RestSyncCommitteeSubscription(
|
||||||
|
validator_index: subInfo.validator_index,
|
||||||
|
sync_committee_indices: subInfo.validator_sync_committee_indices,
|
||||||
|
until_epoch: (currentEpoch + EPOCHS_PER_SYNC_COMMITTEE_PERIOD -
|
||||||
|
currentEpoch.since_sync_committee_period_start()).Epoch
|
||||||
|
)
|
||||||
|
res.add(sub)
|
||||||
|
res
|
||||||
|
if len(subscriptions) > 0:
|
||||||
|
let res = await vc.prepareSyncCommitteeSubnets(subscriptions)
|
||||||
|
if not(res):
|
||||||
|
error "Failed to subscribe validators"
|
||||||
|
|
||||||
proc pruneBeaconProposers(vc: ValidatorClientRef, epoch: Epoch) =
|
proc pruneBeaconProposers(vc: ValidatorClientRef, epoch: Epoch) =
|
||||||
var proposers: ProposerMap
|
var proposers: ProposerMap
|
||||||
for epochKey, data in vc.proposers.pairs():
|
for epochKey, data in vc.proposers:
|
||||||
if (epochKey + HISTORICAL_DUTIES_EPOCHS) >= epoch:
|
if (epochKey + HISTORICAL_DUTIES_EPOCHS) >= epoch:
|
||||||
proposers[epochKey] = data
|
proposers[epochKey] = data
|
||||||
else:
|
else:
|
||||||
@ -323,6 +461,12 @@ proc validatorIndexLoop(service: DutiesServiceRef) {.async.} =
|
|||||||
await vc.pollForValidatorIndices()
|
await vc.pollForValidatorIndices()
|
||||||
await service.waitForNextSlot(IndicesLoop)
|
await service.waitForNextSlot(IndicesLoop)
|
||||||
|
|
||||||
|
proc syncCommitteeeDutiesLoop(service: DutiesServiceRef) {.async.} =
|
||||||
|
let vc = service.client
|
||||||
|
while true:
|
||||||
|
await vc.pollForSyncCommitteeDuties()
|
||||||
|
await service.waitForNextSlot(SyncCommitteeLoop)
|
||||||
|
|
||||||
template checkAndRestart(serviceLoop: DutiesServiceLoop,
|
template checkAndRestart(serviceLoop: DutiesServiceLoop,
|
||||||
future: Future[void], body: untyped): untyped =
|
future: Future[void], body: untyped): untyped =
|
||||||
if future.finished():
|
if future.finished():
|
||||||
@ -345,6 +489,7 @@ proc mainLoop(service: DutiesServiceRef) {.async.} =
|
|||||||
fut1 = service.attesterDutiesLoop()
|
fut1 = service.attesterDutiesLoop()
|
||||||
fut2 = service.proposerDutiesLoop()
|
fut2 = service.proposerDutiesLoop()
|
||||||
fut3 = service.validatorIndexLoop()
|
fut3 = service.validatorIndexLoop()
|
||||||
|
fut4 = service.syncCommitteeeDutiesLoop()
|
||||||
|
|
||||||
while true:
|
while true:
|
||||||
var breakLoop = false
|
var breakLoop = false
|
||||||
@ -354,7 +499,8 @@ proc mainLoop(service: DutiesServiceRef) {.async.} =
|
|||||||
if not(fut1.finished()): fut1.cancel()
|
if not(fut1.finished()): fut1.cancel()
|
||||||
if not(fut2.finished()): fut2.cancel()
|
if not(fut2.finished()): fut2.cancel()
|
||||||
if not(fut3.finished()): fut3.cancel()
|
if not(fut3.finished()): fut3.cancel()
|
||||||
await allFutures(fut1, fut2, fut3)
|
if not(fut4.finished()): fut4.cancel()
|
||||||
|
await allFutures(fut1, fut2, fut3, fut4)
|
||||||
breakLoop = true
|
breakLoop = true
|
||||||
|
|
||||||
if breakLoop:
|
if breakLoop:
|
||||||
@ -363,6 +509,7 @@ proc mainLoop(service: DutiesServiceRef) {.async.} =
|
|||||||
checkAndRestart(AttesterLoop, fut1, service.attesterDutiesLoop())
|
checkAndRestart(AttesterLoop, fut1, service.attesterDutiesLoop())
|
||||||
checkAndRestart(ProposerLoop, fut2, service.proposerDutiesLoop())
|
checkAndRestart(ProposerLoop, fut2, service.proposerDutiesLoop())
|
||||||
checkAndRestart(IndicesLoop, fut3, service.validatorIndexLoop())
|
checkAndRestart(IndicesLoop, fut3, service.validatorIndexLoop())
|
||||||
|
checkAndRestart(SyncCommitteeLoop, fut4, service.syncCommitteeeDutiesLoop())
|
||||||
except CatchableError as exc:
|
except CatchableError as exc:
|
||||||
warn "Service crashed with unexpected error", err_name = exc.name,
|
warn "Service crashed with unexpected error", err_name = exc.name,
|
||||||
err_msg = exc.msg
|
err_msg = exc.msg
|
||||||
|
@ -7,7 +7,7 @@ logScope: service = "fork_service"
|
|||||||
proc validateForkSchedule(forks: openArray[Fork]): bool {.raises: [Defect].} =
|
proc validateForkSchedule(forks: openArray[Fork]): bool {.raises: [Defect].} =
|
||||||
# Check if `forks` list is linked list.
|
# Check if `forks` list is linked list.
|
||||||
var current_version = forks[0].current_version
|
var current_version = forks[0].current_version
|
||||||
for index, item in forks.pairs():
|
for index, item in forks:
|
||||||
if index > 0:
|
if index > 0:
|
||||||
if item.previous_version != current_version:
|
if item.previous_version != current_version:
|
||||||
return false
|
return false
|
||||||
|
356
beacon_chain/validator_client/sync_committee_service.nim
Normal file
356
beacon_chain/validator_client/sync_committee_service.nim
Normal file
@ -0,0 +1,356 @@
|
|||||||
|
import
|
||||||
|
std/[sets, sequtils],
|
||||||
|
chronicles,
|
||||||
|
"."/[common, api, block_service],
|
||||||
|
../spec/datatypes/[phase0, altair, bellatrix],
|
||||||
|
../spec/eth2_apis/rest_types
|
||||||
|
|
||||||
|
type
|
||||||
|
ContributionItem* = object
|
||||||
|
aggregator_index: uint64
|
||||||
|
selection_proof: ValidatorSig
|
||||||
|
validator: AttachedValidator
|
||||||
|
subcommitteeIdx: SyncSubcommitteeIndex
|
||||||
|
|
||||||
|
proc serveSyncCommitteeMessage*(service: SyncCommitteeServiceRef,
|
||||||
|
slot: Slot,
|
||||||
|
beaconBlockRoot: Eth2Digest,
|
||||||
|
duty: SyncDutyAndProof): Future[bool] {.async.} =
|
||||||
|
let
|
||||||
|
vc = service.client
|
||||||
|
fork = vc.forkAtEpoch(slot.epoch)
|
||||||
|
genesisValidatorsRoot = vc.beaconGenesis.genesis_validators_root
|
||||||
|
|
||||||
|
vindex = duty.data.validator_index
|
||||||
|
subcommitteeIdx = getSubcommitteeIndex(duty.data.validator_sync_committee_index)
|
||||||
|
|
||||||
|
validator =
|
||||||
|
block:
|
||||||
|
let res = vc.getValidator(duty.data.pubkey)
|
||||||
|
if res.isNone():
|
||||||
|
return false
|
||||||
|
res.get()
|
||||||
|
|
||||||
|
message =
|
||||||
|
block:
|
||||||
|
let res = await signSyncCommitteeMessage(validator, fork,
|
||||||
|
genesisValidatorsRoot,
|
||||||
|
slot, beaconBlockRoot)
|
||||||
|
if res.isErr():
|
||||||
|
error "Unable to sign committee message using remote signer",
|
||||||
|
validator = shortLog(validator), slot = slot,
|
||||||
|
block_root = shortLog(beaconBlockRoot)
|
||||||
|
return
|
||||||
|
res.get()
|
||||||
|
|
||||||
|
debug "Sending sync committee message", message = shortLog(message),
|
||||||
|
validator = shortLog(validator), validator_index = vindex,
|
||||||
|
delay = vc.getDelay(message.slot.sync_committee_message_deadline())
|
||||||
|
|
||||||
|
let res =
|
||||||
|
try:
|
||||||
|
await vc.submitPoolSyncCommitteeSignature(message)
|
||||||
|
except ValidatorApiError:
|
||||||
|
error "Unable to publish sync committee message",
|
||||||
|
message = shortLog(message),
|
||||||
|
validator = shortLog(validator),
|
||||||
|
validator_index = vindex
|
||||||
|
return false
|
||||||
|
except CatchableError as exc:
|
||||||
|
error "Unexpected error occurred while publishing sync committee message",
|
||||||
|
message = shortLog(message),
|
||||||
|
validator = shortLog(validator),
|
||||||
|
validator_index = vindex,
|
||||||
|
err_name = exc.name, err_msg = exc.msg
|
||||||
|
return false
|
||||||
|
|
||||||
|
let delay = vc.getDelay(message.slot.sync_committee_message_deadline())
|
||||||
|
if res:
|
||||||
|
notice "Sync committee message published",
|
||||||
|
message = shortLog(message),
|
||||||
|
validator = shortLog(validator),
|
||||||
|
validator_index = vindex,
|
||||||
|
delay = delay
|
||||||
|
else:
|
||||||
|
warn "Sync committee message was not accepted by beacon node",
|
||||||
|
message = shortLog(message),
|
||||||
|
validator = shortLog(validator),
|
||||||
|
validator_index = vindex, delay = delay
|
||||||
|
|
||||||
|
return res
|
||||||
|
|
||||||
|
proc produceAndPublishSyncCommitteeMessages(service: SyncCommitteeServiceRef,
|
||||||
|
slot: Slot,
|
||||||
|
beaconBlockRoot: Eth2Digest,
|
||||||
|
duties: seq[SyncDutyAndProof]) {.async.} =
|
||||||
|
let vc = service.client
|
||||||
|
|
||||||
|
let pendingSyncCommitteeMessages =
|
||||||
|
block:
|
||||||
|
var res: seq[Future[bool]]
|
||||||
|
for duty in duties:
|
||||||
|
debug "Serving sync message duty", duty = duty.data, epoch = slot.epoch()
|
||||||
|
res.add(service.serveSyncCommitteeMessage(slot,
|
||||||
|
beaconBlockRoot,
|
||||||
|
duty))
|
||||||
|
res
|
||||||
|
|
||||||
|
let statistics =
|
||||||
|
block:
|
||||||
|
var errored, succeed, failed = 0
|
||||||
|
try:
|
||||||
|
await allFutures(pendingSyncCommitteeMessages)
|
||||||
|
except CancelledError as exc:
|
||||||
|
for fut in pendingSyncCommitteeMessages:
|
||||||
|
if not(fut.finished()):
|
||||||
|
fut.cancel()
|
||||||
|
await allFutures(pendingSyncCommitteeMessages)
|
||||||
|
raise exc
|
||||||
|
|
||||||
|
for future in pendingSyncCommitteeMessages:
|
||||||
|
if future.done():
|
||||||
|
if future.read():
|
||||||
|
inc(succeed)
|
||||||
|
else:
|
||||||
|
inc(failed)
|
||||||
|
else:
|
||||||
|
inc(errored)
|
||||||
|
(succeed, errored, failed)
|
||||||
|
|
||||||
|
let delay = vc.getDelay(slot.attestation_deadline())
|
||||||
|
debug "Sync committee message statistics", total = len(pendingSyncCommitteeMessages),
|
||||||
|
succeed = statistics[0], failed_to_deliver = statistics[1],
|
||||||
|
not_accepted = statistics[2], delay = delay, slot = slot,
|
||||||
|
duties_count = len(duties)
|
||||||
|
|
||||||
|
proc serveContributionAndProof*(service: SyncCommitteeServiceRef,
|
||||||
|
proof: ContributionAndProof,
|
||||||
|
validator: AttachedValidator): Future[bool] {.async.} =
|
||||||
|
let
|
||||||
|
vc = service.client
|
||||||
|
slot = proof.contribution.slot
|
||||||
|
validatorIdx = validator.index.get()
|
||||||
|
genesisRoot = vc.beaconGenesis.genesis_validators_root
|
||||||
|
fork = vc.forkAtEpoch(slot.epoch)
|
||||||
|
signedProof = (ref SignedContributionAndProof)(
|
||||||
|
message: proof)
|
||||||
|
|
||||||
|
let signature =
|
||||||
|
block:
|
||||||
|
let res = await validator.sign(signedProof, fork, genesisRoot)
|
||||||
|
if res.isErr():
|
||||||
|
error "Unable to sign sync committee contribution using remote signer",
|
||||||
|
validator = shortLog(validator),
|
||||||
|
contribution = shortLog(proof.contribution),
|
||||||
|
error_msg = res.error()
|
||||||
|
return false
|
||||||
|
res.get()
|
||||||
|
|
||||||
|
debug "Sending sync contribution",
|
||||||
|
contribution = shortLog(signedProof.message.contribution),
|
||||||
|
validator = shortLog(validator), validator_index = validatorIdx,
|
||||||
|
delay = vc.getDelay(slot.sync_contribution_deadline())
|
||||||
|
|
||||||
|
let restSignedProof = RestSignedContributionAndProof.init(
|
||||||
|
signedProof.message, signedProof.signature)
|
||||||
|
|
||||||
|
let res =
|
||||||
|
try:
|
||||||
|
await vc.publishContributionAndProofs(@[restSignedProof])
|
||||||
|
except ValidatorApiError as err:
|
||||||
|
error "Unable to publish sync contribution",
|
||||||
|
contribution = shortLog(signedProof.message.contribution),
|
||||||
|
validator = shortLog(validator),
|
||||||
|
validator_index = validatorIdx,
|
||||||
|
err_msg = err.msg
|
||||||
|
false
|
||||||
|
except CatchableError as err:
|
||||||
|
error "Unexpected error occurred while publishing sync contribution",
|
||||||
|
contribution = shortLog(signedProof.message.contribution),
|
||||||
|
validator = shortLog(validator),
|
||||||
|
err_name = err.name, err_msg = err.msg
|
||||||
|
false
|
||||||
|
|
||||||
|
if res:
|
||||||
|
notice "Sync contribution published",
|
||||||
|
validator = shortLog(validator),
|
||||||
|
validator_index = validatorIdx
|
||||||
|
else:
|
||||||
|
warn "Sync contribution was not accepted by beacon node",
|
||||||
|
contribution = shortLog(signedProof.message.contribution),
|
||||||
|
validator = shortLog(validator),
|
||||||
|
validator_index = validatorIdx
|
||||||
|
return res
|
||||||
|
|
||||||
|
proc produceAndPublishContributions(service: SyncCommitteeServiceRef,
|
||||||
|
slot: Slot,
|
||||||
|
beaconBlockRoot: Eth2Digest,
|
||||||
|
duties: seq[SyncDutyAndProof]) {.async.} =
|
||||||
|
let
|
||||||
|
vc = service.client
|
||||||
|
contributionItems =
|
||||||
|
block:
|
||||||
|
var res: seq[ContributionItem]
|
||||||
|
for duty in duties:
|
||||||
|
let validator = vc.attachedValidators.getValidator(duty.data.pubkey)
|
||||||
|
if not isNil(validator):
|
||||||
|
if duty.slotSig.isSome:
|
||||||
|
template slotSignature: auto = duty.slotSig.get
|
||||||
|
if is_sync_committee_aggregator(slotSignature):
|
||||||
|
res.add(ContributionItem(
|
||||||
|
aggregator_index: uint64(duty.data.validator_index),
|
||||||
|
selection_proof: slotSignature,
|
||||||
|
validator: validator,
|
||||||
|
subcommitteeIdx: getSubcommitteeIndex(
|
||||||
|
duty.data.validator_sync_committee_index)
|
||||||
|
))
|
||||||
|
res
|
||||||
|
|
||||||
|
if len(contributionItems) > 0:
|
||||||
|
let pendingAggregates =
|
||||||
|
block:
|
||||||
|
var res: seq[Future[bool]]
|
||||||
|
for item in contributionItems:
|
||||||
|
let aggContribution =
|
||||||
|
try:
|
||||||
|
await vc.produceSyncCommitteeContribution(slot,
|
||||||
|
item.subcommitteeIdx,
|
||||||
|
beaconBlockRoot)
|
||||||
|
except ValidatorApiError:
|
||||||
|
error "Unable to get sync message contribution data", slot = slot,
|
||||||
|
beaconBlockRoot = shortLog(beaconBlockRoot)
|
||||||
|
return
|
||||||
|
except CatchableError as exc:
|
||||||
|
error "Unexpected error occurred while getting sync message contribution",
|
||||||
|
slot = slot, beaconBlockRoot = shortLog(beaconBlockRoot),
|
||||||
|
err_name = exc.name, err_msg = exc.msg
|
||||||
|
return
|
||||||
|
|
||||||
|
let proof = ContributionAndProof(
|
||||||
|
aggregator_index: item.aggregator_index,
|
||||||
|
contribution: aggContribution,
|
||||||
|
selection_proof: item.selection_proof
|
||||||
|
)
|
||||||
|
res.add(service.serveContributionAndProof(proof, item.validator))
|
||||||
|
res
|
||||||
|
|
||||||
|
let statistics =
|
||||||
|
block:
|
||||||
|
var errored, succeed, failed = 0
|
||||||
|
try:
|
||||||
|
await allFutures(pendingAggregates)
|
||||||
|
except CancelledError as err:
|
||||||
|
for fut in pendingAggregates:
|
||||||
|
if not(fut.finished()):
|
||||||
|
fut.cancel()
|
||||||
|
await allFutures(pendingAggregates)
|
||||||
|
raise err
|
||||||
|
|
||||||
|
for future in pendingAggregates:
|
||||||
|
if future.done():
|
||||||
|
if future.read():
|
||||||
|
inc(succeed)
|
||||||
|
else:
|
||||||
|
inc(failed)
|
||||||
|
else:
|
||||||
|
inc(errored)
|
||||||
|
(succeed, errored, failed)
|
||||||
|
|
||||||
|
let delay = vc.getDelay(slot.aggregate_deadline())
|
||||||
|
debug "Sync message contribution statistics", total = len(pendingAggregates),
|
||||||
|
succeed = statistics[0], failed_to_deliver = statistics[1],
|
||||||
|
not_accepted = statistics[2], delay = delay, slot = slot
|
||||||
|
|
||||||
|
else:
|
||||||
|
debug "No contribution and proofs scheduled for slot", slot = slot
|
||||||
|
|
||||||
|
proc publishSyncMessagesAndContributions(service: SyncCommitteeServiceRef,
|
||||||
|
slot: Slot,
|
||||||
|
duties: seq[SyncDutyAndProof]) {.async.} =
|
||||||
|
let
|
||||||
|
vc = service.client
|
||||||
|
startTime = Moment.now()
|
||||||
|
|
||||||
|
try:
|
||||||
|
let timeout = syncCommitteeMessageSlotOffset
|
||||||
|
await vc.waitForBlockPublished(slot).wait(nanoseconds(timeout.nanoseconds))
|
||||||
|
let dur = Moment.now() - startTime
|
||||||
|
debug "Block proposal awaited", slot = slot, duration = dur
|
||||||
|
except AsyncTimeoutError:
|
||||||
|
let dur = Moment.now() - startTime
|
||||||
|
debug "Block was not produced in time", slot = slot, duration = dur
|
||||||
|
|
||||||
|
block:
|
||||||
|
let delay = vc.getDelay(slot.sync_committee_message_deadline())
|
||||||
|
debug "Producing sync committee messages", delay = delay, slot = slot,
|
||||||
|
duties_count = len(duties)
|
||||||
|
let beaconBlockRoot =
|
||||||
|
block:
|
||||||
|
try:
|
||||||
|
let res = await vc.getHeadBlockRoot()
|
||||||
|
res.root
|
||||||
|
except CatchableError as exc:
|
||||||
|
error "Could not request sync message block root to sign"
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
await service.produceAndPublishSyncCommitteeMessages(slot,
|
||||||
|
beaconBlockRoot,
|
||||||
|
duties)
|
||||||
|
except ValidatorApiError:
|
||||||
|
error "Unable to proceed sync committee messages", slot = slot,
|
||||||
|
duties_count = len(duties)
|
||||||
|
return
|
||||||
|
except CatchableError as exc:
|
||||||
|
error "Unexpected error while producing sync committee messages",
|
||||||
|
slot = slot,
|
||||||
|
duties_count = len(duties),
|
||||||
|
err_name = exc.name, err_msg = exc.msg
|
||||||
|
return
|
||||||
|
|
||||||
|
let contributionTime =
|
||||||
|
# chronos.Duration subtraction cannot return a negative value; in such
|
||||||
|
# case it will return `ZeroDuration`.
|
||||||
|
vc.beaconClock.durationToNextSlot() - OneThirdDuration
|
||||||
|
if contributionTime != ZeroDuration:
|
||||||
|
await sleepAsync(contributionTime)
|
||||||
|
|
||||||
|
block:
|
||||||
|
let delay = vc.getDelay(slot.sync_contribution_deadline())
|
||||||
|
debug "Producing contribution and proofs", delay = delay
|
||||||
|
await service.produceAndPublishContributions(slot, beaconBlockRoot, duties)
|
||||||
|
|
||||||
|
proc spawnSyncCommitteeTasks(service: SyncCommitteeServiceRef, slot: Slot) =
|
||||||
|
let vc = service.client
|
||||||
|
|
||||||
|
removeOldSyncPeriodDuties(vc, slot)
|
||||||
|
let duties = vc.getSyncCommitteeDutiesForSlot(slot + 1)
|
||||||
|
|
||||||
|
asyncSpawn service.publishSyncMessagesAndContributions(slot, duties)
|
||||||
|
|
||||||
|
proc mainLoop(service: SyncCommitteeServiceRef) {.async.} =
|
||||||
|
let vc = service.client
|
||||||
|
service.state = ServiceState.Running
|
||||||
|
try:
|
||||||
|
while true:
|
||||||
|
let sleepTime =
|
||||||
|
syncCommitteeMessageSlotOffset + vc.beaconClock.durationToNextSlot()
|
||||||
|
|
||||||
|
let sres = vc.getCurrentSlot()
|
||||||
|
if sres.isSome():
|
||||||
|
let currentSlot = sres.get()
|
||||||
|
service.spawnSyncCommitteeTasks(currentSlot)
|
||||||
|
await sleepAsync(sleepTime)
|
||||||
|
except CatchableError as exc:
|
||||||
|
warn "Service crashed with unexpected error", err_name = exc.name,
|
||||||
|
err_msg = exc.msg
|
||||||
|
|
||||||
|
proc init*(t: typedesc[SyncCommitteeServiceRef],
|
||||||
|
vc: ValidatorClientRef): Future[SyncCommitteeServiceRef] {.async.} =
|
||||||
|
debug "Initializing service"
|
||||||
|
var res = SyncCommitteeServiceRef(client: vc,
|
||||||
|
state: ServiceState.Initialized)
|
||||||
|
return res
|
||||||
|
|
||||||
|
proc start*(service: SyncCommitteeServiceRef) =
|
||||||
|
service.lifeFut = mainLoop(service)
|
@ -276,7 +276,7 @@ proc sendSyncCommitteeMessages*(node: BeaconNode,
|
|||||||
var resCur: Table[uint64, int]
|
var resCur: Table[uint64, int]
|
||||||
var resNxt: Table[uint64, int]
|
var resNxt: Table[uint64, int]
|
||||||
|
|
||||||
for index, msg in msgs.pairs():
|
for index, msg in msgs:
|
||||||
if msg.validator_index < lenu64(state.data.validators):
|
if msg.validator_index < lenu64(state.data.validators):
|
||||||
let msgPeriod = sync_committee_period(msg.slot + 1)
|
let msgPeriod = sync_committee_period(msg.slot + 1)
|
||||||
if msgPeriod == curPeriod:
|
if msgPeriod == curPeriod:
|
||||||
@ -316,7 +316,7 @@ proc sendSyncCommitteeMessages*(node: BeaconNode,
|
|||||||
|
|
||||||
await allFutures(pending)
|
await allFutures(pending)
|
||||||
|
|
||||||
for index, future in pending.pairs():
|
for index, future in pending:
|
||||||
if future.done():
|
if future.done():
|
||||||
let fres = future.read()
|
let fres = future.read()
|
||||||
if fres.isErr():
|
if fres.isErr():
|
||||||
@ -898,7 +898,7 @@ proc handleSyncCommitteeContributions(node: BeaconNode,
|
|||||||
subcommitteeIdx: subcommitteeIdx)
|
subcommitteeIdx: subcommitteeIdx)
|
||||||
|
|
||||||
selectionProofs.add validator.getSyncCommitteeSelectionProof(
|
selectionProofs.add validator.getSyncCommitteeSelectionProof(
|
||||||
fork, genesis_validators_root, slot, subcommitteeIdx.asUInt64)
|
fork, genesis_validators_root, slot, subcommitteeIdx)
|
||||||
|
|
||||||
await allFutures(selectionProofs)
|
await allFutures(selectionProofs)
|
||||||
|
|
||||||
@ -908,7 +908,7 @@ proc handleSyncCommitteeContributions(node: BeaconNode,
|
|||||||
var contributionsSent = 0
|
var contributionsSent = 0
|
||||||
|
|
||||||
time = timeIt:
|
time = timeIt:
|
||||||
for i, proof in selectionProofs.pairs():
|
for i, proof in selectionProofs:
|
||||||
if not proof.completed:
|
if not proof.completed:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
@ -281,7 +281,7 @@ proc updateEpoch(self: var ValidatorMonitor, epoch: Epoch) =
|
|||||||
# and hope things improve
|
# and hope things improve
|
||||||
notice "Resetting validator monitoring", epoch, monitorEpoch
|
notice "Resetting validator monitoring", epoch, monitorEpoch
|
||||||
|
|
||||||
for (_, monitor) in self.monitors.mpairs():
|
for _, monitor in self.monitors:
|
||||||
reset(monitor.summaries)
|
reset(monitor.summaries)
|
||||||
return
|
return
|
||||||
|
|
||||||
|
@ -260,11 +260,11 @@ proc signWithRemoteValidator*(v: AttachedValidator, fork: Fork,
|
|||||||
proc signWithRemoteValidator*(v: AttachedValidator, fork: Fork,
|
proc signWithRemoteValidator*(v: AttachedValidator, fork: Fork,
|
||||||
genesis_validators_root: Eth2Digest,
|
genesis_validators_root: Eth2Digest,
|
||||||
slot: Slot,
|
slot: Slot,
|
||||||
subIndex: uint64): Future[SignatureResult]
|
subcommittee: SyncSubcommitteeIndex): Future[SignatureResult]
|
||||||
{.async.} =
|
{.async.} =
|
||||||
let request = Web3SignerRequest.init(
|
let request = Web3SignerRequest.init(
|
||||||
fork, genesis_validators_root,
|
fork, genesis_validators_root,
|
||||||
SyncAggregatorSelectionData(slot: slot, subcommittee_index: subIndex),
|
SyncAggregatorSelectionData(slot: slot, subcommittee_index: uint64 subcommittee)
|
||||||
)
|
)
|
||||||
debug "Signing sync aggregator selection data using remote signer",
|
debug "Signing sync aggregator selection data using remote signer",
|
||||||
validator = shortLog(v)
|
validator = shortLog(v)
|
||||||
@ -384,7 +384,7 @@ proc getSyncCommitteeSelectionProof*(v: AttachedValidator,
|
|||||||
fork: Fork,
|
fork: Fork,
|
||||||
genesis_validators_root: Eth2Digest,
|
genesis_validators_root: Eth2Digest,
|
||||||
slot: Slot,
|
slot: Slot,
|
||||||
subcommittee_index: uint64
|
subcommittee_index: SyncSubcommitteeIndex
|
||||||
): Future[SignatureResult] {.async.} =
|
): Future[SignatureResult] {.async.} =
|
||||||
return
|
return
|
||||||
case v.kind
|
case v.kind
|
||||||
|
@ -23,7 +23,7 @@ const
|
|||||||
|
|
||||||
type
|
type
|
||||||
StartUpCommand {.pure.} = enum
|
StartUpCommand {.pure.} = enum
|
||||||
pubsub, asl, asr, aggasr, lat
|
pubsub, asl, asr, aggasr, scmsr, csr, lat, traceAll
|
||||||
|
|
||||||
LogTraceConf = object
|
LogTraceConf = object
|
||||||
logFiles {.
|
logFiles {.
|
||||||
@ -74,8 +74,14 @@ type
|
|||||||
discard
|
discard
|
||||||
of aggasr:
|
of aggasr:
|
||||||
discard
|
discard
|
||||||
|
of scmsr:
|
||||||
|
discard
|
||||||
|
of csr:
|
||||||
|
discard
|
||||||
of lat:
|
of lat:
|
||||||
discard
|
discard
|
||||||
|
of traceAll:
|
||||||
|
discard
|
||||||
|
|
||||||
GossipDirection = enum
|
GossipDirection = enum
|
||||||
None, Incoming, Outgoing
|
None, Incoming, Outgoing
|
||||||
@ -140,22 +146,71 @@ type
|
|||||||
wallSlot: uint64
|
wallSlot: uint64
|
||||||
signature: string
|
signature: string
|
||||||
|
|
||||||
|
SyncCommitteeMessageObject = object
|
||||||
|
slot: uint64
|
||||||
|
beaconBlockRoot {.serializedFieldName: "beacon_block_root".}: string
|
||||||
|
validatorIndex {.serializedFieldName: "validator_index".}: uint64
|
||||||
|
signature: string
|
||||||
|
|
||||||
|
ContributionObject = object
|
||||||
|
slot: uint64
|
||||||
|
beaconBlockRoot {.serializedFieldName: "beacon_block_root".}: string
|
||||||
|
subnetId: uint64
|
||||||
|
aggregationBits {.serializedFieldName: "aggregation_bits".}: string
|
||||||
|
|
||||||
|
ContributionMessageObject = object
|
||||||
|
aggregatorIndex {.serializedFieldName: "aggregator_index".}: uint64
|
||||||
|
contribution: ContributionObject
|
||||||
|
selectionProof {.serializedFieldName: "selection_proof".}: string
|
||||||
|
|
||||||
|
ContributionSentObject = object
|
||||||
|
message: ContributionMessageObject
|
||||||
|
signature: string
|
||||||
|
|
||||||
|
SCMSentMessage = object of LogMessage
|
||||||
|
message: SyncCommitteeMessageObject
|
||||||
|
validator: string
|
||||||
|
|
||||||
|
SCMReceivedMessage = object of LogMessage
|
||||||
|
wallSlot: uint64
|
||||||
|
syncCommitteeMsg: SyncCommitteeMessageObject
|
||||||
|
subcommitteeIdx: uint64
|
||||||
|
|
||||||
|
ContributionSentMessage = object of LogMessage
|
||||||
|
contribution: ContributionSentObject
|
||||||
|
|
||||||
|
ContributionReceivedMessage = object of LogMessage
|
||||||
|
contribution: ContributionObject
|
||||||
|
wallSlot: uint64
|
||||||
|
aggregatorIndex {.serializedFieldName: "aggregator_index".}: uint64
|
||||||
|
signature: string
|
||||||
|
selectionProof {.serializedFieldName: "selection_proof".}: string
|
||||||
|
|
||||||
GossipMessage = object
|
GossipMessage = object
|
||||||
kind: GossipDirection
|
kind: GossipDirection
|
||||||
id: string
|
id: string
|
||||||
datetime: DateTime
|
datetime: DateTime
|
||||||
processed: bool
|
processed: bool
|
||||||
|
|
||||||
SaMessageType {.pure.} = enum
|
SMessageType {.pure.} = enum
|
||||||
AttestationSent, SlotStart
|
AttestationSent, SCMSent, SlotStart
|
||||||
|
|
||||||
SlotAttMessage = object
|
SlotMessage = object
|
||||||
case kind: SaMessageType
|
case kind: SMessageType
|
||||||
of SaMessageType.AttestationSent:
|
of SMessageType.AttestationSent:
|
||||||
asmsg: AttestationSentMessage
|
asmsg: AttestationSentMessage
|
||||||
of SaMessageType.SlotStart:
|
of SMessageType.SCMSent:
|
||||||
|
scmsmsg: SCMSentMessage
|
||||||
|
of SMessageType.SlotStart:
|
||||||
ssmsg: SlotStartMessage
|
ssmsg: SlotStartMessage
|
||||||
|
|
||||||
|
# SlotMessage = object
|
||||||
|
# case kind: SMessageType
|
||||||
|
# of SMessageType.SCMSent:
|
||||||
|
# scmsmsg: SCMSentMessage
|
||||||
|
# of SMessageType.SlotStart:
|
||||||
|
# ssmsg: SlotStartMessage
|
||||||
|
|
||||||
SRANode = object
|
SRANode = object
|
||||||
directory: NodeDirectory
|
directory: NodeDirectory
|
||||||
sends: seq[AttestationSentMessage]
|
sends: seq[AttestationSentMessage]
|
||||||
@ -163,6 +218,13 @@ type
|
|||||||
aggSends: seq[AggregatedAttestationSentMessage]
|
aggSends: seq[AggregatedAttestationSentMessage]
|
||||||
aggRecvs: TableRef[string, AggregatedAttestationReceivedMessage]
|
aggRecvs: TableRef[string, AggregatedAttestationReceivedMessage]
|
||||||
|
|
||||||
|
SRSCNode = object
|
||||||
|
directory: NodeDirectory
|
||||||
|
sends: seq[SCMSentMessage]
|
||||||
|
recvs: TableRef[string, SCMReceivedMessage]
|
||||||
|
contributionSends: seq[ContributionSentMessage]
|
||||||
|
contributionRecvs: TableRef[string, ContributionReceivedMessage]
|
||||||
|
|
||||||
proc readValue(reader: var JsonReader, value: var DateTime) =
|
proc readValue(reader: var JsonReader, value: var DateTime) =
|
||||||
let s = reader.readValue(string)
|
let s = reader.readValue(string)
|
||||||
try:
|
try:
|
||||||
@ -198,8 +260,8 @@ proc readLogFile(file: string): seq[JsonNode] =
|
|||||||
|
|
||||||
proc readLogFileForAttsMessages(file: string,
|
proc readLogFileForAttsMessages(file: string,
|
||||||
ignoreErrors = true,
|
ignoreErrors = true,
|
||||||
dumpErrors = false): seq[SlotAttMessage] =
|
dumpErrors = false): seq[SlotMessage] =
|
||||||
var res = newSeq[SlotAttMessage]()
|
var res = newSeq[SlotMessage]()
|
||||||
var stream = newFileStream(file)
|
var stream = newFileStream(file)
|
||||||
var line: string
|
var line: string
|
||||||
var counter = 0
|
var counter = 0
|
||||||
@ -225,13 +287,13 @@ proc readLogFileForAttsMessages(file: string,
|
|||||||
if m.msg == "Attestation sent":
|
if m.msg == "Attestation sent":
|
||||||
let am = Json.decode(line, AttestationSentMessage,
|
let am = Json.decode(line, AttestationSentMessage,
|
||||||
allowUnknownFields = true)
|
allowUnknownFields = true)
|
||||||
let m = SlotAttMessage(kind: SaMessageType.AttestationSent,
|
let m = SlotMessage(kind: SMessageType.AttestationSent,
|
||||||
asmsg: am)
|
asmsg: am)
|
||||||
res.add(m)
|
res.add(m)
|
||||||
elif m.msg == "Slot start":
|
elif m.msg == "Slot start":
|
||||||
let sm = Json.decode(line, SlotStartMessage,
|
let sm = Json.decode(line, SlotStartMessage,
|
||||||
allowUnknownFields = true)
|
allowUnknownFields = true)
|
||||||
let m = SlotAttMessage(kind: SaMessageType.SlotStart,
|
let m = SlotMessage(kind: SMessageType.SlotStart,
|
||||||
ssmsg: sm)
|
ssmsg: sm)
|
||||||
res.add(m)
|
res.add(m)
|
||||||
|
|
||||||
@ -298,6 +360,110 @@ proc readLogFileForASRMessages(file: string, srnode: var SRANode,
|
|||||||
finally:
|
finally:
|
||||||
stream.close()
|
stream.close()
|
||||||
|
|
||||||
|
proc readLogFileForSCMSendMessages(file: string,
|
||||||
|
ignoreErrors = true,
|
||||||
|
dumpErrors = false): seq[SlotMessage] =
|
||||||
|
var res = newSeq[SlotMessage]()
|
||||||
|
var stream = newFileStream(file)
|
||||||
|
var line: string
|
||||||
|
var counter = 0
|
||||||
|
try:
|
||||||
|
while not(stream.atEnd()):
|
||||||
|
line = stream.readLine()
|
||||||
|
inc(counter)
|
||||||
|
var m: LogMessage
|
||||||
|
try:
|
||||||
|
m = Json.decode(line, LogMessage, allowUnknownFields = true)
|
||||||
|
except SerializationError as exc:
|
||||||
|
if dumpErrors:
|
||||||
|
error "Serialization error while reading file, ignoring", file = file,
|
||||||
|
line_number = counter, errorMsg = exc.formatMsg(line)
|
||||||
|
else:
|
||||||
|
error "Serialization error while reading file, ignoring", file = file,
|
||||||
|
line_number = counter
|
||||||
|
if not(ignoreErrors):
|
||||||
|
raise exc
|
||||||
|
else:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if m.msg == "Sync committee message sent":
|
||||||
|
let scmm = Json.decode(line, SCMSentMessage,
|
||||||
|
allowUnknownFields = true)
|
||||||
|
let m = SlotMessage(kind: SMessageType.SCMSent,
|
||||||
|
scmsmsg: scmm)
|
||||||
|
res.add(m)
|
||||||
|
elif m.msg == "Slot start":
|
||||||
|
let sm = Json.decode(line, SlotStartMessage,
|
||||||
|
allowUnknownFields = true)
|
||||||
|
let m = SlotMessage(kind: SMessageType.SlotStart,
|
||||||
|
ssmsg: sm)
|
||||||
|
res.add(m)
|
||||||
|
|
||||||
|
if counter mod 10_000 == 0:
|
||||||
|
info "Processing file", file = extractFilename(file),
|
||||||
|
lines_processed = counter,
|
||||||
|
lines_filtered = len(res)
|
||||||
|
result = res
|
||||||
|
|
||||||
|
except CatchableError as exc:
|
||||||
|
warn "Error reading data from file", file = file, errorMsg = exc.msg
|
||||||
|
finally:
|
||||||
|
stream.close()
|
||||||
|
|
||||||
|
proc readLogFileForSCMSRMessages(file: string, srnode: var SRSCNode,
|
||||||
|
ignoreErrors = true, dumpErrors = false) =
|
||||||
|
var stream = newFileStream(file)
|
||||||
|
var line: string
|
||||||
|
var counter = 0
|
||||||
|
try:
|
||||||
|
while not(stream.atEnd()):
|
||||||
|
var m: LogMessage
|
||||||
|
line = stream.readLine()
|
||||||
|
inc(counter)
|
||||||
|
try:
|
||||||
|
m = Json.decode(line, LogMessage, allowUnknownFields = true)
|
||||||
|
except SerializationError as exc:
|
||||||
|
if dumpErrors:
|
||||||
|
error "Serialization error while reading file, ignoring", file = file,
|
||||||
|
line_number = counter, errorMsg = exc.formatMsg(line)
|
||||||
|
else:
|
||||||
|
error "Serialization error while reading file, ignoring", file = file,
|
||||||
|
line_number = counter
|
||||||
|
if not(ignoreErrors):
|
||||||
|
raise exc
|
||||||
|
else:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if m.msg == "Sync committee message sent":
|
||||||
|
let sm = Json.decode(line, SCMSentMessage,
|
||||||
|
allowUnknownFields = true)
|
||||||
|
srnode.sends.add(sm)
|
||||||
|
elif m.msg == "Sync committee message received":
|
||||||
|
let rm = Json.decode(line, SCMReceivedMessage,
|
||||||
|
allowUnknownFields = true)
|
||||||
|
discard srnode.recvs.hasKeyOrPut(rm.syncCommitteeMsg.signature, rm)
|
||||||
|
|
||||||
|
elif m.msg == "Contribution received":
|
||||||
|
let rm = Json.decode(line, ContributionReceivedMessage,
|
||||||
|
allowUnknownFields = true)
|
||||||
|
discard srnode.contributionRecvs.hasKeyOrPut(rm.signature, rm)
|
||||||
|
|
||||||
|
elif m.msg == "Contribution sent":
|
||||||
|
let sm = Json.decode(line, ContributionSentMessage,
|
||||||
|
allowUnknownFields = true)
|
||||||
|
srnode.contributionSends.add(sm)
|
||||||
|
|
||||||
|
if counter mod 10_000 == 0:
|
||||||
|
info "Processing file", file = extractFilename(file),
|
||||||
|
lines_processed = counter,
|
||||||
|
sends_filtered = len(srnode.sends),
|
||||||
|
recvs_filtered = len(srnode.recvs)
|
||||||
|
|
||||||
|
except CatchableError as exc:
|
||||||
|
warn "Error reading data from file", file = file, errorMsg = exc.msg
|
||||||
|
finally:
|
||||||
|
stream.close()
|
||||||
|
|
||||||
proc readLogFileForSecondMessages(file: string, ignoreErrors = true,
|
proc readLogFileForSecondMessages(file: string, ignoreErrors = true,
|
||||||
dumpErrors = false): seq[LogMessage] =
|
dumpErrors = false): seq[LogMessage] =
|
||||||
var stream = newFileStream(file)
|
var stream = newFileStream(file)
|
||||||
@ -480,10 +646,10 @@ proc runAttSend(logConf: LogTraceConf, logFiles: seq[string]) =
|
|||||||
|
|
||||||
var currentSlot: Option[SlotStartMessage]
|
var currentSlot: Option[SlotStartMessage]
|
||||||
for item in data:
|
for item in data:
|
||||||
if item.kind == SaMessageType.SlotStart:
|
if item.kind == SMessageType.SlotStart:
|
||||||
currentSlot = some(item.ssmsg)
|
currentSlot = some(item.ssmsg)
|
||||||
inc(slotMessagesCount)
|
inc(slotMessagesCount)
|
||||||
elif item.kind == SaMessageType.AttestationSent:
|
elif item.kind == SMessageType.AttestationSent:
|
||||||
if currentSlot.isSome():
|
if currentSlot.isSome():
|
||||||
let attestationTime = currentSlot.get().timestamp -
|
let attestationTime = currentSlot.get().timestamp -
|
||||||
item.asmsg.timestamp
|
item.asmsg.timestamp
|
||||||
@ -606,6 +772,109 @@ proc runAggAttSendReceive(logConf: LogTraceConf, nodes: seq[NodeDirectory]) =
|
|||||||
successful_broadcasts = success, failed_broadcasts = failed,
|
successful_broadcasts = success, failed_broadcasts = failed,
|
||||||
total_broadcasts = len(srnodes[i].aggSends)
|
total_broadcasts = len(srnodes[i].aggSends)
|
||||||
|
|
||||||
|
proc runSCMSendReceive(logConf: LogTraceConf, nodes: seq[NodeDirectory]) =
|
||||||
|
info "Check for Sync Committee Message sent/received messages"
|
||||||
|
if len(nodes) < 2:
|
||||||
|
error "Number of nodes' log files insufficient", nodes_count = len(nodes)
|
||||||
|
quit(1)
|
||||||
|
var srnodes = newSeq[SRSCNode]()
|
||||||
|
|
||||||
|
for node in nodes:
|
||||||
|
var srnode = SRSCNode(
|
||||||
|
directory: node,
|
||||||
|
sends: newSeq[SCMSentMessage](),
|
||||||
|
recvs: newTable[string, SCMReceivedMessage](),
|
||||||
|
contributionSends: newSeq[ContributionSentMessage](),
|
||||||
|
contributionRecvs: newTable[string, ContributionReceivedMessage]()
|
||||||
|
)
|
||||||
|
info "Processing node", node = node.name
|
||||||
|
for logfile in node.logs:
|
||||||
|
let path = node.path & DirSep & logfile
|
||||||
|
info "Processing node's logfile", node = node.name, logfile = path
|
||||||
|
readLogFileForSCMSRMessages(path, srnode,
|
||||||
|
logConf.ignoreSerializationErrors,
|
||||||
|
logConf.dumpSerializationErrors)
|
||||||
|
srnodes.add(srnode)
|
||||||
|
|
||||||
|
if len(nodes) < 2:
|
||||||
|
error "Number of nodes' log files insufficient", nodes_count = len(nodes)
|
||||||
|
quit(1)
|
||||||
|
|
||||||
|
for i in 0 ..< len(srnodes):
|
||||||
|
var success = 0
|
||||||
|
var failed = 0
|
||||||
|
for item in srnodes[i].sends:
|
||||||
|
var k = (i + 1) mod len(srnodes)
|
||||||
|
var misses = newSeq[string]()
|
||||||
|
while k != i:
|
||||||
|
if item.message.signature notin srnodes[k].recvs:
|
||||||
|
misses.add(srnodes[k].directory.name)
|
||||||
|
k = (k + 1) mod len(srnodes)
|
||||||
|
|
||||||
|
if len(misses) == 0:
|
||||||
|
inc(success)
|
||||||
|
else:
|
||||||
|
inc(failed)
|
||||||
|
info "Sync committee message was not received", sender = srnodes[i].directory.name,
|
||||||
|
signature = item.message.signature,
|
||||||
|
receivers = misses.toSimple(), send_stamp = item.timestamp
|
||||||
|
|
||||||
|
info "Statistics for sender node", sender = srnodes[i].directory.name,
|
||||||
|
successful_broadcasts = success, failed_broadcasts = failed,
|
||||||
|
total_broadcasts = len(srnodes[i].sends)
|
||||||
|
|
||||||
|
proc runContributionSendReceive(logConf: LogTraceConf, nodes: seq[NodeDirectory]) =
|
||||||
|
info "Check for contribution sent/received messages"
|
||||||
|
if len(nodes) < 2:
|
||||||
|
error "Number of nodes' log files insufficient", nodes_count = len(nodes)
|
||||||
|
quit(1)
|
||||||
|
var srnodes = newSeq[SRSCNode]()
|
||||||
|
|
||||||
|
for node in nodes:
|
||||||
|
var srnode = SRSCNode(
|
||||||
|
directory: node,
|
||||||
|
sends: newSeq[SCMSentMessage](),
|
||||||
|
recvs: newTable[string, SCMReceivedMessage](),
|
||||||
|
contributionSends: newSeq[ContributionSentMessage](),
|
||||||
|
contributionRecvs: newTable[string, ContributionReceivedMessage]()
|
||||||
|
)
|
||||||
|
info "Processing node", node = node.name
|
||||||
|
for logfile in node.logs:
|
||||||
|
let path = node.path & DirSep & logfile
|
||||||
|
info "Processing node's logfile", node = node.name, logfile = path
|
||||||
|
readLogFileForSCMSRMessages(path, srnode,
|
||||||
|
logConf.ignoreSerializationErrors,
|
||||||
|
logConf.dumpSerializationErrors)
|
||||||
|
srnodes.add(srnode)
|
||||||
|
|
||||||
|
if len(nodes) < 2:
|
||||||
|
error "Number of nodes' log files insufficient", nodes_count = len(nodes)
|
||||||
|
quit(1)
|
||||||
|
|
||||||
|
for i in 0 ..< len(srnodes):
|
||||||
|
var success = 0
|
||||||
|
var failed = 0
|
||||||
|
for item in srnodes[i].contributionSends:
|
||||||
|
var k = (i + 1) mod len(srnodes)
|
||||||
|
var misses = newSeq[string]()
|
||||||
|
while k != i:
|
||||||
|
if item.contribution.signature notin srnodes[k].contributionRecvs:
|
||||||
|
misses.add(srnodes[k].directory.name)
|
||||||
|
k = (k + 1) mod len(srnodes)
|
||||||
|
|
||||||
|
if len(misses) == 0:
|
||||||
|
inc(success)
|
||||||
|
else:
|
||||||
|
inc(failed)
|
||||||
|
info "Contribution was not received",
|
||||||
|
sender = srnodes[i].directory.name,
|
||||||
|
signature = item.contribution.signature,
|
||||||
|
receivers = misses.toSimple(), send_stamp = item.timestamp
|
||||||
|
|
||||||
|
info "Statistics for sender node", sender = srnodes[i].directory.name,
|
||||||
|
successful_broadcasts = success, failed_broadcasts = failed,
|
||||||
|
total_broadcasts = len(srnodes[i].contributionSends)
|
||||||
|
|
||||||
proc runLatencyCheck(logConf: LogTraceConf, logFiles: seq[string],
|
proc runLatencyCheck(logConf: LogTraceConf, logFiles: seq[string],
|
||||||
nodes: seq[NodeDirectory]) =
|
nodes: seq[NodeDirectory]) =
|
||||||
info "Check for async responsiveness"
|
info "Check for async responsiveness"
|
||||||
@ -686,8 +955,20 @@ proc run(conf: LogTraceConf) =
|
|||||||
runAttSendReceive(conf, logNodes)
|
runAttSendReceive(conf, logNodes)
|
||||||
of StartUpCommand.aggasr:
|
of StartUpCommand.aggasr:
|
||||||
runAggAttSendReceive(conf, logNodes)
|
runAggAttSendReceive(conf, logNodes)
|
||||||
|
of StartUpCommand.scmsr:
|
||||||
|
runSCMSendReceive(conf, logNodes)
|
||||||
|
of StartUpCommand.csr:
|
||||||
|
runContributionSendReceive(conf, logNodes)
|
||||||
of StartUpCommand.lat:
|
of StartUpCommand.lat:
|
||||||
runLatencyCheck(conf, logFiles, logNodes)
|
runLatencyCheck(conf, logFiles, logNodes)
|
||||||
|
of StartUpCommand.traceAll:
|
||||||
|
runContributionSendReceive(conf, logNodes)
|
||||||
|
runSCMSendReceive(conf, logNodes)
|
||||||
|
runAggAttSendReceive(conf, logNodes)
|
||||||
|
runAttSendReceive(conf, logNodes)
|
||||||
|
runLatencyCheck(conf, logFiles, logNodes)
|
||||||
|
runPubsub(conf, logFiles)
|
||||||
|
runAttSend(conf, logFiles)
|
||||||
|
|
||||||
when isMainModule:
|
when isMainModule:
|
||||||
echo LogTraceHeader
|
echo LogTraceHeader
|
||||||
|
@ -165,7 +165,7 @@ proc collectEpochRewardsAndPenalties*(
|
|||||||
total_balance = info.balances.current_epoch
|
total_balance = info.balances.current_epoch
|
||||||
total_balance_sqrt = integer_squareroot(total_balance)
|
total_balance_sqrt = integer_squareroot(total_balance)
|
||||||
|
|
||||||
for index, validator in info.validators.pairs:
|
for index, validator in info.validators:
|
||||||
if not is_eligible_validator(validator):
|
if not is_eligible_validator(validator):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
@ -646,7 +646,7 @@ proc cmdValidatorPerf(conf: DbConf, cfg: RuntimeConfig) =
|
|||||||
case info.kind
|
case info.kind
|
||||||
of EpochInfoFork.Phase0:
|
of EpochInfoFork.Phase0:
|
||||||
template info: untyped = info.phase0Data
|
template info: untyped = info.phase0Data
|
||||||
for i, s in info.validators.pairs():
|
for i, s in info.validators:
|
||||||
let perf = addr perfs[i]
|
let perf = addr perfs[i]
|
||||||
if RewardFlags.isActiveInPreviousEpoch in s.flags:
|
if RewardFlags.isActiveInPreviousEpoch in s.flags:
|
||||||
if s.is_previous_epoch_attester.isSome():
|
if s.is_previous_epoch_attester.isSome():
|
||||||
@ -713,7 +713,7 @@ proc cmdValidatorPerf(conf: DbConf, cfg: RuntimeConfig) =
|
|||||||
|
|
||||||
echo "validator_index,attestation_hits,attestation_misses,head_attestation_hits,head_attestation_misses,target_attestation_hits,target_attestation_misses,delay_avg,first_slot_head_attester_when_first_slot_empty,first_slot_head_attester_when_first_slot_not_empty"
|
echo "validator_index,attestation_hits,attestation_misses,head_attestation_hits,head_attestation_misses,target_attestation_hits,target_attestation_misses,delay_avg,first_slot_head_attester_when_first_slot_empty,first_slot_head_attester_when_first_slot_not_empty"
|
||||||
|
|
||||||
for (i, perf) in perfs.pairs:
|
for i, perf in perfs:
|
||||||
var
|
var
|
||||||
count = 0'u64
|
count = 0'u64
|
||||||
sum = 0'u64
|
sum = 0'u64
|
||||||
@ -929,7 +929,7 @@ proc cmdValidatorDb(conf: DbConf, cfg: RuntimeConfig) =
|
|||||||
doAssert state.data.balances.len == previousEpochBalances.len
|
doAssert state.data.balances.len == previousEpochBalances.len
|
||||||
doAssert state.data.balances.len == rewardsAndPenalties.len
|
doAssert state.data.balances.len == rewardsAndPenalties.len
|
||||||
|
|
||||||
for index, validator in info.validators.pairs:
|
for index, validator in info.validators:
|
||||||
template rp: untyped = rewardsAndPenalties[index]
|
template rp: untyped = rewardsAndPenalties[index]
|
||||||
|
|
||||||
checkBalance(index, validator, state.data.balances[index].int64,
|
checkBalance(index, validator, state.data.balances[index].int64,
|
||||||
|
@ -419,7 +419,7 @@ proc prepareRequest(uri: Uri,
|
|||||||
var res: seq[tuple[key: string, value: string]]
|
var res: seq[tuple[key: string, value: string]]
|
||||||
if jheaders.kind != JObject:
|
if jheaders.kind != JObject:
|
||||||
return err("Field `headers` should be an object")
|
return err("Field `headers` should be an object")
|
||||||
for key, value in jheaders.fields.pairs():
|
for key, value in jheaders.fields:
|
||||||
if value.kind != JString:
|
if value.kind != JString:
|
||||||
return err("Field `headers` element should be only strings")
|
return err("Field `headers` element should be only strings")
|
||||||
res.add((key, value.str))
|
res.add((key, value.str))
|
||||||
@ -770,7 +770,7 @@ proc structCmp(j1, j2: JsonNode, strict: bool): bool =
|
|||||||
if strict:
|
if strict:
|
||||||
if len(j1.fields) != len(j2.fields):
|
if len(j1.fields) != len(j2.fields):
|
||||||
return false
|
return false
|
||||||
for key, value in j1.fields.pairs():
|
for key, value in j1.fields:
|
||||||
let j2node = j2.getOrDefault(key)
|
let j2node = j2.getOrDefault(key)
|
||||||
if isNil(j2node):
|
if isNil(j2node):
|
||||||
return false
|
return false
|
||||||
@ -778,7 +778,7 @@ proc structCmp(j1, j2: JsonNode, strict: bool): bool =
|
|||||||
return false
|
return false
|
||||||
true
|
true
|
||||||
else:
|
else:
|
||||||
for key, value in j2.fields.pairs():
|
for key, value in j2.fields:
|
||||||
let j1node = j1.getOrDefault(key)
|
let j1node = j1.getOrDefault(key)
|
||||||
if isNil(j1node):
|
if isNil(j1node):
|
||||||
return false
|
return false
|
||||||
@ -1017,7 +1017,7 @@ proc startTests(conf: RestTesterConf, uri: Uri,
|
|||||||
return 1
|
return 1
|
||||||
res.get()
|
res.get()
|
||||||
|
|
||||||
for index, item in rules.pairs():
|
for index, item in rules:
|
||||||
inputQueue.addLastNoWait(TestCase(index: index, rule: item))
|
inputQueue.addLastNoWait(TestCase(index: index, rule: item))
|
||||||
|
|
||||||
for i in 0 ..< len(workers):
|
for i in 0 ..< len(workers):
|
||||||
@ -1067,7 +1067,7 @@ proc startTests(conf: RestTesterConf, uri: Uri,
|
|||||||
alignLeft("MESSAGE", 20) & "\r\n" &
|
alignLeft("MESSAGE", 20) & "\r\n" &
|
||||||
'-'.repeat(45 + 20 + 7 + 20 + 20)
|
'-'.repeat(45 + 20 + 7 + 20 + 20)
|
||||||
echo headerLine
|
echo headerLine
|
||||||
for index, item in rules.pairs():
|
for index, item in rules:
|
||||||
let errorFlag =
|
let errorFlag =
|
||||||
block:
|
block:
|
||||||
var tmp = "---"
|
var tmp = "---"
|
||||||
|
@ -194,7 +194,7 @@ cli do(slots = SLOTS_PER_EPOCH * 6,
|
|||||||
|
|
||||||
let
|
let
|
||||||
selectionProofSig = get_sync_committee_selection_proof(
|
selectionProofSig = get_sync_committee_selection_proof(
|
||||||
fork, genesis_validators_root, slot, uint64 subcommitteeIdx,
|
fork, genesis_validators_root, slot, subcommitteeIdx,
|
||||||
validatorPrivKey).toValidatorSig
|
validatorPrivKey).toValidatorSig
|
||||||
|
|
||||||
if is_sync_committee_aggregator(selectionProofSig):
|
if is_sync_committee_aggregator(selectionProofSig):
|
||||||
|
@ -244,7 +244,7 @@ cli do(validatorsDir: string, secretsDir: string,
|
|||||||
agg: AggregateSignature
|
agg: AggregateSignature
|
||||||
inited = false
|
inited = false
|
||||||
|
|
||||||
for i, pubkey in pubkeys.pairs():
|
for i, pubkey in pubkeys:
|
||||||
validatorKeys.withValue(pubkey, privkey):
|
validatorKeys.withValue(pubkey, privkey):
|
||||||
let sig = get_sync_committee_message_signature(
|
let sig = get_sync_committee_message_signature(
|
||||||
fork, genesis_validators_root, slot, blockRoot, privkey[])
|
fork, genesis_validators_root, slot, blockRoot, privkey[])
|
||||||
|
@ -91,7 +91,7 @@ CI run: $(basename "$0") --disable-htop -- --verify-finalization
|
|||||||
and validator clients, with all beacon nodes being paired up
|
and validator clients, with all beacon nodes being paired up
|
||||||
with a corresponding validator client)
|
with a corresponding validator client)
|
||||||
--lighthouse-vc-nodes number of Lighthouse VC nodes (assigned before Nimbus VC nodes, default: ${LIGHTHOUSE_VC_NODES})
|
--lighthouse-vc-nodes number of Lighthouse VC nodes (assigned before Nimbus VC nodes, default: ${LIGHTHOUSE_VC_NODES})
|
||||||
--enable-logtrace display logtrace "aggasr" analysis
|
--enable-logtrace display logtrace analysis
|
||||||
--log-level set the log level (default: "${LOG_LEVEL}")
|
--log-level set the log level (default: "${LOG_LEVEL}")
|
||||||
--reuse-existing-data-dir instead of deleting and recreating the data dir, keep it and reuse everything we can from it
|
--reuse-existing-data-dir instead of deleting and recreating the data dir, keep it and reuse everything we can from it
|
||||||
--reuse-binaries don't (re)build the binaries we need and don't delete them at the end (speeds up testing)
|
--reuse-binaries don't (re)build the binaries we need and don't delete them at the end (speeds up testing)
|
||||||
|
@ -242,7 +242,7 @@ suite "Message signatures":
|
|||||||
test "Sync committee selection proof signatures":
|
test "Sync committee selection proof signatures":
|
||||||
let
|
let
|
||||||
slot = default(Slot)
|
slot = default(Slot)
|
||||||
subcommittee_index = default(uint64)
|
subcommittee_index = default(SyncSubcommitteeIndex)
|
||||||
|
|
||||||
check:
|
check:
|
||||||
# Matching public/private keys and genesis validator roots
|
# Matching public/private keys and genesis validator roots
|
||||||
|
@ -304,6 +304,7 @@ proc makeSyncAggregate(
|
|||||||
cfg: RuntimeConfig): SyncAggregate =
|
cfg: RuntimeConfig): SyncAggregate =
|
||||||
if syncCommitteeRatio <= 0.0:
|
if syncCommitteeRatio <= 0.0:
|
||||||
return SyncAggregate.init()
|
return SyncAggregate.init()
|
||||||
|
|
||||||
let
|
let
|
||||||
syncCommittee =
|
syncCommittee =
|
||||||
withState(state):
|
withState(state):
|
||||||
@ -323,11 +324,13 @@ proc makeSyncAggregate(
|
|||||||
latest_block_root =
|
latest_block_root =
|
||||||
withState(state): state.latest_block_root
|
withState(state): state.latest_block_root
|
||||||
syncCommitteePool = newClone(SyncCommitteeMsgPool.init(keys.newRng()))
|
syncCommitteePool = newClone(SyncCommitteeMsgPool.init(keys.newRng()))
|
||||||
|
|
||||||
type
|
type
|
||||||
Aggregator = object
|
Aggregator = object
|
||||||
subcommitteeIdx: SyncSubcommitteeIndex
|
subcommitteeIdx: SyncSubcommitteeIndex
|
||||||
validatorIdx: ValidatorIndex
|
validatorIdx: ValidatorIndex
|
||||||
selectionProof: ValidatorSig
|
selectionProof: ValidatorSig
|
||||||
|
|
||||||
var aggregators: seq[Aggregator]
|
var aggregators: seq[Aggregator]
|
||||||
for subcommitteeIdx in SyncSubcommitteeIndex:
|
for subcommitteeIdx in SyncSubcommitteeIndex:
|
||||||
let
|
let
|
||||||
@ -357,8 +360,9 @@ proc makeSyncAggregate(
|
|||||||
MockPrivKeys[validatorIdx])
|
MockPrivKeys[validatorIdx])
|
||||||
selectionProofSig = get_sync_committee_selection_proof(
|
selectionProofSig = get_sync_committee_selection_proof(
|
||||||
fork, genesis_validators_root,
|
fork, genesis_validators_root,
|
||||||
slot, subcommitteeIdx.uint64,
|
slot, subcommitteeIdx,
|
||||||
MockPrivKeys[validatorIdx])
|
MockPrivKeys[validatorIdx])
|
||||||
|
|
||||||
syncCommitteePool[].addSyncCommitteeMessage(
|
syncCommitteePool[].addSyncCommitteeMessage(
|
||||||
slot,
|
slot,
|
||||||
latest_block_root,
|
latest_block_root,
|
||||||
@ -366,11 +370,13 @@ proc makeSyncAggregate(
|
|||||||
signature,
|
signature,
|
||||||
subcommitteeIdx,
|
subcommitteeIdx,
|
||||||
positions)
|
positions)
|
||||||
|
|
||||||
if is_sync_committee_aggregator(selectionProofSig.toValidatorSig):
|
if is_sync_committee_aggregator(selectionProofSig.toValidatorSig):
|
||||||
aggregators.add Aggregator(
|
aggregators.add Aggregator(
|
||||||
subcommitteeIdx: subcommitteeIdx,
|
subcommitteeIdx: subcommitteeIdx,
|
||||||
validatorIdx: validatorIdx,
|
validatorIdx: validatorIdx,
|
||||||
selectionProof: selectionProofSig.toValidatorSig)
|
selectionProof: selectionProofSig.toValidatorSig)
|
||||||
|
|
||||||
for aggregator in aggregators:
|
for aggregator in aggregators:
|
||||||
var contribution: SyncCommitteeContribution
|
var contribution: SyncCommitteeContribution
|
||||||
if syncCommitteePool[].produceContribution(
|
if syncCommitteePool[].produceContribution(
|
||||||
@ -389,6 +395,7 @@ proc makeSyncAggregate(
|
|||||||
signature: contributionSig.toValidatorSig)
|
signature: contributionSig.toValidatorSig)
|
||||||
syncCommitteePool[].addContribution(
|
syncCommitteePool[].addContribution(
|
||||||
signedContributionAndProof, contribution.signature.load.get)
|
signedContributionAndProof, contribution.signature.load.get)
|
||||||
|
|
||||||
syncCommitteePool[].produceSyncAggregate(latest_block_root)
|
syncCommitteePool[].produceSyncAggregate(latest_block_root)
|
||||||
|
|
||||||
iterator makeTestBlocks*(
|
iterator makeTestBlocks*(
|
||||||
|
Loading…
x
Reference in New Issue
Block a user