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

org.bboxdb.distribution.zookeeper.ZookeeperClient Maven / Gradle / Ivy

/*******************************************************************************
 *
 *    Copyright (C) 2015-2018 the BBoxDB project
 *  
 *    Licensed under the Apache License, Version 2.0 (the "License");
 *    you may not use this file except in compliance with the License.
 *    You may obtain a copy of the License at
 *  
 *      http://www.apache.org/licenses/LICENSE-2.0
 *  
 *    Unless required by applicable law or agreed to in writing, software
 *    distributed under the License is distributed on an "AS IS" BASIS,
 *    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *    See the License for the specific language governing permissions and
 *    limitations under the License. 
 *    
 *******************************************************************************/
package org.bboxdb.distribution.zookeeper;

import java.util.Collection;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.Phaser;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;

import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.KeeperException;
import org.apache.zookeeper.KeeperException.Code;
import org.apache.zookeeper.WatchedEvent;
import org.apache.zookeeper.Watcher;
import org.apache.zookeeper.ZooDefs;
import org.apache.zookeeper.ZooKeeper;
import org.apache.zookeeper.ZooKeeper.States;
import org.apache.zookeeper.data.Stat;
import org.bboxdb.commons.ServiceState;
import org.bboxdb.commons.concurrent.AcquirableResource;
import org.bboxdb.misc.BBoxDBService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class ZookeeperClient implements BBoxDBService, AcquirableResource {

	/**
	 * The list of the zookeeper hosts
	 */
	private final String connectionString;

	/**
	 * The name of the bboxdb cluster
	 */
	private final String clustername;

	/**
	 * The zookeeper client instance
	 */
	private ZooKeeper zookeeper;
	
	/**
	 * Service state
	 */
	private final ServiceState serviceState;
	
	/**
	 * The usage counter
	 */
	private Phaser usage;

	/**
	 * The timeout for the zookeeper session in milliseconds
	 */
	private final static int ZOOKEEPER_SESSION_TIMEOUT = (int) TimeUnit.SECONDS.toMillis(30);

	/**
	 * The connect timeout in seconds
	 */
	private final static int ZOOKEEPER_CONNECT_TIMEOUT_IN_SEC = 5;

	/**
	 * The logger
	 */
	private final static Logger logger = LoggerFactory.getLogger(ZookeeperClient.class);

	public ZookeeperClient(final Collection zookeeperHosts, final String clustername) {
		
		Objects.requireNonNull(zookeeperHosts);
		
		if (zookeeperHosts.isEmpty()) {
			throw new IllegalArgumentException("No Zookeeper hosts are defined");
		}
		
		this.connectionString = zookeeperHosts.stream().collect(Collectors.joining(","));
		this.clustername = Objects.requireNonNull(clustername);
		this.serviceState = new ServiceState();
	}

	/**
	 * Connect to zookeeper
	 */
	@Override
	public void init() {

		try {
			serviceState.reset();
			serviceState.dipatchToStarting();
			usage = new Phaser(1);

			final CountDownLatch connectLatch = new CountDownLatch(1);
			
			zookeeper = new ZooKeeper(connectionString, ZOOKEEPER_SESSION_TIMEOUT, new Watcher() {
				@Override
				public void process(final WatchedEvent event) {
					if (event.getState() == Watcher.Event.KeeperState.SyncConnected) {
						connectLatch.countDown();
					}
				}
			});

			final boolean waitResult = connectLatch.await(ZOOKEEPER_CONNECT_TIMEOUT_IN_SEC, TimeUnit.SECONDS);

			if (waitResult == false) {
				throw new ZookeeperException("Unable to connect in " + ZOOKEEPER_CONNECT_TIMEOUT_IN_SEC 
						+" seconds. Connect string is: " + connectionString);
			}

			createDirectoryStructureIfNeeded();
			
			serviceState.dispatchToRunning();
		} catch (Exception e) {
			logger.warn("Got exception while connecting to zookeeper", e);
			closeZookeeperConnectionNE();
			serviceState.dispatchToFailed(e);
		}
	}

	/**
	 * Disconnect from zookeeper
	 * 
	 * This method is synchronized to guarantee that all running callbacks are
	 * completed before the shutdown is executed
	 */
	@Override
	public synchronized void shutdown() {

		if (! serviceState.isInRunningState()) {
			logger.warn("Unable to shutdown, service is in {} state", serviceState);
			return;
		}

		serviceState.dispatchToStopping();
		closeZookeeperConnectionNE();
		serviceState.dispatchToTerminated();
	}

	/**
	 * Close the zookeeper connection without any exception
	 */
	private void closeZookeeperConnectionNE() {
		if (zookeeper == null) {
			return;
		}

		try {
			logger.info("Disconnecting from zookeeper");
			
			// Wait until nobody uses the instance
			assert (! usage.isTerminated()) : "Usage counter is terminated";
			usage.arriveAndAwaitAdvance();
			
			zookeeper.close();
		} catch (InterruptedException e) {
			logger.warn("Got exception while closing zookeeper connection", e);
			Thread.currentThread().interrupt();
		}

		zookeeper = null;
	}
	
	/**
	 * Get the children and register without creating a watch
	 */
	public List getChildren(final String path)
			throws ZookeeperException, ZookeeperNotFoundException {
		
		return getChildren(path, null);
	}
	
	/**
	 * Get the children and register a watch
	 * 
	 * @param path
	 * @param watcher
	 * @return
	 * @throws ZookeeperException
	 * @throws ZookeeperNotFoundException
	 */
	public List getChildren(final String path, final Watcher watcher)
			throws ZookeeperException, ZookeeperNotFoundException {
		
		if(! serviceState.isInRunningState()) {
			throw new ZookeeperException("Zookeeper is not connected");
		}

		try {
			return zookeeper.getChildren(path, watcher);
		} catch (KeeperException e) {

			// Was node deleted between exists and getData call?
			if (e.code() == Code.NONODE) {
				throw new ZookeeperNotFoundException("The path does not exist: " + path, e);
			} else {
				throw new ZookeeperException(e);
			}

		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		}
	}

	/**
	 * Write the given data to zookeeper
	 * 
	 * @param path
	 * @param value
	 * @throws ZookeeperException
	 */
	public boolean setData(final String path, final String value) throws ZookeeperException {
		return setData(path, value, -1);
	}

	/**
	 * Write the given data to zookeeper if the version matches
	 * 
	 * @param path
	 * @param value
	 * @throws ZookeeperException
	 */
	public boolean setData(final String path, final String value, final int version) throws ZookeeperException {

		try {
			zookeeper.setData(path, value.getBytes(), -1);

			return true;
		} catch (KeeperException e) {

			// Version does not match
			if (e.code() == Code.BADVERSION) {
				return false;
			}

			throw new ZookeeperException(e);
		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		}
	}

	/**
	 * Read the data and the stat from the given path
	 * 
	 * @throws ZookeeperException
	 */
	public String getData(final String path, final Stat stat) throws ZookeeperException {
		try {
			return new String(zookeeper.getData(path, false, stat));
		} catch (KeeperException e) {
			throw new ZookeeperException(e);
		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		}
	}

	/**
	 * Read the data from the given path
	 * 
	 * @param path
	 * @return
	 * @throws ZookeeperException
	 */
	public String getData(final String path) throws ZookeeperException {
		return getData(path, null);
	}

	/**
	 * Register the name of the cluster in the zookeeper directory
	 * 
	 * @throws ZookeeperException
	 * @throws KeeperException
	 * @throws InterruptedException
	 */
	private void createDirectoryStructureIfNeeded() throws ZookeeperException {

		// Active instances
		final String activeInstancesPath = getActiveInstancesPath();
		createDirectoryStructureRecursive(activeInstancesPath);

		// Details of the instances
		final String detailsPath = getDetailsPath();
		createDirectoryStructureRecursive(detailsPath);		
	}

	/**
	 * Get the path of the zookeeper clustername
	 * 
	 * @param clustername
	 * @return
	 */
	public String getClusterPath() {
		return "/" + clustername;
	}

	/**
	 * Get the path for the systems
	 * 
	 * @param clustername
	 * @return
	 */
	public String getInstancesPath() {
		return getClusterPath() + "/" + ZookeeperNodeNames.NAME_SYSTEMS;
	}

	/**
	 * Get the path of the zookeeper nodes
	 */
	public String getActiveInstancesPath() {
		return getInstancesPath() + "/active";
	}

	/**
	 * Get the details path
	 * @return
	 */
	public String getDetailsPath() {
		return getInstancesPath() + "/details";
	}
	
	@Override
	public String getServicename() {
		return "Zookeeper Client";
	}

	/**
	 * Create the given directory structure recursive
	 * 
	 * @param path
	 * @throws ZookeeperException
	 */
	public void createDirectoryStructureRecursive(final String path) throws ZookeeperException {

		try {
			// Does the full path already exists?
			if (zookeeper.exists(path, false) != null) {
				return;
			}

			// Otherwise check and create all sub paths
			final String[] allNodes = path.split("/");
			final StringBuilder sb = new StringBuilder();

			// Start by 1 to skip the initial /
			for (int i = 1; i < allNodes.length; i++) {
				final String nextNode = allNodes[i];
				sb.append("/");
				sb.append(nextNode);

				final String partialPath = sb.toString();

				if (zookeeper.exists(partialPath, false) == null) {
					try {
						logger.debug("Path '{}' not found, creating", partialPath);
						zookeeper.create(partialPath, "".getBytes(), ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
					} catch(KeeperException e) {
						if(e.code() == Code.NODEEXISTS) {
							// Ignore exception if node was created already
						} else {
							throw e;
						}
					}
				}
			}
		} catch (KeeperException e) {
			throw new ZookeeperException(e);
		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		}
	}

	/**
	 * Reat the given path and return a string - simple version
	 * @param pathName
	 * @return
	 * @throws ZookeeperNotFoundException 
	 * @throws ZookeeperException 
	 */
	public String readPathAndReturnString(final String pathName) 
			throws ZookeeperException, ZookeeperNotFoundException {
		
		return readPathAndReturnString(pathName, null);
	}
	
	/**
	 * Read the given path and returns a string result
	 * 
	 * @param pathName
	 * @param watcher
	 * @return
	 * @throws ZookeeperException
	 * @throws ZookeeperNotFoundException
	 */
	public String readPathAndReturnString(final String pathName, final Watcher watcher) 
			throws ZookeeperException, ZookeeperNotFoundException {
		
		final byte[] bytes = readPathAndReturnBytes(pathName, watcher);
		return new String(bytes);
	}

	/**
	 * Read the given path and returns a byte array result
	 * 
	 * @param pathName
	 * @return
	 * @throws ZookeeperException
	 * @throws ZookeeperNotFoundException
	 */
	public byte[] readPathAndReturnBytes(final String pathName, final Watcher watcher)
			throws ZookeeperException, ZookeeperNotFoundException {

		try {
			if (zookeeper.exists(pathName, false) == null) {
					throw new ZookeeperNotFoundException("The path does not exist: " + pathName);
			}

			return zookeeper.getData(pathName, watcher, null);
		} catch (KeeperException e) {

			// Was node deleted between exists and getData call?
			if (e.code() == Code.NONODE) {
				throw new ZookeeperNotFoundException("The path does not exist: " + pathName, e);
			} else {
				throw new ZookeeperException(e);
			}

		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		}
	}

	/**
	 * Dies the given path exists?
	 * 
	 * @param pathName
	 * @return
	 * @throws ZookeeperException
	 */
	public boolean exists(final String pathName) throws ZookeeperException {
		try {
			if (zookeeper.exists(pathName, false) != null) {
				return true;
			}
		} catch (KeeperException e) {
			throw new ZookeeperException(e);
		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		}

		return false;
	}

	/**
	 * Delete the node recursive
	 * 
	 * @param path
	 * @throws ZookeeperException
	 */
	public void deleteNodesRecursive(final String path) throws ZookeeperException {
		try {

			final List childs = zookeeper.getChildren(path, false);

			for (final String child : childs) {
				deleteNodesRecursive(path + "/" + child);
			}

			zookeeper.delete(path, -1);

		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		} catch (KeeperException e) {
			if (e.code() == KeeperException.Code.NONODE) {
				// We try to delete concurrently deleted
				// nodes. So we can ignore the exception.
			} else {
				throw new ZookeeperException(e);
			}
		}
	}

	/**
	 * Delete all known data about a cluster
	 * 
	 * @param distributionGroup
	 * @throws ZookeeperException
	 */
	public void deleteCluster() throws ZookeeperException {
		final String path = getClusterPath();
		deleteNodesRecursive(path);
	}

	/**
	 * Create a new persistent node
	 * 
	 * @param path
	 * @param bytes
	 * @return
	 * @throws ZookeeperException
	 */
	public String createPersistentNode(final String path, final byte[] bytes) throws ZookeeperException {
		try {
			return zookeeper.create(path, bytes, ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
		} catch (KeeperException e) {
			throw new ZookeeperException(e);
		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		}
	}

	/**
	 * Create a new persistent sequential node
	 * 
	 * @param path
	 * @param bytes
	 * @return
	 * @throws ZookeeperException
	 */
	public String createPersistentSequencialNode(final String path, final byte[] bytes) throws ZookeeperException {
		try {
			return zookeeper.create(path, bytes, ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_SEQUENTIAL);
		} catch (KeeperException e) {
			throw new ZookeeperException(e);
		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		}
	}
	
	/**
	 * Replace the persistent node
	 * @param path
	 * @param bytes
	 * @throws ZookeeperException 
	 */
	public void replacePersistentNode(final String path, final byte[] bytes) throws ZookeeperException {
		try {
			if (zookeeper.exists(path, false) != null) {
				zookeeper.setData(path, bytes, -1);
			} else {
				createDirectoryStructureRecursive(path);
				zookeeper.setData(path, bytes, -1);
			}
		} catch (KeeperException e) {
			throw new ZookeeperException(e);
		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		}
	}
	
	/**
	 * Replace the ephemeral node
	 * @param path
	 * @param bytes
	 * @throws ZookeeperException
	 */
	public void replaceEphemeralNode(final String path, final byte[] bytes) throws ZookeeperException {
		
		try {
			// Delete old state if exists (e.g. caused by a fast restart of the
			// service)
			if (zookeeper.exists(path, false) != null) {
				zookeeper.delete(path, -1);
			}
	
			// Register new state
			zookeeper.create(path, bytes, ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL);
		} catch (KeeperException e) {
			throw new ZookeeperException(e);
		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
			throw new ZookeeperException(e);
		}
	}

	/**
	 * Replace the value for the given path, only if the old value matches. This
	 * operation is performed atomic.
	 * 
	 * @param path
	 * @param oldValue
	 * @param newValue
	 * @return
	 * @throws ZookeeperException
	 */
	public boolean testAndReplaceValue(final String path, final String oldValue, final String newValue)
			throws ZookeeperException {

		if (oldValue == null || newValue == null) {
			throw new IllegalArgumentException("Invalid parameter null for old or new value");
		}

		if (!exists(path)) {
			logger.debug("Unable to replace value, path {} does not exists", path);
			return false;
		}

		// Retry value assignment
		for (int retry = 0; retry < 10; retry++) {
			final Stat stat = new Stat();
			final String zookeeperValue = getData(path, stat);

			// Old value does not match
			if (!oldValue.equals(zookeeperValue)) {
				logger.debug("Unable to replace value, zk value {} for path {} does not match expected value {}",
						zookeeperValue, path, oldValue);

				return false;
			}

			// Replace only if version of the read matches
			final boolean result = setData(path, newValue, stat.getVersion());

			if (result == true) {
				return true;
			}
		}

		logger.debug("Unable to replace {} with {} in path", oldValue, newValue, path);

		return false;
	}

	/**
	 * Is the zookeeper client connected?
	 * 
	 * @return
	 */
	public boolean isConnected() {
		if (zookeeper == null) {
			return false;
		}

		return zookeeper.getState() == States.CONNECTED;
	}

	/**
	 * Returns the name of the cluster
	 * 
	 * @return
	 */
	public String getClustername() {
		return clustername;
	}

	/**
	 * Get the service state
	 * @return
	 */
	public ServiceState getServiceState() {
		return serviceState;
	}
	
	/**
	 * Get the group adapter
	 * @return
	 */
	public DistributionGroupAdapter getDistributionGroupAdapter() {
		return new DistributionGroupAdapter(this);
	}
	
	/**
	 * Get the distribution region adapter
	 * @return 
	 */
	public DistributionRegionAdapter getDistributionRegionAdapter() {
		return new DistributionRegionAdapter(this);
	}
	
	/**
	 * Get the tuple store adapter
	 */
	public TupleStoreAdapter getTupleStoreAdapter() {
		return new TupleStoreAdapter(this);
	}

	@Override
	public boolean acquire() {
		
		if(! serviceState.isInRunningState()) {
			return false;
		}
		
		assert (! usage.isTerminated()) : "Usage counter is terminated";

		usage.register();
		return true;
	}

	@Override
	public void release() {
		assert (usage.getUnarrivedParties() > 0) : "Usage counter is: " + usage.getUnarrivedParties();
		assert (! usage.isTerminated()) : "Usage counter is terminated";
		
		usage.arriveAndDeregister();
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy