
org.infinispan.distribution.ch.impl.ReplicatedConsistentHash Maven / Gradle / Ivy
package org.infinispan.distribution.ch.impl;
import org.infinispan.commons.hash.Hash;
import org.infinispan.commons.marshall.InstanceReusingAdvancedExternalizer;
import org.infinispan.distribution.ch.ConsistentHash;
import org.infinispan.marshall.core.Ids;
import org.infinispan.remoting.transport.Address;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.util.*;
/**
* Special implementation of {@link org.infinispan.distribution.ch.ConsistentHash} for replicated caches.
* The hash-space has several segments owned by all members and the primary ownership of each segment is evenly
* spread between members.
*
* @author Dan Berindei
* @author [email protected]
* @since 5.2
*/
public class ReplicatedConsistentHash implements ConsistentHash {
private final Hash hashFunction;
private final int[] primaryOwners;
private final List members;
private final Set membersSet;
private final Set segments;
public ReplicatedConsistentHash(Hash hashFunction, List members, int[] primaryOwners) {
this.hashFunction = hashFunction;
this.members = Collections.unmodifiableList(new ArrayList(members));
this.membersSet = Collections.unmodifiableSet(new HashSet(members));
this.primaryOwners = primaryOwners;
Set segmentIds = new HashSet(primaryOwners.length);
for (int i = 0; i < primaryOwners.length; i++) {
segmentIds.add(i);
}
segments = Collections.unmodifiableSet(segmentIds);
}
@Override
public int getNumSegments() {
return primaryOwners.length;
}
@Override
public int getNumOwners() {
return members.size();
}
@Override
public List getMembers() {
return members;
}
@Override
public Hash getHashFunction() {
return hashFunction;
}
@Override
public int getSegment(Object key) {
// The result must always be positive, so we make sure the dividend is positive first
return (hashFunction.hash(key) & Integer.MAX_VALUE) % primaryOwners.length;
}
@Override
public List locateOwnersForSegment(int segmentId) {
Address primaryOwner = locatePrimaryOwnerForSegment(segmentId);
List owners = new ArrayList(members.size());
owners.add(primaryOwner);
for (Address member : members) {
if (!member.equals(primaryOwner)) {
owners.add(member);
}
}
return owners;
}
@Override
public Address locatePrimaryOwnerForSegment(int segmentId) {
return members.get(primaryOwners[segmentId]);
}
@Override
public Set getSegmentsForOwner(Address owner) {
if (owner == null) {
throw new IllegalArgumentException("owner cannot be null");
}
if (!membersSet.contains(owner)) {
throw new IllegalArgumentException("The node is not a member : " + owner);
}
return segments;
}
@Override
public Set getPrimarySegmentsForOwner(Address owner) {
int index = members.indexOf(owner);
if (index == -1) {
throw new IllegalArgumentException("The node is not a member : " + owner);
}
Set primarySegments = new HashSet();
for (int i = 0; i < primaryOwners.length; ++i) {
if (primaryOwners[i] == index) {
primarySegments.add(i);
}
}
return primarySegments;
}
@Override
public String getRoutingTableAsString() {
return Arrays.toString(primaryOwners);
}
@Override
public Address locatePrimaryOwner(Object key) {
return locatePrimaryOwnerForSegment(getSegment(key));
}
@Override
public List locateOwners(Object key) {
return locateOwnersForSegment(getSegment(key));
}
@Override
public Set locateAllOwners(Collection
© 2015 - 2025 Weber Informatics LLC | Privacy Policy