import akka.actor.Cancellable;
import com.google.common.base.Preconditions;
import java.util.ArrayList;
-import java.util.Collections;
+import java.util.Collection;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;
import java.util.UUID;
import java.util.concurrent.TimeUnit;
+import org.opendaylight.controller.cluster.raft.ServerConfigurationPayload.ServerInfo;
import org.opendaylight.controller.cluster.raft.base.messages.ApplyState;
import org.opendaylight.controller.cluster.raft.behaviors.AbstractLeader;
import org.opendaylight.controller.cluster.raft.messages.AddServer;
}
protected void persistNewServerConfiguration(RaftActor raftActor, ServerOperationContext<?> operationContext){
- List <String> newConfig = new ArrayList<String>(raftContext.getPeerIds());
- newConfig.add(raftContext.getId());
+ Collection<PeerInfo> peers = raftContext.getPeers();
+ List<ServerInfo> newConfig = new ArrayList<>(peers.size() + 1);
+ for(PeerInfo peer: peers) {
+ newConfig.add(new ServerInfo(peer.getId(), peer.isVoting()));
+ }
+
+ newConfig.add(new ServerInfo(raftContext.getId(), true));
LOG.debug("{}: New server configuration : {}", raftContext.getId(), newConfig);
- ServerConfigurationPayload payload = new ServerConfigurationPayload(newConfig, Collections.<String>emptyList());
+ ServerConfigurationPayload payload = new ServerConfigurationPayload(newConfig);
raftActor.persistData(operationContext.getClientRequestor(), operationContext.getContextId(), payload);
*/
package org.opendaylight.controller.cluster.raft;
+import com.google.common.base.Preconditions;
import com.google.protobuf.GeneratedMessage.GeneratedExtension;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.Serializable;
import java.util.List;
import java.util.Map;
+import javax.annotation.Nonnull;
import org.opendaylight.controller.cluster.raft.protobuff.client.messages.Payload;
import org.opendaylight.controller.protobuff.messages.cluster.raft.AppendEntriesMessages;
import org.slf4j.Logger;
private static final Logger LOG = LoggerFactory.getLogger(ServerConfigurationPayload.class);
- private final List<String> newServerConfig;
- private final List<String> oldServerConfig;
+ private final List<ServerInfo> serverConfig;
private transient int serializedSize = -1;
- public ServerConfigurationPayload(List<String> newServerConfig, List<String> oldServerConfig) {
- this.newServerConfig = newServerConfig;
- this.oldServerConfig = oldServerConfig;
+ public ServerConfigurationPayload(@Nonnull List<ServerInfo> serverConfig) {
+ this.serverConfig = Preconditions.checkNotNull(serverConfig);
}
- public List<String> getNewServerConfig() {
- return newServerConfig;
- }
-
-
- public List<String> getOldServerConfig() {
- return oldServerConfig;
+ @Nonnull
+ public List<ServerInfo> getServerConfig() {
+ return serverConfig;
}
@Override
try {
ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream out = new ObjectOutputStream(bos);
- out.writeObject(newServerConfig);
- out.writeObject(oldServerConfig);
+ out.writeObject(serverConfig);
out.close();
serializedSize = bos.toByteArray().length;
@Override
public String toString() {
- return "ServerConfigurationPayload [newServerConfig=" + newServerConfig + ", oldServerConfig="
- + oldServerConfig + "]";
+ return "ServerConfigurationPayload [serverConfig=" + serverConfig + "]";
+ }
+
+ public static class ServerInfo implements Serializable {
+ private static final long serialVersionUID = 1L;
+
+ private final String id;
+ private final boolean isVoting;
+
+ public ServerInfo(@Nonnull String id, boolean isVoting) {
+ this.id = Preconditions.checkNotNull(id);
+ this.isVoting = isVoting;
+ }
+
+ @Nonnull
+ public String getId() {
+ return id;
+ }
+
+ public boolean isVoting() {
+ return isVoting;
+ }
+
+ @Override
+ public int hashCode() {
+ final int prime = 31;
+ int result = 1;
+ result = prime * result + (isVoting ? 1231 : 1237);
+ result = prime * result + id.hashCode();
+ return result;
+ }
+
+ @Override
+ public boolean equals(Object obj) {
+ if (obj == null || getClass() != obj.getClass()) {
+ return false;
+ }
+
+ ServerInfo other = (ServerInfo) obj;
+ return isVoting == other.isVoting && id.equals(other.id);
+ }
+
+ @Override
+ public String toString() {
+ return "ServerInfo [id=" + id + ", isVoting=" + isVoting + "]";
+ }
}
}
import java.util.Collections;
import java.util.List;
import java.util.Map;
-//import java.util.List;
import java.util.concurrent.TimeUnit;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.opendaylight.controller.cluster.DataPersistenceProvider;
import org.opendaylight.controller.cluster.NonPersistentDataProvider;
+import org.opendaylight.controller.cluster.raft.ServerConfigurationPayload.ServerInfo;
import org.opendaylight.controller.cluster.raft.base.messages.ApplySnapshot;
import org.opendaylight.controller.cluster.raft.base.messages.ApplyState;
import org.opendaylight.controller.cluster.raft.base.messages.CaptureSnapshotReply;
assertEquals("Leader journal last index", 3, leaderActorContext.getReplicatedLog().lastIndex());
assertEquals("Leader commit index", 3, leaderActorContext.getCommitIndex());
assertEquals("Leader last applied index", 3, leaderActorContext.getLastApplied());
- verifyServerConfigurationPayloadEntry(leaderActorContext.getReplicatedLog(), LEADER_ID, FOLLOWER_ID, NEW_SERVER_ID);
+ verifyServerConfigurationPayloadEntry(leaderActorContext.getReplicatedLog(), votingServer(LEADER_ID),
+ votingServer(FOLLOWER_ID), votingServer(NEW_SERVER_ID));
// Verify ServerConfigurationPayload entry in both followers
assertEquals("Follower journal last index", 3, followerActorContext.getReplicatedLog().lastIndex());
- verifyServerConfigurationPayloadEntry(followerActorContext.getReplicatedLog(), LEADER_ID, FOLLOWER_ID, NEW_SERVER_ID);
+ verifyServerConfigurationPayloadEntry(followerActorContext.getReplicatedLog(), votingServer(LEADER_ID),
+ votingServer(FOLLOWER_ID), votingServer(NEW_SERVER_ID));
assertEquals("New follower journal last index", 3, newFollowerActorContext.getReplicatedLog().lastIndex());
- verifyServerConfigurationPayloadEntry(newFollowerActorContext.getReplicatedLog(), LEADER_ID, FOLLOWER_ID, NEW_SERVER_ID);
+ verifyServerConfigurationPayloadEntry(newFollowerActorContext.getReplicatedLog(), votingServer(LEADER_ID),
+ votingServer(FOLLOWER_ID), votingServer(NEW_SERVER_ID));
// Verify new server config was applied in both followers
assertEquals("Leader journal last index", 2, leaderActorContext.getReplicatedLog().lastIndex());
assertEquals("Leader commit index", 2, leaderActorContext.getCommitIndex());
assertEquals("Leader last applied index", 2, leaderActorContext.getLastApplied());
- verifyServerConfigurationPayloadEntry(leaderActorContext.getReplicatedLog(), LEADER_ID, NEW_SERVER_ID);
+ verifyServerConfigurationPayloadEntry(leaderActorContext.getReplicatedLog(), votingServer(LEADER_ID),
+ votingServer(NEW_SERVER_ID));
// Verify ServerConfigurationPayload entry in the new follower
expectFirstMatching(newFollowerCollectorActor, ApplyState.class);
assertEquals("New follower journal last index", 2, newFollowerActorContext.getReplicatedLog().lastIndex());
- verifyServerConfigurationPayloadEntry(newFollowerActorContext.getReplicatedLog(), LEADER_ID, NEW_SERVER_ID);
+ verifyServerConfigurationPayloadEntry(newFollowerActorContext.getReplicatedLog(), votingServer(LEADER_ID),
+ votingServer(NEW_SERVER_ID));
// Verify new server config was applied in the new follower
assertEquals("Leader journal last index", 0, leaderActorContext.getReplicatedLog().lastIndex());
assertEquals("Leader commit index", 0, leaderActorContext.getCommitIndex());
assertEquals("Leader last applied index", 0, leaderActorContext.getLastApplied());
- verifyServerConfigurationPayloadEntry(leaderActorContext.getReplicatedLog(), LEADER_ID, NEW_SERVER_ID);
+ verifyServerConfigurationPayloadEntry(leaderActorContext.getReplicatedLog(), votingServer(LEADER_ID),
+ nonVotingServer(NEW_SERVER_ID));
// Verify ServerConfigurationPayload entry in the new follower
expectFirstMatching(newFollowerCollectorActor, ApplyState.class);
assertEquals("New follower journal last index", 0, newFollowerActorContext.getReplicatedLog().lastIndex());
- verifyServerConfigurationPayloadEntry(newFollowerActorContext.getReplicatedLog(), LEADER_ID, NEW_SERVER_ID);
+ verifyServerConfigurationPayloadEntry(newFollowerActorContext.getReplicatedLog(), votingServer(LEADER_ID),
+ nonVotingServer(NEW_SERVER_ID));
// Verify new server config was applied in the new follower
assertEquals("Leader commit index", 1, leaderActorContext.getCommitIndex());
assertEquals("Leader last applied index", 1, leaderActorContext.getLastApplied());
verifyServerConfigurationPayloadEntry(leaderActorContext.getReplicatedLog(),
- LEADER_ID, NEW_SERVER_ID, NEW_SERVER_ID2);
+ votingServer(LEADER_ID), nonVotingServer(NEW_SERVER_ID), nonVotingServer(NEW_SERVER_ID2));
}
@Test
assertEquals("Leader commit index", 1, leaderActorContext.getCommitIndex());
assertEquals("Leader last applied index", 1, leaderActorContext.getLastApplied());
verifyServerConfigurationPayloadEntry(leaderActorContext.getReplicatedLog(),
- LEADER_ID, NEW_SERVER_ID, NEW_SERVER_ID2);
+ votingServer(LEADER_ID), votingServer(NEW_SERVER_ID), nonVotingServer(NEW_SERVER_ID2));
// Verify ServerConfigurationPayload entry in the new follower
expectFirstMatching(leaderActor, AddServer.class);
}
- private static void verifyServerConfigurationPayloadEntry(ReplicatedLog log, String... cNew) {
+ private ServerInfo votingServer(String id) {
+ return new ServerInfo(id, true);
+ }
+
+ private ServerInfo nonVotingServer(String id) {
+ return new ServerInfo(id, false);
+ }
+
+ private static void verifyServerConfigurationPayloadEntry(ReplicatedLog log, ServerInfo... expected) {
ReplicatedLogEntry logEntry = log.get(log.lastIndex());
assertEquals("Last log entry payload class", ServerConfigurationPayload.class, logEntry.getData().getClass());
ServerConfigurationPayload payload = (ServerConfigurationPayload)logEntry.getData();
- assertEquals("getNewServerConfig", Sets.newHashSet(cNew), Sets.newHashSet(payload.getNewServerConfig()));
+ assertEquals("getNewServerConfig", Sets.newHashSet(expected), Sets.newHashSet(payload.getServerConfig()));
}
private static RaftActorContext newFollowerContext(String id, TestActorRef<? extends UntypedActor> actor) {