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

info.archinnov.achilles.table.CQLTableCreator Maven / Gradle / Ivy

/**
 *
 * Copyright (C) 2012-2013 DuyHai DOAN
 *
 * 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 info.archinnov.achilles.table;

import static info.archinnov.achilles.counter.AchillesCounter.CQL_COUNTER_FQCN;
import static info.archinnov.achilles.counter.AchillesCounter.CQL_COUNTER_PRIMARY_KEY;
import static info.archinnov.achilles.counter.AchillesCounter.CQL_COUNTER_PROPERTY_NAME;
import static info.archinnov.achilles.counter.AchillesCounter.CQL_COUNTER_TABLE;
import static info.archinnov.achilles.counter.AchillesCounter.CQL_COUNTER_VALUE;
import info.archinnov.achilles.entity.metadata.EntityMeta;
import info.archinnov.achilles.entity.metadata.IndexProperties;
import info.archinnov.achilles.entity.metadata.InternalTimeUUID;
import info.archinnov.achilles.entity.metadata.PropertyMeta;
import info.archinnov.achilles.exception.AchillesInvalidTableException;
import info.archinnov.achilles.type.Counter;
import info.archinnov.achilles.validation.Validator;

import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.datastax.driver.core.Cluster;
import com.datastax.driver.core.KeyspaceMetadata;
import com.datastax.driver.core.Session;
import com.datastax.driver.core.TableMetadata;

public class CQLTableCreator extends TableCreator {
	private static final Logger log = LoggerFactory.getLogger(CQLTableCreator.class);

	private Session session;
	private String keyspaceName;
	private Cluster cluster;
	private Map tableMetas;

	private CQLTableValidator validator;

	public CQLTableCreator(Cluster cluster, Session session, String keyspaceName) {
		this.cluster = cluster;
		this.session = session;
		this.keyspaceName = keyspaceName;
		this.tableMetas = fetchTableMetaData();
		validator = new CQLTableValidator(cluster, keyspaceName);
	}

	@Override
	protected void validateOrCreateTableForEntity(EntityMeta entityMeta, boolean forceColumnFamilyCreation) {
		String tableName = entityMeta.getTableName().toLowerCase();
		if (tableMetas.containsKey(tableName)) {
			validator.validateForEntity(entityMeta, tableMetas.get(tableName));
		} else {
			if (forceColumnFamilyCreation) {
				log.debug("Force creation of table for entityMeta {}", entityMeta.getClassName());
				createTableForEntity(entityMeta);
			} else {
				throw new AchillesInvalidTableException("The required table '" + tableName
						+ "' does not exist for entity '" + entityMeta.getClassName() + "'");
			}
		}

	}

	@Override
	protected void validateOrCreateTableForCounter(boolean forceColumnFamilyCreation) {
		if (tableMetas.containsKey(CQL_COUNTER_TABLE)) {
			validator.validateAchillesCounter();
		} else {
			if (forceColumnFamilyCreation) {
				CQLTableBuilder builder = CQLTableBuilder.createTable(CQL_COUNTER_TABLE);
				builder.addColumn(CQL_COUNTER_FQCN, String.class);
				builder.addColumn(CQL_COUNTER_PRIMARY_KEY, String.class);
				builder.addColumn(CQL_COUNTER_PROPERTY_NAME, String.class);
				builder.addColumn(CQL_COUNTER_VALUE, Counter.class);
				builder.addPartitionComponent(CQL_COUNTER_FQCN);
				builder.addPartitionComponent(CQL_COUNTER_PRIMARY_KEY);
				builder.addClusteringComponent(CQL_COUNTER_PROPERTY_NAME);

				builder.addComment("Create default Achilles counter table '" + CQL_COUNTER_TABLE + "'");

				session.execute(builder.generateDDLScript());
			} else {
				throw new AchillesInvalidTableException("The required generic table '" + CQL_COUNTER_TABLE
						+ "' does not exist");
			}
		}

	}

	private void createTableForEntity(EntityMeta entityMeta) {
		log.debug("Creating table for entityMeta {}", entityMeta.getClassName());
		String tableName = entityMeta.getTableName();

		if (entityMeta.isClusteredCounter()) {
			createTableForClusteredCounter(entityMeta);
		} else {
			createTable(entityMeta, tableName);
		}
	}

	private void createTable(EntityMeta entityMeta, String tableName) {
		CQLTableBuilder builder = CQLTableBuilder.createTable(tableName);
		for (PropertyMeta pm : entityMeta.getAllMetasExceptIdMeta()) {
			String propertyName = pm.getPropertyName();
			Class keyClass = pm.getKeyClass();
			Class valueClass = pm.getValueClassForTableCreation();
			switch (pm.type()) {
			case SIMPLE:
			case LAZY_SIMPLE:
				builder.addColumn(propertyName, valueClass);
				if (pm.isIndexed()) {
					builder.addIndex(new IndexProperties(pm.getIndexProperties().getName(), propertyName));
				}
				break;
			case LIST:
			case LAZY_LIST:
				builder.addList(propertyName, valueClass);
				break;
			case SET:
			case LAZY_SET:
				builder.addSet(propertyName, valueClass);
				break;
			case MAP:
			case LAZY_MAP:
				builder.addMap(propertyName, keyClass, pm.getValueClass());
				break;
			default:
				break;
			}

		}
		buildPrimaryKey(entityMeta.getIdMeta(), builder);
		builder.addComment("Create table for entity '" + entityMeta.getClassName() + "'");
		session.execute(builder.generateDDLScript());
		if (builder.hasIndices()) {
			for (String indexScript : builder.generateIndices()) {
				session.execute(indexScript);
			}
		}

	}

	private void createTableForClusteredCounter(EntityMeta meta) {
		PropertyMeta pm = meta.getFirstMeta();

		log.debug("Creating table for counter property {} for entity {}", pm.getPropertyName(), meta.getClassName());

		CQLTableBuilder builder = CQLTableBuilder.createCounterTable(meta.getTableName());
		PropertyMeta idMeta = meta.getIdMeta();
		buildPrimaryKey(idMeta, builder);
		builder.addColumn(pm.getPropertyName(), pm.getValueClass());

		builder.addComment("Create table for clustered counter entity '" + meta.getClassName() + "'");

		session.execute(builder.generateDDLScript());

	}

	private Map fetchTableMetaData() {
		Map tableMetas = new HashMap();
		KeyspaceMetadata keyspaceMeta = cluster.getMetadata().getKeyspace(keyspaceName);

		Validator.validateTableTrue(keyspaceMeta != null, "Keyspace '%s' doest not exist or cannot be found",
				keyspaceName);

		for (TableMetadata tableMeta : keyspaceMeta.getTables()) {
			tableMetas.put(tableMeta.getName(), tableMeta);
		}
		return tableMetas;
	}

	private void buildPrimaryKey(PropertyMeta pm, CQLTableBuilder builder) {
		if (pm.isEmbeddedId()) {
			addPrimaryKeyComponents(pm, builder, true);
			addPrimaryKeyComponents(pm, builder, false);
		} else {
			String columnName = pm.getPropertyName();
			builder.addColumn(columnName, pm.getValueClassForTableCreation());
			builder.addPartitionComponent(columnName);
		}
	}

	private void addPrimaryKeyComponents(PropertyMeta pm, CQLTableBuilder builder, boolean partitionKey) {
		List componentNames;
		List> componentClasses;

		if (partitionKey) {
			componentNames = pm.getPartitionComponentNames();
			componentClasses = pm.getPartitionComponentClasses();
		} else {
			componentNames = pm.getClusteringComponentNames();
			componentClasses = pm.getClusteringComponentClasses();
			builder.setReversedClusteredComponent(pm.getReversedComponent());
		}
		for (int i = 0; i < componentNames.size(); i++) {
			String componentName = componentNames.get(i);
			Class javaType = componentClasses.get(i);
			if (pm.isComponentTimeUUID(componentName)) {
				javaType = InternalTimeUUID.class;
			}

			builder.addColumn(componentName, javaType);
			if (partitionKey)
				builder.addPartitionComponent(componentName);
			else
				builder.addClusteringComponent(componentName);
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy