All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.infinispan.partitionhandling.impl.PreferAvailabilityStrategy Maven / Gradle / Ivy

There is a newer version: 9.1.7.Final
Show newest version
package org.infinispan.partitionhandling.impl;

import org.infinispan.commons.util.InfinispanCollections;
import org.infinispan.distribution.ch.ConsistentHash;
import org.infinispan.partitionhandling.AvailabilityMode;
import org.infinispan.remoting.transport.Address;
import org.infinispan.topology.CacheStatusResponse;
import org.infinispan.topology.CacheTopology;
import org.infinispan.util.logging.Log;
import org.infinispan.util.logging.LogFactory;

import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Objects;

public class PreferAvailabilityStrategy implements AvailabilityStrategy {
   private static final Log log = LogFactory.getLog(PreferAvailabilityStrategy.class);

   @Override
   public void onJoin(AvailabilityStrategyContext context, Address joiner) {
      context.queueRebalance(context.getExpectedMembers());
   }

   @Override
   public void onGracefulLeave(AvailabilityStrategyContext context, Address leaver) {
      CacheTopology currentTopology = context.getCurrentTopology();
      List
newMembers = new ArrayList<>(currentTopology.getMembers()); newMembers.remove(leaver); if (newMembers.isEmpty()) { log.debugf("The last node of cache %s left", context.getCacheName()); context.updateCurrentTopology(newMembers); return; } if (context.getStableTopology() != null && isDataLost(context.getStableTopology().getCurrentCH(), newMembers)) { log.lostDataBecauseOfGracefulLeaver(context.getCacheName(), leaver); } // We have to do this in case rebalancing is disabled, or there is another rebalance in progress context.updateCurrentTopology(newMembers); context.queueRebalance(newMembers); } @Override public void onClusterViewChange(AvailabilityStrategyContext context, List
clusterMembers) { CacheTopology currentTopology = context.getCurrentTopology(); List
newMembers = new ArrayList<>(currentTopology.getMembers()); if (!newMembers.retainAll(clusterMembers)) { log.tracef("Cache %s did not lose any members, skipping rebalance", context.getCacheName()); return; } checkForLostData(context, newMembers); // We have to do the update in case rebalancing is disabled, or there is another rebalance in progress context.updateCurrentTopology(newMembers); context.queueRebalance(newMembers); } protected void checkForLostData(AvailabilityStrategyContext context, List
newMembers) { CacheTopology stableTopology = context.getStableTopology(); List
stableMembers = stableTopology.getMembers(); List
lostMembers = new ArrayList<>(stableMembers); lostMembers.removeAll(newMembers); if (isDataLost(stableTopology.getCurrentCH(), newMembers)) { log.lostDataBecauseOfAbruptLeavers(context.getCacheName(), lostMembers); } else if (lostMembers.size() >= Math.ceil(stableMembers.size() / 2d)) { log.minorityPartition(context.getCacheName(), newMembers, lostMembers, stableMembers); } } @Override public void onPartitionMerge(AvailabilityStrategyContext context, Collection statusResponses) { // Pick the biggest stable topology (i.e. the one with most members) CacheTopology maxStableTopology = null; for (CacheStatusResponse response : statusResponses) { CacheTopology stableTopology = response.getStableTopology(); if (stableTopology == null) { // The node hasn't properly joined yet. continue; } if (maxStableTopology == null || maxStableTopology.getMembers().size() < stableTopology.getMembers().size()) { maxStableTopology = stableTopology; } } // Now pick the biggest current topology derived from the biggest stable topology CacheTopology maxTopology = null; for (CacheStatusResponse response : statusResponses) { CacheTopology stableTopology = response.getStableTopology(); if (!Objects.equals(stableTopology, maxStableTopology)) continue; CacheTopology topology = response.getCacheTopology(); if (topology == null) { // The node hasn't properly joined yet. continue; } if (maxTopology == null || maxTopology.getMembers().size() < topology.getMembers().size()) { maxTopology = topology; } } if (maxTopology == null) { log.debugf("No current topology, recovered only joiners for cache %s", context.getCacheName()); } // Since we picked the biggest topology, its topology id may not be the biggest int maxTopologyId = 0; for (CacheStatusResponse response : statusResponses) { CacheTopology topology = response.getCacheTopology(); if (topology != null && maxTopologyId < topology.getTopologyId()) { maxTopologyId = topology.getTopologyId(); } } // Increment the topology id so that it's bigger than any topology that might have been sent by the old // coordinator. +1 is enough because there nodes wait for the new JGroups view before answering the status // request, and after they have the new view they can't process topology updates with the old view id. // Also cancel any pending rebalance by removing the pending CH, because we don't recover the rebalance // confirmation status (yet). CacheTopology mergedTopology = maxTopology; if (maxTopology != null) { mergedTopology = new CacheTopology(maxTopologyId + 1, maxTopology.getRebalanceId(), maxTopology.getCurrentCH(), null); } context.updateTopologiesAfterMerge(mergedTopology, maxStableTopology, null); // First update the CHs to remove any nodes that left from the current topology List
newMembers = context.getExpectedMembers(); List
survivingMembers = new ArrayList<>(mergedTopology.getMembers()); if (survivingMembers.retainAll(newMembers)) { checkForLostData(context, survivingMembers); } context.updateCurrentTopology(survivingMembers); // Then start a rebalance with the merged members context.queueRebalance(newMembers); } @Override public void onRebalanceEnd(AvailabilityStrategyContext context) { // Do nothing, if we need another rebalance it's already in the queue } @Override public void onManualAvailabilityChange(AvailabilityStrategyContext context, AvailabilityMode newAvailabilityMode) { // The cache should always be AVAILABLE } private boolean isDataLost(ConsistentHash currentCH, List
newMembers) { for (int i = 0; i < currentCH.getNumSegments(); i++) { if (!InfinispanCollections.containsAny(newMembers, currentCH.locateOwnersForSegment(i))) return true; } return false; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy