diff --git a/atlasdb-config/src/test/java/com/palantir/atlasdb/factory/LeadersTest.java b/atlasdb-config/src/test/java/com/palantir/atlasdb/factory/LeadersTest.java index dd109d15101..5a3069fc796 100644 --- a/atlasdb-config/src/test/java/com/palantir/atlasdb/factory/LeadersTest.java +++ b/atlasdb-config/src/test/java/com/palantir/atlasdb/factory/LeadersTest.java @@ -46,8 +46,8 @@ public class LeadersTest { @Test public void canCreateProxyAndLocalListOfPaxosLearners() { PaxosLearner localLearner = mock(PaxosLearner.class); - PaxosValue value = mock(PaxosValue.class); - when(localLearner.getGreatestLearnedValue()).thenReturn(value); + Optional presentPaxosValue = Optional.of(mock(PaxosValue.class)); + when(localLearner.getGreatestLearnedValue()).thenReturn(presentPaxosValue); List paxosLearners = Leaders.createProxyAndLocalList( new MetricRegistry(), @@ -59,7 +59,7 @@ public void canCreateProxyAndLocalListOfPaxosLearners() { MatcherAssert.assertThat(paxosLearners.size(), is(REMOTE_SERVICE_ADDRESSES.size() + 1)); paxosLearners.forEach(object -> MatcherAssert.assertThat(object, not(nullValue()))); - MatcherAssert.assertThat(Iterables.getLast(paxosLearners).getGreatestLearnedValue(), is(value)); + MatcherAssert.assertThat(Iterables.getLast(paxosLearners).getGreatestLearnedValue(), is(presentPaxosValue)); verify(localLearner).getGreatestLearnedValue(); verifyNoMoreInteractions(localLearner); } diff --git a/changelog/@unreleased/pr-4361.v2.yml b/changelog/@unreleased/pr-4361.v2.yml new file mode 100644 index 00000000000..326368aecf5 --- /dev/null +++ b/changelog/@unreleased/pr-4361.v2.yml @@ -0,0 +1,9 @@ +type: break +break: + description: The `PaxosLearner` interface now returns `Optional` instead + of a nullable `PaxosValue` for the `getLearnedValue(seq)` and `getGreatestLearnedValue()` + methods. The wire format remains unchanged (so, for example, rolling upgrades + of TimeLock, or servers in an embedded leader configuration, are safe). Users + who require the old behaviour can use `Optional.orElse(null)`. + links: + - https://github.com/palantir/atlasdb/pull/4361 diff --git a/leader-election-api/src/main/java/com/palantir/paxos/PaxosLearner.java b/leader-election-api/src/main/java/com/palantir/paxos/PaxosLearner.java index 7e757e0adde..4268a136115 100644 --- a/leader-election-api/src/main/java/com/palantir/paxos/PaxosLearner.java +++ b/leader-election-api/src/main/java/com/palantir/paxos/PaxosLearner.java @@ -16,9 +16,8 @@ package com.palantir.paxos; import java.util.Collection; +import java.util.Optional; -import javax.annotation.Nonnull; -import javax.annotation.Nullable; import javax.ws.rs.Consumes; import javax.ws.rs.GET; import javax.ws.rs.POST; @@ -44,22 +43,21 @@ public interface PaxosLearner { void learn(@PathParam("seq") long seq, PaxosValue val); /** - * Returns learned value or null if non-exists. + * Returns the learned value at the specified sequence number, or {@link Optional#empty()} if the learner + * does not know such a value. */ - @Nullable @GET @Path("learned-value/{seq:.+}") @Produces(MediaType.APPLICATION_JSON) - PaxosValue getLearnedValue(@PathParam("seq") long seq); + Optional getLearnedValue(@PathParam("seq") long seq); /** - * Returns the learned value for the greatest known round or null if nothing has been learned. + * Returns the learned value for the greatest known round or {@link Optional#empty()} if nothing has been learned. */ - @Nullable @GET @Path("greatest-learned-value") @Produces(MediaType.APPLICATION_JSON) - PaxosValue getGreatestLearnedValue(); + Optional getGreatestLearnedValue(); /** * Returns some collection of learned values since the seq-th round (inclusive). @@ -67,7 +65,6 @@ public interface PaxosLearner { * @param seq lower round cutoff for returned values * @return some set of learned values for rounds since the seq-th round */ - @Nonnull @GET @Path("learned-values-since/{seq:.+}") @Produces(MediaType.APPLICATION_JSON) diff --git a/leader-election-impl/src/main/java/com/palantir/leader/LocalPingableLeader.java b/leader-election-impl/src/main/java/com/palantir/leader/LocalPingableLeader.java index a6dda6be966..189932b2fbb 100644 --- a/leader-election-impl/src/main/java/com/palantir/leader/LocalPingableLeader.java +++ b/leader-election-impl/src/main/java/com/palantir/leader/LocalPingableLeader.java @@ -16,7 +16,6 @@ package com.palantir.leader; -import java.util.Optional; import java.util.UUID; import com.palantir.paxos.PaxosLearner; @@ -34,7 +33,7 @@ public LocalPingableLeader(PaxosLearner knowledge, UUID localUuid) { @Override public boolean ping() { - return getGreatestLearnedPaxosValue() + return knowledge.getGreatestLearnedValue() .map(this::isThisNodeTheLeaderFor) .orElse(false); } @@ -44,10 +43,6 @@ public String getUUID() { return localUuid.toString(); } - private Optional getGreatestLearnedPaxosValue() { - return Optional.ofNullable(knowledge.getGreatestLearnedValue()); - } - private boolean isThisNodeTheLeaderFor(PaxosValue value) { return value.getLeaderUUID().equals(localUuid.toString()); } diff --git a/leader-election-impl/src/main/java/com/palantir/leader/PaxosLeaderElectionService.java b/leader-election-impl/src/main/java/com/palantir/leader/PaxosLeaderElectionService.java index 99a42aa3e2d..4d052494779 100644 --- a/leader-election-impl/src/main/java/com/palantir/leader/PaxosLeaderElectionService.java +++ b/leader-election-impl/src/main/java/com/palantir/leader/PaxosLeaderElectionService.java @@ -144,7 +144,7 @@ public Optional getCurrentTokenIfLeading() { } private LeadershipState determineLeadershipState() { - Optional greatestLearnedValue = getGreatestLearnedPaxosValue(); + Optional greatestLearnedValue = knowledge.getGreatestLearnedValue(); StillLeadingStatus leadingStatus = determineLeadershipStatus(greatestLearnedValue); return LeadershipState.of(greatestLearnedValue, leadingStatus); @@ -181,10 +181,6 @@ private void proposeLeadershipAfter(Optional value) { } } - private Optional getGreatestLearnedPaxosValue() { - return Optional.ofNullable(knowledge.getGreatestLearnedValue()); - } - @Override public StillLeadingStatus isStillLeading(LeadershipToken token) { if (!(token instanceof PaxosLeadershipToken)) { @@ -224,7 +220,7 @@ private boolean isLatestRound(PaxosValue value) { } private boolean isLatestRound(Optional valueIfAny) { - return valueIfAny.equals(getGreatestLearnedPaxosValue()); + return valueIfAny.equals(knowledge.getGreatestLearnedValue()); } private void recordLeadershipStatus( @@ -256,7 +252,7 @@ private boolean updateLearnedStateFromPeers(Optional greatestLearned for (PaxosUpdate update : updates.get()) { ImmutableCollection values = update.getValues(); for (PaxosValue value : values) { - if (knowledge.getLearnedValue(value.getRound()) == null) { + if (!knowledge.getLearnedValue(value.getRound()).isPresent()) { knowledge.learn(value.getRound(), value); learned = true; } diff --git a/leader-election-impl/src/main/java/com/palantir/paxos/PaxosLearnerImpl.java b/leader-election-impl/src/main/java/com/palantir/paxos/PaxosLearnerImpl.java index e14c1c04f08..fd0c4d96f8e 100644 --- a/leader-election-impl/src/main/java/com/palantir/paxos/PaxosLearnerImpl.java +++ b/leader-election-impl/src/main/java/com/palantir/paxos/PaxosLearnerImpl.java @@ -16,14 +16,17 @@ package com.palantir.paxos; import java.io.IOException; -import java.util.ArrayList; import java.util.Collection; +import java.util.Optional; import java.util.SortedMap; import java.util.concurrent.ConcurrentSkipListMap; +import java.util.stream.Collectors; +import java.util.stream.LongStream; import org.slf4j.Logger; import org.slf4j.LoggerFactory; +import com.google.common.collect.ImmutableList; import com.palantir.leader.PaxosKnowledgeEventRecorder; import com.palantir.logsafe.SafeArg; @@ -69,7 +72,7 @@ public void learn(long seq, PaxosValue val) { } @Override - public PaxosValue getLearnedValue(long seq) { + public Optional getLearnedValue(long seq) { try { if (!state.containsKey(seq)) { byte[] bytes = log.readRound(seq); @@ -78,38 +81,35 @@ public PaxosValue getLearnedValue(long seq) { state.put(seq, value); } } - return state.get(seq); + return Optional.ofNullable(state.get(seq)); } catch (IOException e) { logger.error("Unable to get corrupt learned value for sequence {}", - SafeArg.of("sequence", seq), e); - return null; + SafeArg.of("sequence", seq), + e); + return Optional.empty(); } } @Override public Collection getLearnedValuesSince(long seq) { - PaxosValue greatestLearnedValue = getGreatestLearnedValue(); - long greatestSeq = -1L; - if (greatestLearnedValue != null) { - greatestSeq = greatestLearnedValue.seq; + Optional greatestSeq = getGreatestLearnedValue().map(PaxosValue::getRound); + if (!greatestSeq.isPresent()) { + return ImmutableList.of(); } - Collection values = new ArrayList<>(); - for (long i = seq; i <= greatestSeq; i++) { - PaxosValue value; - value = getLearnedValue(i); - if (value != null) { - values.add(value); - } - } - return values; + return LongStream.rangeClosed(seq, greatestSeq.get()) + .boxed() + .map(this::getLearnedValue) + .filter(Optional::isPresent) + .map(Optional::get) + .collect(Collectors.toList()); } @Override - public PaxosValue getGreatestLearnedValue() { - if (!state.isEmpty()) { - return state.get(state.lastKey()); + public Optional getGreatestLearnedValue() { + if (state.isEmpty()) { + return Optional.empty(); } - return null; + return Optional.ofNullable(state.get(state.lastKey())); } } diff --git a/leader-election-impl/src/main/java/com/palantir/paxos/SingleLeaderLearnerNetworkClient.java b/leader-election-impl/src/main/java/com/palantir/paxos/SingleLeaderLearnerNetworkClient.java index 040af37f424..0515a8a6dd4 100644 --- a/leader-election-impl/src/main/java/com/palantir/paxos/SingleLeaderLearnerNetworkClient.java +++ b/leader-election-impl/src/main/java/com/palantir/paxos/SingleLeaderLearnerNetworkClient.java @@ -84,7 +84,7 @@ public PaxosResponses getLearnedValue( Function, T> mapper) { return PaxosQuorumChecker.collectQuorumResponses( allLearners, - learner -> mapper.apply(Optional.ofNullable(learner.getLearnedValue(seq))), + learner -> mapper.apply(learner.getLearnedValue(seq)), quorumSize, executors, PaxosQuorumChecker.DEFAULT_REMOTE_REQUESTS_TIMEOUT).withoutRemotes(); diff --git a/timelock-agent/src/main/java/com/palantir/timelock/paxos/TimelockPaxosLearnerAdapter.java b/timelock-agent/src/main/java/com/palantir/timelock/paxos/TimelockPaxosLearnerAdapter.java index 783153e220b..842a8607c3f 100644 --- a/timelock-agent/src/main/java/com/palantir/timelock/paxos/TimelockPaxosLearnerAdapter.java +++ b/timelock-agent/src/main/java/com/palantir/timelock/paxos/TimelockPaxosLearnerAdapter.java @@ -18,12 +18,10 @@ import java.util.Collection; import java.util.List; +import java.util.Optional; import java.util.function.Function; import java.util.stream.Collectors; -import javax.annotation.Nonnull; -import javax.annotation.Nullable; - import com.palantir.atlasdb.timelock.paxos.Client; import com.palantir.atlasdb.timelock.paxos.PaxosRemoteClients; import com.palantir.atlasdb.timelock.paxos.PaxosUseCase; @@ -50,19 +48,16 @@ public void learn(long seq, PaxosValue val) { clientAwarePaxosLearner.learn(paxosUseCase, client, seq, val); } - @Nullable @Override - public PaxosValue getLearnedValue(long seq) { + public Optional getLearnedValue(long seq) { return clientAwarePaxosLearner.getLearnedValue(paxosUseCase, client, seq); } - @Nullable @Override - public PaxosValue getGreatestLearnedValue() { + public Optional getGreatestLearnedValue() { return clientAwarePaxosLearner.getGreatestLearnedValue(paxosUseCase, client); } - @Nonnull @Override public Collection getLearnedValuesSince(long seq) { return clientAwarePaxosLearner.getLearnedValuesSince(paxosUseCase, client, seq); diff --git a/timelock-agent/src/main/java/com/palantir/timelock/paxos/TimelockPaxosLearnerRpcClient.java b/timelock-agent/src/main/java/com/palantir/timelock/paxos/TimelockPaxosLearnerRpcClient.java index 874e365be65..f885df3eb9a 100644 --- a/timelock-agent/src/main/java/com/palantir/timelock/paxos/TimelockPaxosLearnerRpcClient.java +++ b/timelock-agent/src/main/java/com/palantir/timelock/paxos/TimelockPaxosLearnerRpcClient.java @@ -17,9 +17,8 @@ package com.palantir.timelock.paxos; import java.util.Collection; +import java.util.Optional; -import javax.annotation.Nonnull; -import javax.annotation.Nullable; import javax.ws.rs.Consumes; import javax.ws.rs.GET; import javax.ws.rs.POST; @@ -54,24 +53,21 @@ void learn( @PathParam("seq") long seq, PaxosValue val); - @Nullable @GET @Path("learned-value/{seq}") @Produces(MediaType.APPLICATION_JSON) - PaxosValue getLearnedValue( + Optional getLearnedValue( @PathParam("useCase") PaxosUseCase paxosUseCase, @PathParam("client") String client, @PathParam("seq") long seq); - @Nullable @GET @Path("greatest-learned-value") @Produces(MediaType.APPLICATION_JSON) - PaxosValue getGreatestLearnedValue( + Optional getGreatestLearnedValue( @PathParam("useCase") PaxosUseCase paxosUseCase, @PathParam("client") String client); - @Nonnull @GET @Path("learned-values-since/{seq}") @Produces(MediaType.APPLICATION_JSON) diff --git a/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/BatchPingableLeaderResource.java b/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/BatchPingableLeaderResource.java index 7fa0409f305..c2fe5a32af8 100644 --- a/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/BatchPingableLeaderResource.java +++ b/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/BatchPingableLeaderResource.java @@ -16,7 +16,7 @@ package com.palantir.atlasdb.timelock.paxos; -import java.util.Objects; +import java.util.Optional; import java.util.Set; import java.util.UUID; import java.util.stream.Collectors; @@ -40,7 +40,8 @@ public Set ping(Set clients) { return KeyedStream.of(clients) .map(components::learner) .map(PaxosLearner::getGreatestLearnedValue) - .filter(Objects::nonNull) + .filter(Optional::isPresent) + .map(Optional::get) .filter(this::isThisNodeTheLeaderFor) .keys() .collect(Collectors.toSet()); diff --git a/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/LocalBatchPaxosLearner.java b/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/LocalBatchPaxosLearner.java index 89770814625..2935f78e639 100644 --- a/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/LocalBatchPaxosLearner.java +++ b/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/LocalBatchPaxosLearner.java @@ -18,7 +18,7 @@ import java.util.Collection; import java.util.Map; -import java.util.Objects; +import java.util.Optional; import java.util.Set; import com.google.common.collect.SetMultimap; @@ -45,7 +45,8 @@ public SetMultimap getLearnedValues(Set> cli .mapKeys(WithSeq::value) .map(WithSeq::seq) .map((client, seq) -> paxosComponents.learner(client).getLearnedValue(seq)) - .filter(Objects::nonNull) + .filter(Optional::isPresent) + .map(Optional::get) .collectToSetMultimap(); } diff --git a/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/PaxosTimestampBoundStore.java b/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/PaxosTimestampBoundStore.java index da07e6a6cb0..c9700c9d744 100644 --- a/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/PaxosTimestampBoundStore.java +++ b/timelock-impl/src/main/java/com/palantir/atlasdb/timelock/paxos/PaxosTimestampBoundStore.java @@ -33,6 +33,7 @@ import com.palantir.common.remoting.ServiceNotAvailableException; import com.palantir.leader.NotCurrentLeaderException; import com.palantir.logsafe.SafeArg; +import com.palantir.logsafe.exceptions.SafeIllegalStateException; import com.palantir.paxos.ImmutablePaxosLong; import com.palantir.paxos.PaxosAcceptor; import com.palantir.paxos.PaxosAcceptorNetworkClient; @@ -232,7 +233,10 @@ public synchronized void storeUpperLimit(long limit) throws MultipleRunningTimes while (true) { try { proposer.propose(newSeq, PtBytes.toBytes(limit)); - PaxosValue value = knowledge.getLearnedValue(newSeq); + PaxosValue value = knowledge.getLearnedValue(newSeq) + .orElseThrow(() -> new SafeIllegalStateException("Timestamp bound store: Paxos proposal" + + " returned without learning a value. This is unexpected and would suggest a bug in" + + " AtlasDB code. Please contact support.")); checkAgreedBoundIsOurs(limit, newSeq, value); long newLimit = PtBytes.toLong(value.getData()); agreedState = ImmutableSequenceAndBound.of(newSeq, newLimit); diff --git a/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/BatchPingableLeaderResourceTests.java b/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/BatchPingableLeaderResourceTests.java index b76412dff20..b0462e54d9b 100644 --- a/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/BatchPingableLeaderResourceTests.java +++ b/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/BatchPingableLeaderResourceTests.java @@ -19,6 +19,7 @@ import static org.assertj.core.api.Assertions.assertThat; import static org.mockito.Mockito.when; +import java.util.Optional; import java.util.Random; import java.util.UUID; @@ -80,13 +81,13 @@ public void filtersOutClientsWhereNothingHasBeenLearnt() { when(components.learner(CLIENT_1).getGreatestLearnedValue()) .thenReturn(paxosValue(LEADER_UUID)); when(components.learner(clientWhereNothingHasBeenLearnt).getGreatestLearnedValue()) - .thenReturn(null); + .thenReturn(Optional.empty()); assertThat(resource.ping(ImmutableSet.of(CLIENT_1, clientWhereNothingHasBeenLearnt))) .containsOnly(CLIENT_1); } - private static PaxosValue paxosValue(UUID uuid) { - return new PaxosValue(uuid.toString(), new Random().nextLong(), null); + private static Optional paxosValue(UUID uuid) { + return Optional.of(new PaxosValue(uuid.toString(), new Random().nextLong(), null)); } } diff --git a/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/LocalBatchPaxosLearnerTests.java b/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/LocalBatchPaxosLearnerTests.java index 2be6b79c685..f7f49887492 100644 --- a/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/LocalBatchPaxosLearnerTests.java +++ b/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/LocalBatchPaxosLearnerTests.java @@ -22,6 +22,7 @@ import static org.mockito.Mockito.when; import java.util.Map; +import java.util.Optional; import java.util.Set; import java.util.UUID; @@ -75,10 +76,10 @@ public void weProxyLearnRequestsThrough() { public void weProxyGetLearnedValuesThrough() { PaxosValue paxosValue1 = paxosValue(1); PaxosValue paxosValue2 = paxosValue(2); - when(paxosComponents.learner(CLIENT_1).getLearnedValue(1)).thenReturn(paxosValue1); - when(paxosComponents.learner(CLIENT_1).getLearnedValue(2)).thenReturn(null); - when(paxosComponents.learner(CLIENT_2).getLearnedValue(1)).thenReturn(paxosValue1); - when(paxosComponents.learner(CLIENT_2).getLearnedValue(2)).thenReturn(paxosValue2); + when(paxosComponents.learner(CLIENT_1).getLearnedValue(1)).thenReturn(Optional.of(paxosValue1)); + when(paxosComponents.learner(CLIENT_1).getLearnedValue(2)).thenReturn(Optional.empty()); + when(paxosComponents.learner(CLIENT_2).getLearnedValue(1)).thenReturn(Optional.of(paxosValue1)); + when(paxosComponents.learner(CLIENT_2).getLearnedValue(2)).thenReturn(Optional.of(paxosValue2)); SetMultimap expected = ImmutableSetMultimap.builder() .putAll(CLIENT_1, paxosValue1) diff --git a/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/LocalPaxosComponentsTest.java b/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/LocalPaxosComponentsTest.java index 83d1d49146d..4a91308d221 100644 --- a/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/LocalPaxosComponentsTest.java +++ b/timelock-impl/src/test/java/com/palantir/atlasdb/timelock/paxos/LocalPaxosComponentsTest.java @@ -65,9 +65,9 @@ public void setUp() throws IOException { public void newClientCanBeCreated() { PaxosLearner learner = paxosComponents.learner(CLIENT); learner.learn(PAXOS_ROUND_ONE, PAXOS_VALUE); - assertThat(learner.getGreatestLearnedValue()).isNotNull(); - assertThat(learner.getGreatestLearnedValue().getLeaderUUID()).isEqualTo(PAXOS_UUID); - assertThat(learner.getGreatestLearnedValue().getData()).isEqualTo(PAXOS_DATA); + + assertThat(learner.getGreatestLearnedValue()).map(PaxosValue::getLeaderUUID).contains(PAXOS_UUID); + assertThat(learner.getGreatestLearnedValue()).map(PaxosValue::getData).contains(PAXOS_DATA); PaxosAcceptor acceptor = paxosComponents.acceptor(CLIENT); acceptor.accept(PAXOS_ROUND_TWO, PAXOS_PROPOSAL);