Skip to content

Commit

Permalink
Schema registry interface improvement (#8696)
Browse files Browse the repository at this point in the history

Co-authored-by: Enrico Del Fante <enrico.delfante@consensys.net>
  • Loading branch information
zilm13 and tbenr authored Oct 14, 2024
1 parent 28ac0b4 commit 289561c
Show file tree
Hide file tree
Showing 8 changed files with 323 additions and 260 deletions.

This file was deleted.

This file was deleted.

This file was deleted.

Original file line number Diff line number Diff line change
@@ -0,0 +1,153 @@
/*
* 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.spec.schemas.registry;

import static com.google.common.base.Preconditions.checkArgument;

import com.google.common.base.MoreObjects;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;
import java.util.TreeMap;
import java.util.function.BiFunction;
import tech.pegasys.teku.spec.SpecMilestone;
import tech.pegasys.teku.spec.config.SpecConfig;
import tech.pegasys.teku.spec.schemas.registry.SchemaTypes.SchemaId;

class BaseSchemaProvider<T> implements SchemaProvider<T> {
private final TreeMap<SpecMilestone, SchemaProviderCreator<T>> milestoneToSchemaCreator =
new TreeMap<>();
private final SchemaId<T> schemaId;

private BaseSchemaProvider(
final SchemaId<T> schemaId,
final List<SchemaProviderCreator<T>> schemaProviderCreators,
final SpecMilestone untilMilestone,
final boolean isConstant) {
this.schemaId = schemaId;
final List<SchemaProviderCreator<T>> creatorsList = new ArrayList<>(schemaProviderCreators);

SchemaProviderCreator<T> lastCreator = null;

for (final SpecMilestone milestone : SpecMilestone.getMilestonesUpTo(untilMilestone)) {
if (!creatorsList.isEmpty() && creatorsList.getFirst().milestone == milestone) {
lastCreator = creatorsList.removeFirst();
}

if (lastCreator != null) {
milestoneToSchemaCreator.put(
milestone, isConstant ? lastCreator : lastCreator.withMilestone(milestone));
}
}
}

@Override
public SpecMilestone getEffectiveMilestone(final SpecMilestone milestone) {
return getSchemaCreator(milestone).milestone;
}

@Override
public T getSchema(final SchemaRegistry registry) {
final SpecMilestone milestone = registry.getMilestone();
return createSchema(registry, milestone, registry.getSpecConfig());
}

@Override
public SchemaId<T> getSchemaId() {
return schemaId;
}

protected T createSchema(
final SchemaRegistry registry,
final SpecMilestone effectiveMilestone,
final SpecConfig specConfig) {
return getSchemaCreator(effectiveMilestone).creator.apply(registry, specConfig);
}

private SchemaProviderCreator<T> getSchemaCreator(final SpecMilestone milestone) {
final SchemaProviderCreator<T> maybeSchemaCreator = milestoneToSchemaCreator.get(milestone);
if (maybeSchemaCreator == null) {
throw new IllegalArgumentException(
"It is not supposed to create a specific version for " + milestone);
}
return maybeSchemaCreator;
}

@Override
public Set<SpecMilestone> getSupportedMilestones() {
return milestoneToSchemaCreator.keySet();
}

protected record SchemaProviderCreator<T>(
SpecMilestone milestone, BiFunction<SchemaRegistry, SpecConfig, T> creator) {

private SchemaProviderCreator<T> withMilestone(final SpecMilestone milestone) {
return new SchemaProviderCreator<>(milestone, creator);
}

@Override
public String toString() {
return MoreObjects.toStringHelper(this).add("milestone", milestone).toString();
}
}

static <T> Builder<T> providerBuilder(final SchemaId<T> schemaId) {
return new Builder<>(schemaId);
}

static class Builder<T> {
private final SchemaId<T> schemaId;
final List<SchemaProviderCreator<T>> schemaProviderCreators = new ArrayList<>();
private SpecMilestone untilMilestone = SpecMilestone.getHighestMilestone();
private boolean isConstant = false;

private Builder(final SchemaId<T> schemaId) {
this.schemaId = schemaId;
}

public Builder<T> withCreator(
final SpecMilestone milestone,
final BiFunction<SchemaRegistry, SpecConfig, T> creationSchema) {
checkArgument(
schemaProviderCreators.isEmpty()
|| milestone.isGreaterThan(schemaProviderCreators.getLast().milestone),
"Creator's milestones must added in strict ascending order for %s",
schemaId);

schemaProviderCreators.add(new SchemaProviderCreator<>(milestone, creationSchema));
return this;
}

public Builder<T> until(final SpecMilestone untilMilestone) {
this.untilMilestone = untilMilestone;
return this;
}

public Builder<T> constant() {
this.isConstant = true;
return this;
}

public BaseSchemaProvider<T> build() {
checkArgument(
!schemaProviderCreators.isEmpty(), "There should be at least 1 creator for %s", schemaId);

checkArgument(
untilMilestone.isGreaterThanOrEqualTo(schemaProviderCreators.getLast().milestone),
"until must be greater or equal than last creator milestone in %s",
schemaId);
return new BaseSchemaProvider<>(schemaId, schemaProviderCreators, untilMilestone, isConstant);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -13,35 +13,11 @@

package tech.pegasys.teku.spec.schemas.registry;

import static tech.pegasys.teku.spec.SpecMilestone.BELLATRIX;
import static tech.pegasys.teku.spec.SpecMilestone.CAPELLA;
import static tech.pegasys.teku.spec.SpecMilestone.DENEB;
import static tech.pegasys.teku.spec.SpecMilestone.ELECTRA;

import java.util.EnumSet;
import java.util.Set;
import tech.pegasys.teku.spec.SpecMilestone;
import tech.pegasys.teku.spec.schemas.registry.SchemaTypes.SchemaId;

interface SchemaProvider<T> {
Set<SpecMilestone> ALL_MILESTONES = EnumSet.allOf(SpecMilestone.class);
Set<SpecMilestone> FROM_BELLATRIX = from(BELLATRIX);
Set<SpecMilestone> FROM_CAPELLA = from(CAPELLA);
Set<SpecMilestone> FROM_DENEB = from(DENEB);
Set<SpecMilestone> FROM_ELECTRA = from(ELECTRA);

static Set<SpecMilestone> from(final SpecMilestone milestone) {
return EnumSet.copyOf(SpecMilestone.getAllMilestonesFrom(milestone));
}

static Set<SpecMilestone> fromTo(
final SpecMilestone fromMilestone, final SpecMilestone toMilestone) {
return EnumSet.copyOf(
SpecMilestone.getAllMilestonesFrom(fromMilestone).stream()
.filter(toMilestone::isLessThanOrEqualTo)
.toList());
}

T getSchema(SchemaRegistry registry);

Set<SpecMilestone> getSupportedMilestones();
Expand Down
Loading

0 comments on commit 289561c

Please sign in to comment.