Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

PART 3 - GET/Attestation Pool API - Add API interface #8438

Merged
merged 19 commits into from
Jul 26, 2024
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
- Added a state pruner that can limit the number of finalized states stored when running an archive node.
- Updated bootnodes for Sepolia network.
- Implemented [GetBlockAttestationV2](https://ethereum.github.io/beacon-APIs/?urls.primaryName=dev#/Beacon/getBlockAttestationsV2) (adding support for Electra attestations)
- Implemented [GetAttestationsV2](https://ethereum.github.io/beacon-APIs/?urls.primaryName=dev#/Beacon/getPoolAttestationsV2) (adding support for Electra attestations)
- Implemented [GetAggregateAttestationV2](https://ethereum.github.io/beacon-APIs/?urls.primaryName=dev#/Validator/getAggregatedAttestationV2) (adding support for Electra attestations)
- Updated a number of parameters to reduce issues when using `p2p-subscribe-all-subnets-enabled`. If you have adjusted queue sizes manually when using all-subnets, please refer to details below. Manual settings will still override these defaults.
- When `p2p-subscribe-all-subnets-enabled`, `p2p-peer-lower-bound` now defaults to 60 (previously 64), and `p2p-peer-upper-bound` now defaults to 80 (previously 100).
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -77,7 +77,7 @@ void setup() throws Exception {

@Test
void shouldPassMergeOptimisticallyAndBeginFinalizationAfterSafeSlotsToImport() throws Exception {
tekuNode2.waitForNonDefaultExecutionPayload();
tekuNode2.waitForNonDefaultExecutionPayload(10);
tekuNode2.waitForOptimisticBlock();

// Now make execution node sync and clarify switch from optimistic sync back to the normal
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -461,6 +461,10 @@ public void waitForMilestone(final SpecMilestone expectedMilestone) {
}

public void waitForNonDefaultExecutionPayload() {
waitForNonDefaultExecutionPayload(5);
}

public void waitForNonDefaultExecutionPayload(final int timeoutInMinutes) {
LOG.debug("Wait for a block containing a non default execution payload");

waitFor(
Expand All @@ -469,7 +473,7 @@ public void waitForNonDefaultExecutionPayload() {
assertThat(block).isPresent();
checkExecutionPayloadInBlock(block.get());
},
5,
timeoutInMinutes,
MINUTES);
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
"operationId" : "getPoolAttestations",
"summary" : "Get Attestations from operations pool",
"description" : "Retrieves attestations known by the node but not necessarily incorporated into any block.",
"deprecated" : true,
"parameters" : [ {
"name" : "slot",
"in" : "query",
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
{
"get" : {
"tags" : [ "Beacon" ],
"operationId" : "getPoolAttestationsV2",
"summary" : "Get Attestations from operations pool",
"description" : "Retrieves attestations known by the node but not necessarily incorporated into any block.",
"parameters" : [ {
"name" : "slot",
"in" : "query",
"schema" : {
"type" : "string",
"description" : "`UInt64` Slot to query in the canonical chain.",
"example" : "1",
"format" : "uint64"
}
}, {
"name" : "committee_index",
"in" : "query",
"schema" : {
"type" : "string",
"description" : "`uint64` Committee index to query.",
"example" : "1",
"format" : "uint64"
}
} ],
"responses" : {
"200" : {
"description" : "Request successful",
"content" : {
"application/json" : {
"schema" : {
"$ref" : "#/components/schemas/GetPoolAttestationsV2Response"
}
}
}
},
"400" : {
"description" : "The request could not be processed, check the response for more information.",
"content" : {
"application/json" : {
"schema" : {
"$ref" : "#/components/schemas/HttpErrorResponse"
}
}
}
},
"500" : {
"description" : "Internal server error",
"content" : {
"application/json" : {
"schema" : {
"$ref" : "#/components/schemas/HttpErrorResponse"
}
}
}
}
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
{
"title" : "GetPoolAttestationsV2Response",
"type" : "object",
"required" : [ "version", "data" ],
"properties" : {
"version" : {
"type" : "string",
"enum" : [ "phase0", "altair", "bellatrix", "capella", "deneb", "electra" ]
},
"data" : {
"type" : "array",
"items" : {
"title" : "Attestation",
"type" : "object",
"oneOf" : [ {
"$ref" : "#/components/schemas/AttestationPhase0"
}, {
"$ref" : "#/components/schemas/AttestationElectra"
} ]
}
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,7 @@
import tech.pegasys.teku.beaconrestapi.handlers.v1.validator.PostSyncCommitteeSubscriptions;
import tech.pegasys.teku.beaconrestapi.handlers.v1.validator.PostSyncDuties;
import tech.pegasys.teku.beaconrestapi.handlers.v1.validator.PostValidatorLiveness;
import tech.pegasys.teku.beaconrestapi.handlers.v2.beacon.GetAttestationsV2;
import tech.pegasys.teku.beaconrestapi.handlers.v2.beacon.GetAttesterSlashingsV2;
import tech.pegasys.teku.beaconrestapi.handlers.v2.beacon.GetBlock;
import tech.pegasys.teku.beaconrestapi.handlers.v2.beacon.GetBlockAttestationsV2;
Expand Down Expand Up @@ -236,6 +237,7 @@ private static RestApi create(
.endpoint(new GetBlockAttestations(dataProvider, spec))
.endpoint(new GetBlockAttestationsV2(dataProvider, schemaCache))
.endpoint(new GetAttestations(dataProvider, spec))
.endpoint(new GetAttestationsV2(dataProvider, schemaCache))
.endpoint(new PostAttestation(dataProvider, schemaCache))
.endpoint(new GetAttesterSlashings(dataProvider, spec))
.endpoint(new GetAttesterSlashingsV2(dataProvider, schemaCache))
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@ public GetAttestations(final NodeDataProvider nodeDataProvider, final Spec spec)
.description(
"Retrieves attestations known by the node but not necessarily incorporated into any block.")
.tags(TAG_BEACON)
.deprecated(true)
.queryParam(SLOT_PARAMETER.withDescription(SLOT_QUERY_DESCRIPTION))
.queryParam(COMMITTEE_INDEX_PARAMETER)
.response(SC_OK, "Request successful", getResponseType(spec))
Expand All @@ -70,7 +71,6 @@ public void handleRequest(final RestApiRequest request) throws JsonProcessingExc
request.respondOk(nodeDataProvider.getAttestations(slot, committeeIndex));
}

// TODO EIP-7549 handle Electra attestations
private static SerializableTypeDefinition<List<Attestation>> getResponseType(final Spec spec) {
return SerializableTypeDefinition.<List<Attestation>>object()
.name("GetPoolAttestationsResponse")
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,123 @@
/*
* Copyright Consensys Software Inc., 2024
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on
* an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the
* specific language governing permissions and limitations under the License.
*/

package tech.pegasys.teku.beaconrestapi.handlers.v2.beacon;

import static tech.pegasys.teku.beaconrestapi.BeaconRestApiTypes.COMMITTEE_INDEX_PARAMETER;
import static tech.pegasys.teku.beaconrestapi.BeaconRestApiTypes.SLOT_PARAMETER;
import static tech.pegasys.teku.beaconrestapi.handlers.v1.beacon.MilestoneDependentTypesUtil.getMultipleSchemaDefinitionFromMilestone;
import static tech.pegasys.teku.ethereum.json.types.EthereumTypes.MILESTONE_TYPE;
import static tech.pegasys.teku.infrastructure.http.HttpStatusCodes.SC_OK;
import static tech.pegasys.teku.infrastructure.http.RestApiConstants.CACHE_NONE;
import static tech.pegasys.teku.infrastructure.http.RestApiConstants.HEADER_CONSENSUS_VERSION;
import static tech.pegasys.teku.infrastructure.http.RestApiConstants.SLOT_QUERY_DESCRIPTION;
import static tech.pegasys.teku.infrastructure.http.RestApiConstants.TAG_BEACON;
import static tech.pegasys.teku.infrastructure.json.types.SerializableTypeDefinition.listOf;

import com.fasterxml.jackson.core.JsonProcessingException;
import io.javalin.http.Header;
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import tech.pegasys.teku.api.DataProvider;
import tech.pegasys.teku.api.NodeDataProvider;
import tech.pegasys.teku.api.schema.Version;
import tech.pegasys.teku.beaconrestapi.handlers.v1.beacon.MilestoneDependentTypesUtil;
import tech.pegasys.teku.infrastructure.json.types.SerializableTypeDefinition;
import tech.pegasys.teku.infrastructure.restapi.endpoints.EndpointMetadata;
import tech.pegasys.teku.infrastructure.restapi.endpoints.RestApiEndpoint;
import tech.pegasys.teku.infrastructure.restapi.endpoints.RestApiRequest;
import tech.pegasys.teku.infrastructure.unsigned.UInt64;
import tech.pegasys.teku.spec.SpecMilestone;
import tech.pegasys.teku.spec.datastructures.metadata.ObjectAndMetaData;
import tech.pegasys.teku.spec.datastructures.operations.Attestation;
import tech.pegasys.teku.spec.schemas.SchemaDefinitionCache;
import tech.pegasys.teku.spec.schemas.SchemaDefinitions;

public class GetAttestationsV2 extends RestApiEndpoint {

public static final String ROUTE = "/eth/v2/beacon/pool/attestations";

private final NodeDataProvider nodeDataProvider;

public GetAttestationsV2(
final DataProvider dataProvider, final SchemaDefinitionCache schemaDefinitionCache) {

this(dataProvider.getNodeDataProvider(), schemaDefinitionCache);
}

public GetAttestationsV2(
final NodeDataProvider nodeDataProvider, final SchemaDefinitionCache schemaDefinitionCache) {
super(
EndpointMetadata.get(ROUTE)
.operationId("getPoolAttestationsV2")
.summary("Get Attestations from operations pool")
.description(
"Retrieves attestations known by the node but not necessarily incorporated into any block.")
.tags(TAG_BEACON)
.queryParam(SLOT_PARAMETER.withDescription(SLOT_QUERY_DESCRIPTION))
.queryParam(COMMITTEE_INDEX_PARAMETER)
.response(SC_OK, "Request successful", getResponseType(schemaDefinitionCache))
.build());
this.nodeDataProvider = nodeDataProvider;
}

@Override
public void handleRequest(final RestApiRequest request) throws JsonProcessingException {
request.header(Header.CACHE_CONTROL, CACHE_NONE);
final Optional<UInt64> slot =
request.getOptionalQueryParameter(SLOT_PARAMETER.withDescription(SLOT_QUERY_DESCRIPTION));
final Optional<UInt64> committeeIndex =
request.getOptionalQueryParameter(COMMITTEE_INDEX_PARAMETER);
final ObjectAndMetaData<List<Attestation>> attestationsAndMetaData =
nodeDataProvider.getAttestationsAndMetaData(slot, committeeIndex);

request.header(
HEADER_CONSENSUS_VERSION,
Version.fromMilestone(attestationsAndMetaData.getMilestone()).name());
request.respondOk(attestationsAndMetaData);
}

private static SerializableTypeDefinition<ObjectAndMetaData<List<Attestation>>> getResponseType(
final SchemaDefinitionCache schemaDefinitionCache) {

final List<MilestoneDependentTypesUtil.ConditionalSchemaGetter<Attestation>> schemaGetters =
generateAttestationSchemaGetters(schemaDefinitionCache);

final SerializableTypeDefinition<Attestation> attestationType =
getMultipleSchemaDefinitionFromMilestone(
schemaDefinitionCache, "Attestation", schemaGetters);

return SerializableTypeDefinition.<ObjectAndMetaData<List<Attestation>>>object()
.name("GetPoolAttestationsV2Response")
.withField("version", MILESTONE_TYPE, ObjectAndMetaData::getMilestone)
.withField("data", listOf(attestationType), ObjectAndMetaData::getData)
.build();
}

private static List<MilestoneDependentTypesUtil.ConditionalSchemaGetter<Attestation>>
generateAttestationSchemaGetters(final SchemaDefinitionCache schemaDefinitionCache) {
final List<MilestoneDependentTypesUtil.ConditionalSchemaGetter<Attestation>> schemaGetterList =
new ArrayList<>();

schemaGetterList.add(
new MilestoneDependentTypesUtil.ConditionalSchemaGetter<>(
(attestation, milestone) ->
schemaDefinitionCache
.milestoneAtSlot(attestation.getData().getSlot())
.equals(milestone),
SpecMilestone.PHASE0,
SchemaDefinitions::getAttestationSchema));
return schemaGetterList;
}
}
Loading
Loading