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

io.paradoxical.cassieq.clustering.election.HazelcastLeadershipProvider Maven / Gradle / Ivy

package io.paradoxical.cassieq.clustering.election;

import com.godaddy.logging.Logger;
import com.google.inject.Inject;
import com.hazelcast.core.HazelcastInstance;
import com.hazelcast.core.IMap;
import io.paradoxical.cassieq.clustering.HazelcastBase;
import io.paradoxical.cassieq.configurations.ClusteringConfig;
import io.paradoxical.cassieq.model.ClusterMember;
import io.paradoxical.cassieq.model.LeadershipRole;

import java.util.Optional;
import java.util.concurrent.TimeUnit;
import java.util.function.Function;

import static com.godaddy.logging.LoggerFactory.getLogger;

public class HazelcastLeadershipProvider extends HazelcastBase implements LeadershipProvider {
    private final HazelcastInstance hazelcastInstance;
    private final ClusteringConfig config;

    private static final Logger logger = getLogger(HazelcastLeadershipProvider.class);

    @Inject
    public HazelcastLeadershipProvider(
            HazelcastInstance hazelcastInstance,
            ClusteringConfig config) {
        super(hazelcastInstance);
        this.hazelcastInstance = hazelcastInstance;
        this.config = config;
    }

    @Override
    public boolean tryAcquireLeader(final LeadershipRole key) {
        Optional wasExecuted = clusterSyncedExecute(key, keySetter -> {
            final ClusterMember currentOwner = keySetter.value();

            // its already us
            if (currentOwner != null && currentOwner.equals(thisClusterMember())) {
                return true;
            }

            // stale or unclaimed
            if (currentOwner == null || !inCluster(currentOwner)) {
                keySetter.claim();

                return true;
            }

            // its someone else
            return false;
        });

        if (wasExecuted.isPresent()) {
            return wasExecuted.get();
        }

        return false;
    }

    @Override
    public boolean tryRelinquishLeadership(final LeadershipRole key) {
        Optional wasExecuted = clusterSyncedExecute(key, keySetter -> {
            if (keySetter.amLeader()) {
                keySetter.release();

                return true;
            }

            return false;
        });

        if (wasExecuted.isPresent()) {
            return wasExecuted.get();
        }

        return false;
    }

    private  Optional clusterSyncedExecute(final LeadershipRole key, final Function action) {
        final KeySetter keySetter = new KeySetter(key);

        if (keySetter.tryLock()) {
            try {
                return Optional.of(action.apply(keySetter));
            }
            finally {
                keySetter.unlock();
            }
        }

        return Optional.empty();
    }


    class KeySetter {
        private final IMap backingMap;
        private final LeadershipRole key;

        public KeySetter(LeadershipRole key) {
            this.backingMap = hazelcastInstance.getMap("leaders");

            this.key = key;
        }

        public boolean amLeader() {
            final ClusterMember value = value();

            return value != null && value.equals(thisClusterMember());
        }

        public ClusterMember value() {
            final String currentValue = backingMap.get(key);

            if (currentValue == null) {
                return null;
            }

            return ClusterMember.valueOf(currentValue);
        }

        public void claim() {
            backingMap.put(key, thisClusterMember().get());
        }

        public boolean tryLock() {
            try {
                if (!backingMap.tryLock(key, config.getLockWaitSeconds(), TimeUnit.SECONDS)) {
                    logger.with("wait-seconds", config.getLockWaitSeconds())
                          .with("key", key)
                          .warn("Unable to acquire lock in time");

                    return false;
                }
            }
            catch (InterruptedException e) {
                logger.error(e, "Error acquiring lock");

                return false;
            }

            return true;
        }

        public void unlock() {
            backingMap.unlock(key);
        }

        public void release() {
            backingMap.remove(key);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy