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

org.bboxdb.network.client.BBoxDBCluster 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.network.client;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.function.Supplier;

import org.bboxdb.commons.DuplicateResolver;
import org.bboxdb.commons.MicroSecondTimestampProvider;
import org.bboxdb.commons.math.Hyperrectangle;
import org.bboxdb.distribution.TupleStoreConfigurationCache;
import org.bboxdb.distribution.membership.BBoxDBInstance;
import org.bboxdb.distribution.membership.BBoxDBInstanceManager;
import org.bboxdb.distribution.membership.MembershipConnectionService;
import org.bboxdb.distribution.partitioner.SpacePartitioner;
import org.bboxdb.distribution.partitioner.SpacePartitionerCache;
import org.bboxdb.distribution.placement.RandomResourcePlacementStrategy;
import org.bboxdb.distribution.placement.ResourceAllocationException;
import org.bboxdb.distribution.placement.ResourcePlacementStrategy;
import org.bboxdb.distribution.region.DistributionRegion;
import org.bboxdb.distribution.zookeeper.ZookeeperClient;
import org.bboxdb.misc.BBoxDBException;
import org.bboxdb.network.client.future.EmptyResultFuture;
import org.bboxdb.network.client.future.FutureRetryPolicy;
import org.bboxdb.network.client.future.JoinedTupleListFuture;
import org.bboxdb.network.client.future.NetworkOperationFuture;
import org.bboxdb.network.client.future.TupleListFuture;
import org.bboxdb.network.routing.RoutingHeader;
import org.bboxdb.network.routing.RoutingHop;
import org.bboxdb.network.routing.RoutingHopHelper;
import org.bboxdb.storage.entity.DeletedTuple;
import org.bboxdb.storage.entity.DistributionGroupConfiguration;
import org.bboxdb.storage.entity.Tuple;
import org.bboxdb.storage.entity.TupleStoreConfiguration;
import org.bboxdb.storage.entity.TupleStoreName;
import org.bboxdb.storage.sstable.duplicateresolver.DoNothingDuplicateResolver;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class BBoxDBCluster implements BBoxDB {

	/**
	 * The Zookeeper connection
	 */
	private final ZookeeperClient zookeeperClient;

	/**
	 * The resource placement strategy
	 */
	private final ResourcePlacementStrategy resourcePlacementStrategy;

	/**
	 * The membership connection service
	 */
	private final MembershipConnectionService membershipConnectionService;

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

	/**
	 * Create a new instance of the BBoxDB cluster
	 * @param zookeeperNodes
	 * @param clustername
	 */
	public BBoxDBCluster(final Collection zookeeperNodes, final String clustername) {
		zookeeperClient = new ZookeeperClient(zookeeperNodes, clustername);
		resourcePlacementStrategy = new RandomResourcePlacementStrategy();
		membershipConnectionService = MembershipConnectionService.getInstance();
	}

	/**
	 * Create a new instance of the BBoxDB cluster
	 * @param zookeeperNodes
	 * @param clustername
	 */
	public BBoxDBCluster(final String zookeeperNode, final String clustername) {
		this(Arrays.asList(zookeeperNode), clustername);
	}


	@Override
	public boolean connect() {
		zookeeperClient.init();
		BBoxDBInstanceManager.getInstance().startMembershipObserver(zookeeperClient);
		membershipConnectionService.init();
		return zookeeperClient.isConnected();
	}

	@Override
	public void disconnect() {
		membershipConnectionService.shutdown();
		BBoxDBInstanceManager.getInstance().stopMembershipObserver();
		zookeeperClient.shutdown();
	}

	/* (non-Javadoc)
	 * @see org.bboxdb.network.client.BBoxDB#createTable(java.lang.String)
	 */
	@Override
	public EmptyResultFuture createTable(final String table, final TupleStoreConfiguration configuration) 
			throws BBoxDBException {

		if(membershipConnectionService.getNumberOfConnections() == 0) {
			throw new BBoxDBException("createTable called, but connection list is empty");
		}

		try {
			final BBoxDBClient bboxdbClient = getSystemForNewRessources().getBboxDBClient();

			return bboxdbClient.createTable(table, configuration);
		} catch (ResourceAllocationException e) {
			throw new BBoxDBException(e);
		}
	}

	@Override
	public EmptyResultFuture deleteTable(final String deleteTable) throws BBoxDBException {

		if(membershipConnectionService.getNumberOfConnections() == 0) {
			throw new BBoxDBException("deleteTable called, but connection list is empty");
		}

		try {
			final BBoxDBClient bboxdbClient = getSystemForNewRessources().getBboxDBClient();

			return bboxdbClient.deleteTable(deleteTable);
		} catch (ResourceAllocationException e) {
			throw new BBoxDBException(e);
		}
	}

	@Override
	public EmptyResultFuture insertTuple(final String table, final Tuple tuple) throws BBoxDBException {
		
		final DistributionRegion distributionRegion = getRootNode(table);

		final Supplier> supplier = () -> {
			
			final List hops = RoutingHopHelper.getRoutingHopsForWrite(distributionRegion, 
					tuple.getBoundingBox());
			
			if(hops.isEmpty()) {
				logger.error("Got no hops for bbox {}", tuple.getBoundingBox());
				return new ArrayList<>();
			} else {
				final BBoxDBInstance instance = hops.get(0).getDistributedInstance();
				
				final RoutingHeader routingHeader = new RoutingHeader((short) 0, hops);	
	
				final BBoxDBConnection connection 
					= membershipConnectionService.getConnectionForInstance(instance);
				
				final NetworkOperationFuture future 
					= connection.getBboxDBClient().getInsertTupleFuture(table, tuple, routingHeader);
				
				return Arrays.asList(future);
			}
		};
		
		return new EmptyResultFuture(supplier);
	}

	@Override
	public EmptyResultFuture deleteTuple(final String table, final String key) throws BBoxDBException {
		final long timestamp = MicroSecondTimestampProvider.getNewTimestamp();
		return deleteTuple(table, key, timestamp);
	}
	
	@Override
	public EmptyResultFuture deleteTuple(final String table, final String key, final long timestamp) 
			throws BBoxDBException {
	
		return deleteTuple(table, key, timestamp, Hyperrectangle.FULL_SPACE);
	}

	@Override
	public EmptyResultFuture deleteTuple(final String table, final String key, final long timestamp, 
			final Hyperrectangle boundingBox) throws BBoxDBException {
		
		final DeletedTuple tuple = new DeletedTuple(key, timestamp);
		final DistributionRegion distributionRegion = getRootNode(table);

		final Supplier> supplier = () -> {
			final List hops = RoutingHopHelper.getRoutingHopsForWrite(distributionRegion, 
					boundingBox);
		
			final List futures = new ArrayList<>();
			for(final RoutingHop hop : hops) {
				final BBoxDBInstance instance = hop.getDistributedInstance();
				final BBoxDBConnection connection 
					= membershipConnectionService.getConnectionForInstance(instance);
				final RoutingHeader routingHeader = new RoutingHeader((short) 0, Arrays.asList(hop));
				final NetworkOperationFuture future 
					= connection.getBboxDBClient().getInsertTupleFuture(table, tuple, routingHeader);
				futures.add(future);
			}
			
			return futures;
		};

		return new EmptyResultFuture(supplier);
	}
	
	@Override
	public EmptyResultFuture lockTuple(final String table, final Tuple tuple, 
			final boolean deleteOnTimeout) throws BBoxDBException {
		
		final DistributionRegion distributionRegion = getRootNode(table);

		final Supplier> supplier = () -> {
			final List hops = RoutingHopHelper.getRoutingHopsForWrite(distributionRegion, 
					tuple.getBoundingBox());
		
			final List futures = new ArrayList<>();
			
			for(final RoutingHop hop : hops) {
				final BBoxDBInstance instance = hop.getDistributedInstance();
				
				final BBoxDBConnection connection 
					= membershipConnectionService.getConnectionForInstance(instance);
				
				final RoutingHeader routingHeader = new RoutingHeader((short) 0, Arrays.asList(hop));
				
				final NetworkOperationFuture future 
					= connection.getBboxDBClient().createLockTupleFuture(
							table, tuple, deleteOnTimeout, routingHeader);
				
				futures.add(future);
			}
			
			return futures;
		};
	
		// When version locking fails, try again with another version
		return new EmptyResultFuture(supplier, FutureRetryPolicy.RETRY_POLICY_NONE);
	}

	@Override
	public EmptyResultFuture createDistributionGroup(final String distributionGroup, 
			final DistributionGroupConfiguration distributionGroupConfiguration) throws BBoxDBException {

		if(membershipConnectionService.getNumberOfConnections() == 0) {
			throw new BBoxDBException("createDistributionGroup called, but connection list is empty");
		}

		try {
			final BBoxDBClient bboxdbClient = getSystemForNewRessources().getBboxDBClient();

			return bboxdbClient.createDistributionGroup(distributionGroup, distributionGroupConfiguration);
		} catch (ResourceAllocationException e) {
			throw new BBoxDBException(e);
		}
	}

	/**
	 * Find a system with free resources
	 * @return
	 * @throws ResourceAllocationException 
	 */
	private BBoxDBConnection getSystemForNewRessources() throws ResourceAllocationException {
		final List serverConnections = membershipConnectionService.getAllInstances();

		if(serverConnections == null) {
			throw new ResourceAllocationException("Server connections are null");
		}

		final BBoxDBInstance system = resourcePlacementStrategy.getInstancesForNewRessource(serverConnections);

		return membershipConnectionService.getConnectionForInstance(system);
	}

	@Override
	public EmptyResultFuture deleteDistributionGroup(final String distributionGroup) throws BBoxDBException {

		if(membershipConnectionService.getNumberOfConnections() == 0) {
			throw new BBoxDBException("deleteDistributionGroup called, but connection list is empty");
		}

		try {
			final BBoxDBClient bboxdbClient = getSystemForNewRessources().getBboxDBClient();

			return bboxdbClient.deleteDistributionGroup(distributionGroup);
		} catch (ResourceAllocationException e) {
			throw new BBoxDBException(e);
		}
	}

	@Override
	public TupleListFuture queryKey(final String table, final String key) throws BBoxDBException {
		
		if(logger.isDebugEnabled()) {
			logger.debug("Query by for key {} in table {}", key, table);
		}
		
		final DeletedTuple tuple = new DeletedTuple(key);
		final DistributionRegion distributionRegion = getRootNode(table);

		final Supplier> futureProvider = () -> {
			
			final List hops = RoutingHopHelper.getRoutingHopsForRead(distributionRegion, 
					tuple.getBoundingBox());
		
			final List futures = new ArrayList<>();
			
			for(final RoutingHop hop : hops) {
				final BBoxDBInstance instance = hop.getDistributedInstance();
				final BBoxDBConnection connection 
					= membershipConnectionService.getConnectionForInstance(instance);
				final RoutingHeader routingHeader = new RoutingHeader((short) 0, Arrays.asList(hop));
				final NetworkOperationFuture future 
					= connection.getBboxDBClient().getQueryKeyFuture(table, key, routingHeader);
				futures.add(future);
			}
			
			return futures;
		};

		final DuplicateResolver duplicateResolver 
			= TupleStoreConfigurationCache.getInstance().getDuplicateResolverForTupleStore(table);

		return new TupleListFuture(futureProvider, duplicateResolver, table);
	}

	@Override
	public TupleListFuture queryRectangle(final String table, final Hyperrectangle boundingBox) throws BBoxDBException {
		
		if(logger.isDebugEnabled()) {
			logger.debug("Query by for bounding box {} in table {}", boundingBox, table);
		}
		
		final DistributionRegion distributionRegion = getRootNode(table);

		final Supplier> futureProvider = () -> {
			
			final List hops = RoutingHopHelper.getRoutingHopsForRead(distributionRegion, 
					boundingBox);
		
			final List futures = new ArrayList<>();
			
			for(final RoutingHop hop : hops) {
				final BBoxDBInstance instance = hop.getDistributedInstance();
				final BBoxDBConnection connection 
					= membershipConnectionService.getConnectionForInstance(instance);
				final RoutingHeader routingHeader = new RoutingHeader((short) 0, Arrays.asList(hop));
				final NetworkOperationFuture future 
					= connection.getBboxDBClient().getQueryBoundingBoxFuture(table, boundingBox, routingHeader);
				futures.add(future);
			}
			
			return futures;
		};

		return new TupleListFuture(futureProvider, new DoNothingDuplicateResolver(), table);
	}

	/**
	 * Execute a continuous bounding box query
	 * @throws BBoxDBException 
	 * 
	 */
	@Override
	public TupleListFuture queryRectangleContinuous(final String table, final Hyperrectangle boundingBox) 
			throws BBoxDBException {
	
		final DistributionRegion distributionRegion = getRootNode(table);

		final List regions 
			= distributionRegion.getDistributionRegionsForBoundingBox(boundingBox);

		if(regions.size() != 1) {
			throw new BBoxDBException("The bounding box belongs to more than one distribution region, "
					+ "this is not supported in continuous queries");
		}

		final DistributionRegion region = regions.get(0);

		final BBoxDBInstance firstSystem = region.getSystems().get(0);
		final BBoxDBConnection connection = membershipConnectionService.getConnectionForInstance(firstSystem);

		return connection.getBboxDBClient().queryRectangleContinuous(table, boundingBox);
	}

	@Override
	public TupleListFuture queryRectangleAndTime(final String table,
			final Hyperrectangle boundingBox, final long timestamp) throws BBoxDBException {

		if(membershipConnectionService.getNumberOfConnections() == 0) {
			throw new BBoxDBException("queryBoundingBoxAndTime called, but connection list is empty");
		}

		if(logger.isDebugEnabled()) {
			logger.debug("Query by for bounding box {} in table {}", boundingBox, table);
		}
		
		final DistributionRegion distributionRegion = getRootNode(table);

		final Supplier> futureProvider = () -> {
			
			final List hops = RoutingHopHelper.getRoutingHopsForRead(distributionRegion, 
					boundingBox);
		
			final List futures = new ArrayList<>();
			
			for(final RoutingHop hop : hops) {
				final BBoxDBInstance instance = hop.getDistributedInstance();
				final BBoxDBConnection connection 
					= membershipConnectionService.getConnectionForInstance(instance);
				final RoutingHeader routingHeader = new RoutingHeader((short) 0, Arrays.asList(hop));
				final NetworkOperationFuture future 
					= connection.getBboxDBClient().getBoundingBoxAndTimeFuture(table, boundingBox, 
							timestamp, routingHeader);
				futures.add(future);
			}
			
			return futures;
		};

		return new TupleListFuture(futureProvider, new DoNothingDuplicateResolver(), table);
	}

	@Override
	public TupleListFuture queryVersionTime(final String table, final long timestamp) throws BBoxDBException {
		if(membershipConnectionService.getNumberOfConnections() == 0) {
			throw new BBoxDBException("queryTime called, but connection list is empty");
		}

		if(logger.isDebugEnabled()) {
			logger.debug("Query by for timestamp {} in table {}", timestamp, table);
		}

		final DistributionRegion distributionRegion = getRootNode(table);

		final Supplier> futureProvider = () -> {
			
			final List hops = RoutingHopHelper.getRoutingHopsForRead(distributionRegion, 
					Hyperrectangle.FULL_SPACE);
		
			final List futures = new ArrayList<>();
			
			for(final RoutingHop hop : hops) {
				final BBoxDBInstance instance = hop.getDistributedInstance();
				final BBoxDBConnection connection 
					= membershipConnectionService.getConnectionForInstance(instance);
				final RoutingHeader routingHeader = new RoutingHeader((short) 0, Arrays.asList(hop));
				final NetworkOperationFuture future 
					= connection.getBboxDBClient().getVersionTimeFuture(table, timestamp, routingHeader);
				futures.add(future);
			}
			
			return futures;
		};

		return new TupleListFuture(futureProvider, new DoNothingDuplicateResolver(), table);
	}

	@Override
	public TupleListFuture queryInsertedTime(final String table, final long timestamp) throws BBoxDBException {
		if(membershipConnectionService.getNumberOfConnections() == 0) {
			throw new BBoxDBException("queryTime called, but connection list is empty");
		}

		if(logger.isDebugEnabled()) {
			logger.debug("Query by for timestamp {} in table {}", timestamp, table);
		}

		final DistributionRegion distributionRegion = getRootNode(table);

		final Supplier> futureProvider = () -> {
			
			final List hops = RoutingHopHelper.getRoutingHopsForRead(distributionRegion, 
					Hyperrectangle.FULL_SPACE);
		
			final List futures = new ArrayList<>();
			
			for(final RoutingHop hop : hops) {
				final BBoxDBInstance instance = hop.getDistributedInstance();
				final BBoxDBConnection connection 
					= membershipConnectionService.getConnectionForInstance(instance);
				final RoutingHeader routingHeader = new RoutingHeader((short) 0, Arrays.asList(hop));
				final NetworkOperationFuture future 
					= connection.getBboxDBClient().getInsertedTimeFuture(table, 
							timestamp, routingHeader);
				futures.add(future);
			}
			
			return futures;
		};

		return new TupleListFuture(futureProvider, new DoNothingDuplicateResolver(), table);
	}
	
	/* (non-Javadoc)
	 * @see org.bboxdb.network.client.BBoxDB#queryJoin
	 */
	@Override
	public JoinedTupleListFuture queryJoin(final List tableNames, final Hyperrectangle boundingBox) throws BBoxDBException {

		if(membershipConnectionService.getNumberOfConnections() == 0) {
			throw new BBoxDBException("queryJoin called, but connection list is empty");
		}

		if(logger.isDebugEnabled()) {
			logger.debug("Query by for join {} on tables {}", boundingBox, tableNames);
		}
		
		
		final String fullname = tableNames.get(0);
		final DistributionRegion distributionRegion = getRootNode(fullname);
		
		final Supplier> futureProvider = () -> {
			
			final List hops = RoutingHopHelper.getRoutingHopsForRead(distributionRegion, 
					boundingBox);
		
			final List futures = new ArrayList<>();
			
			for(final RoutingHop hop : hops) {
				final BBoxDBInstance instance = hop.getDistributedInstance();
				final BBoxDBConnection connection 
					= membershipConnectionService.getConnectionForInstance(instance);
				final RoutingHeader routingHeader = new RoutingHeader((short) 0, Arrays.asList(hop));
				final NetworkOperationFuture future 
					= connection.getBboxDBClient().getJoinFuture(tableNames, 
							boundingBox, routingHeader);
				futures.add(future);
			}
			
			return futures;
		};

		return new JoinedTupleListFuture(futureProvider);		
	}

	/**
	 * Get the root node from space partitioner
	 * 
	 * @param fullname
	 * @return
	 * @throws BBoxDBException
	 */
	private DistributionRegion getRootNode(final String fullname) throws BBoxDBException {
		final TupleStoreName tupleStoreName = new TupleStoreName(fullname);
		final String distributionGroup = tupleStoreName.getDistributionGroup();
		
		final SpacePartitioner distributionAdapter = SpacePartitionerCache
				.getInstance().getSpacePartitionerForGroupName(distributionGroup);

		return distributionAdapter.getRootNode();
	}

	@Override
	public boolean isConnected() {
		return (membershipConnectionService.getNumberOfConnections() > 0);
	}

	/**
	 * Is the paging for queries enables
	 * @return
	 */
	public boolean isPagingEnabled() {
		return membershipConnectionService.isPagingEnabled();
	}

	/**
	 * Enable or disable paging
	 * @param pagingEnabled
	 */
	public void setPagingEnabled(final boolean pagingEnabled) {
		membershipConnectionService.setPagingEnabled(pagingEnabled);
	}

	/**
	 * Get the amount of tuples per page
	 * @return
	 */
	public short getTuplesPerPage() {
		return membershipConnectionService.getTuplesPerPage();
	}

	/**
	 * Set the tuples per page
	 * @param tuplesPerPage
	 */
	public void setTuplesPerPage(final short tuplesPerPage) {
		membershipConnectionService.setTuplesPerPage(tuplesPerPage);
	}
	
	@Override
	public int getInFlightCalls() {
		return membershipConnectionService
				.getAllConnections()
				.stream()
				.mapToInt(BBoxDBConnection::getInFlightCalls)
				.sum();
	}

	/**
	 * Get the zookeeper client
	 * @return
	 */
	public ZookeeperClient getZookeeperClient() {
		return zookeeperClient;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy