From 2fcf666b5b90119f0fdeb353bf92ebbba1829fc7 Mon Sep 17 00:00:00 2001 From: Enrico Del Fante Date: Mon, 9 Dec 2024 14:13:08 +0100 Subject: [PATCH] fix --- .../coordinator/ValidatorApiHandler.java | 14 ++++-- .../DefaultPerformanceTracker.java | 2 + .../coordinator/ValidatorApiHandlerTest.java | 47 +++++++++++++++++++ 3 files changed, 59 insertions(+), 4 deletions(-) diff --git a/beacon/validator/src/main/java/tech/pegasys/teku/validator/coordinator/ValidatorApiHandler.java b/beacon/validator/src/main/java/tech/pegasys/teku/validator/coordinator/ValidatorApiHandler.java index 04648dd04fb..fa9004a03ca 100644 --- a/beacon/validator/src/main/java/tech/pegasys/teku/validator/coordinator/ValidatorApiHandler.java +++ b/beacon/validator/src/main/java/tech/pegasys/teku/validator/coordinator/ValidatorApiHandler.java @@ -586,16 +586,22 @@ public SafeFuture> sendSignedAttestations( } private SafeFuture processAttestation(final Attestation attestation) { + final ValidatableAttestation validatableAttestation = + ValidatableAttestation.fromValidator(spec, attestation); return attestationManager - .addAttestation(ValidatableAttestation.fromValidator(spec, attestation), Optional.empty()) + .addAttestation(validatableAttestation, Optional.empty()) .thenPeek( result -> { + // when saving the attestation in performance tracker, we want to make sure we save + // the converted attestation + // conversion happens during processing and is saved in the validatable attestation + final Attestation convertedAttestation = validatableAttestation.getAttestation(); if (!result.isReject()) { - dutyMetrics.onAttestationPublished(attestation.getData().getSlot()); - performanceTracker.saveProducedAttestation(attestation); + dutyMetrics.onAttestationPublished(convertedAttestation.getData().getSlot()); + performanceTracker.saveProducedAttestation(convertedAttestation); } else { VALIDATOR_LOGGER.producedInvalidAttestation( - attestation.getData().getSlot(), + convertedAttestation.getData().getSlot(), result.getDescription().orElse("Unknown reason")); } }) diff --git a/beacon/validator/src/main/java/tech/pegasys/teku/validator/coordinator/performance/DefaultPerformanceTracker.java b/beacon/validator/src/main/java/tech/pegasys/teku/validator/coordinator/performance/DefaultPerformanceTracker.java index 381229ffca5..8031f4f0a30 100644 --- a/beacon/validator/src/main/java/tech/pegasys/teku/validator/coordinator/performance/DefaultPerformanceTracker.java +++ b/beacon/validator/src/main/java/tech/pegasys/teku/validator/coordinator/performance/DefaultPerformanceTracker.java @@ -14,6 +14,7 @@ package tech.pegasys.teku.validator.coordinator.performance; import static com.google.common.base.Preconditions.checkArgument; +import static com.google.common.base.Preconditions.checkState; import com.google.common.annotations.VisibleForTesting; import it.unimi.dsi.fastutil.ints.Int2IntMap; @@ -424,6 +425,7 @@ private SafeFuture>> getAttestationsIncludedInEpoc @Override public void saveProducedAttestation(final Attestation attestation) { + checkState(!attestation.isSingleAttestation(), "Single attestation is not supported"); final UInt64 epoch = spec.computeEpochAtSlot(attestation.getData().getSlot()); final Set attestationsInEpoch = producedAttestationsByEpoch.computeIfAbsent(epoch, __ -> concurrentSet()); diff --git a/beacon/validator/src/test/java/tech/pegasys/teku/validator/coordinator/ValidatorApiHandlerTest.java b/beacon/validator/src/test/java/tech/pegasys/teku/validator/coordinator/ValidatorApiHandlerTest.java index d882571c41d..770ecbad932 100644 --- a/beacon/validator/src/test/java/tech/pegasys/teku/validator/coordinator/ValidatorApiHandlerTest.java +++ b/beacon/validator/src/test/java/tech/pegasys/teku/validator/coordinator/ValidatorApiHandlerTest.java @@ -819,6 +819,53 @@ public void sendSignedAttestations_shouldAddAttestationToAttestationManager() { .addAttestation(ValidatableAttestation.from(spec, attestation), Optional.empty()); } + @Test + void sendSignedAttestations_shouldSaveConvertedAttestationFromSingleAttestation() { + spec = TestSpecFactory.createMinimalElectra(); + dataStructureUtil = new DataStructureUtil(spec); + validatorApiHandler = + new ValidatorApiHandler( + chainDataProvider, + nodeDataProvider, + networkDataProvider, + chainDataClient, + syncStateProvider, + blockFactory, + attestationPool, + attestationManager, + attestationTopicSubscriptions, + activeValidatorTracker, + dutyMetrics, + performanceTracker, + spec, + forkChoiceTrigger, + proposersDataManager, + syncCommitteeMessagePool, + syncCommitteeContributionPool, + syncCommitteeSubscriptionManager, + blockProductionPerformanceFactory, + blockPublisher); + + final Attestation attestation = dataStructureUtil.randomSingleAttestation(); + final Attestation convertedAttestation = dataStructureUtil.randomAttestation(); + doAnswer( + invocation -> { + invocation + .getArgument(0, ValidatableAttestation.class) + .convertToAggregatedFormatFromSingleAttestation(convertedAttestation); + return completedFuture(InternalValidationResult.ACCEPT); + }) + .when(attestationManager) + .addAttestation(any(ValidatableAttestation.class), any()); + + final SafeFuture> result = + validatorApiHandler.sendSignedAttestations(List.of(attestation)); + assertThat(result).isCompletedWithValue(emptyList()); + + verify(dutyMetrics).onAttestationPublished(convertedAttestation.getData().getSlot()); + verify(performanceTracker).saveProducedAttestation(convertedAttestation); + } + @Test void sendSignedAttestations_shouldAddToDutyMetricsAndPerformanceTrackerWhenNotInvalid() { final Attestation attestation = dataStructureUtil.randomAttestation();