net.java.ao.schema.SchemaGenerator Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of activeobjects Show documentation
Show all versions of activeobjects Show documentation
This is the full Active Objects library, if you don't know which one to use, you probably want this one.
/*
* Copyright 2007 Daniel Spiewak
*
* 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 net.java.ao.schema;
import com.google.common.base.Predicate;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import net.java.ao.ActiveObjectsConfigurationException;
import net.java.ao.AnnotationDelegate;
import net.java.ao.Common;
import net.java.ao.DatabaseProvider;
import net.java.ao.Polymorphic;
import net.java.ao.RawEntity;
import net.java.ao.SchemaConfiguration;
import net.java.ao.schema.ddl.DDLAction;
import net.java.ao.schema.ddl.DDLField;
import net.java.ao.schema.ddl.DDLForeignKey;
import net.java.ao.schema.ddl.DDLIndex;
import net.java.ao.schema.ddl.DDLTable;
import net.java.ao.schema.ddl.SchemaReader;
import net.java.ao.types.TypeInfo;
import net.java.ao.types.TypeManager;
import net.java.ao.types.TypeQualifiers;
import net.java.ao.util.EnumUtils;
import java.lang.reflect.Method;
import java.sql.Connection;
import java.sql.SQLException;
import java.sql.Statement;
import java.sql.Types;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import static com.google.common.collect.Sets.*;
import static net.java.ao.sql.SqlUtils.closeQuietly;
import static net.java.ao.types.TypeQualifiers.MAX_STRING_LENGTH;
import static net.java.ao.types.TypeQualifiers.qualifiers;
/**
* WARNING: Not part of the public API. This class is public only
* to allow its use within other packages in the ActiveObjects library.
*
* @author Daniel Spiewak
*/
public final class SchemaGenerator
{
private static final Set AUTO_INCREMENT_LEGAL_TYPES = ImmutableSet.of(Types.INTEGER, Types.BIGINT);
public static void migrate(DatabaseProvider provider,
SchemaConfiguration schemaConfiguration,
NameConverters nameConverters,
Class extends RawEntity>>... classes) throws SQLException
{
final Iterable statements = generateImpl(provider, schemaConfiguration, nameConverters, classes);
Connection conn = null;
Statement stmt = null;
try
{
conn = provider.getConnection();
stmt = conn.createStatement();
for (String statement : statements)
{
provider.executeUpdate(stmt, statement);
}
}
finally
{
closeQuietly(stmt, conn);
}
}
private static Iterable generateImpl(DatabaseProvider provider,
SchemaConfiguration schemaConfiguration,
NameConverters nameConverters,
Class extends RawEntity>>... classes) throws SQLException
{
final Collection statements = newLinkedHashSet(); // preserve the order of the elements
final DDLTable[] parsedTables = parseDDL(provider, nameConverters, classes);
final DDLTable[] readTables = SchemaReader.readSchema(provider, nameConverters, schemaConfiguration);
final DDLAction[] actions = SchemaReader.sortTopologically(SchemaReader.diffSchema(provider.getTypeManager(), parsedTables, readTables, provider.isCaseSensetive()));
for (DDLAction action : actions)
{
statements.addAll(Arrays.asList(provider.renderAction(nameConverters, action)));
}
return filterEmpty(statements);
}
private static Iterable filterEmpty(Iterable iterable)
{
return Iterables.filter(iterable, new Predicate()
{
@Override
public boolean apply(String input)
{
return !input.trim().isEmpty();
}
});
}
static DDLTable[] parseDDL(DatabaseProvider provider, NameConverters nameConverters, Class extends RawEntity>>... classes) {
final Map>, Set>>> deps = new HashMap>, Set>>>();
final Set>> roots = new LinkedHashSet>>();
for (Class extends RawEntity>> cls : classes) {
try {
parseDependencies(nameConverters.getFieldNameConverter(), deps, roots, cls);
} catch (StackOverflowError e) {
throw new RuntimeException("Circular dependency detected in or below " + cls.getCanonicalName());
}
}
List parsedTables = new ArrayList();
while (!roots.isEmpty()) {
Class extends RawEntity>>[] rootsArray = roots.toArray(new Class[roots.size()]);
roots.remove(rootsArray[0]);
Class extends RawEntity>> clazz = rootsArray[0];
if (clazz.getAnnotation(Polymorphic.class) == null) {
parsedTables.add(parseInterface(provider, nameConverters.getTableNameConverter(), nameConverters.getFieldNameConverter(), clazz));
}
List>> toRemove = new LinkedList>>();
Iterator>> depIterator = deps.keySet().iterator();
while (depIterator.hasNext()) {
Class extends RawEntity>> depClass = depIterator.next();
Set>> individualDeps = deps.get(depClass);
individualDeps.remove(clazz);
if (individualDeps.isEmpty()) {
roots.add(depClass);
toRemove.add(depClass);
}
}
for (Class extends RawEntity>> remove : toRemove) {
deps.remove(remove);
}
}
return parsedTables.toArray(new DDLTable[parsedTables.size()]);
}
private static void parseDependencies(FieldNameConverter fieldConverter, Map>,
Set>>> deps, Set>> roots, Class extends RawEntity>>... classes) {
for (Class extends RawEntity>> clazz : classes) {
if (deps.containsKey(clazz)) {
continue;
}
Set>> individualDeps = new LinkedHashSet>>();
for (Method method : clazz.getMethods()) {
String attributeName = fieldConverter.getName(method);
Class> type = Common.getAttributeTypeFromMethod(method);
if (attributeName != null && type != null && Common.interfaceInheritsFrom(type, RawEntity.class)) {
if (!type.equals(clazz)) {
individualDeps.add((Class extends RawEntity>>) type);
parseDependencies(fieldConverter, deps, roots, (Class extends RawEntity>>) type);
}
}
}
if (individualDeps.size() == 0) {
roots.add(clazz);
} else {
deps.put(clazz, individualDeps);
}
}
}
private static DDLTable parseInterface(DatabaseProvider provider, TableNameConverter nameConverter, FieldNameConverter fieldConverter, Class extends RawEntity>> clazz)
{
String sqlName = nameConverter.getName(clazz);
DDLTable table = new DDLTable();
table.setName(sqlName);
table.setFields(parseFields(provider, fieldConverter, clazz));
table.setForeignKeys(parseForeignKeys(nameConverter, fieldConverter, clazz));
table.setIndexes(parseIndexes(provider, nameConverter, fieldConverter, clazz));
return table;
}
/**
* Not intended for public usage. This method is declared public
* only to enable use within other ActiveObjects packages. Consider this
* function unsupported.
*/
public static DDLField[] parseFields(DatabaseProvider provider, FieldNameConverter fieldConverter, Class extends RawEntity>> clazz) {
List fields = new ArrayList();
List attributes = new LinkedList();
for (Method method : Common.getValueFieldsMethods(clazz, fieldConverter))
{
String attributeName = fieldConverter.getName(method);
final Class> type = Common.getAttributeTypeFromMethod(method);
if (attributeName != null && type != null)
{
if (attributes.contains(attributeName)) {
continue;
}
attributes.add(attributeName);
final AnnotationDelegate annotations = Common.getAnnotationDelegate(fieldConverter, method);
DDLField field = new DDLField();
field.setName(attributeName);
final TypeManager typeManager = provider.getTypeManager();
final TypeInfo> sqlType = getSQLTypeFromMethod(typeManager, type, method, annotations);
field.setType(sqlType);
field.setJdbcType(sqlType.getJdbcWriteType());
field.setPrimaryKey(isPrimaryKey(annotations, field));
field.setNotNull(annotations.isAnnotationPresent(NotNull.class) || annotations.isAnnotationPresent(Unique.class) || annotations.isAnnotationPresent(PrimaryKey.class));
field.setUnique(annotations.isAnnotationPresent(Unique.class));
final boolean isAutoIncrement = isAutoIncrement(type, annotations, field.getType());
field.setAutoIncrement(isAutoIncrement);
if (!isAutoIncrement)
{
if (annotations.isAnnotationPresent(Default.class))
{
final Object defaultValue = convertStringDefaultValue(annotations.getAnnotation(Default.class).value(), sqlType, method);
if (type.isEnum() && ((Integer) defaultValue) > EnumUtils.size((Class extends Enum>) type) - 1)
{
throw new ActiveObjectsConfigurationException("There is no enum value of '" + type + "'for which the ordinal is " + defaultValue);
}
field.setDefaultValue(defaultValue);
}
else if (ImmutableSet.>of(short.class, float.class, int.class, long.class, double.class).contains(type))
{
// set the default value for primitive types (float, short, int, long, char)
field.setDefaultValue(convertStringDefaultValue("0", sqlType, method));
}
}
if (field.isPrimaryKey()) {
fields.add(0, field);
} else {
fields.add(field);
}
if (Common.interfaceInheritsFrom(type, RawEntity.class)
&& type.getAnnotation(Polymorphic.class) != null) {
field.setDefaultValue(null); // polymorphic fields can't have default
attributeName = fieldConverter.getPolyTypeName(method);
field = new DDLField();
field.setName(attributeName);
field.setType(typeManager.getType(String.class, qualifiers().stringLength(127)));
field.setJdbcType(java.sql.Types.VARCHAR);
if (annotations.getAnnotation(NotNull.class) != null) {
field.setNotNull(true);
}
fields.add(field);
}
}
}
return fields.toArray(new DDLField[fields.size()]);
}
private static boolean isPrimaryKey(AnnotationDelegate annotations, DDLField field)
{
final boolean isPrimaryKey = annotations.isAnnotationPresent(PrimaryKey.class);
if (isPrimaryKey && !field.getType().isAllowedAsPrimaryKey())
{
throw new ActiveObjectsConfigurationException(PrimaryKey.class.getName() + " is not supported for type: " + field.getType());
}
return isPrimaryKey;
}
private static boolean isAutoIncrement(Class> type, AnnotationDelegate annotations, TypeInfo> dbType)
{
final boolean isAutoIncrement = annotations.isAnnotationPresent(AutoIncrement.class);
if (isAutoIncrement && (!AUTO_INCREMENT_LEGAL_TYPES.contains(dbType.getJdbcWriteType()) || type.isEnum()))
{
throw new ActiveObjectsConfigurationException(AutoIncrement.class.getName() + " is not supported for type: " + dbType);
}
return isAutoIncrement;
}
private static TypeInfo> getSQLTypeFromMethod(TypeManager typeManager, Class> type, Method method, AnnotationDelegate annotations) {
TypeQualifiers qualifiers = qualifiers();
StringLength lengthAnno = annotations.getAnnotation(StringLength.class);
if (lengthAnno != null) {
final int length = lengthAnno.value();
if (length > MAX_STRING_LENGTH)
{
throw new ActiveObjectsConfigurationException("@StringLength must be <= " + MAX_STRING_LENGTH + " or UNLIMITED").forMethod(method);
}
try {
qualifiers = qualifiers.stringLength(length);
}
catch (ActiveObjectsConfigurationException e) {
throw new ActiveObjectsConfigurationException(e.getMessage()).forMethod(method);
}
}
return typeManager.getType(type, qualifiers);
}
private static DDLForeignKey[] parseForeignKeys(TableNameConverter nameConverter, FieldNameConverter fieldConverter,
Class extends RawEntity>> clazz) {
Set back = new LinkedHashSet();
for (Method method : clazz.getMethods()) {
String attributeName = fieldConverter.getName(method);
Class> type = Common.getAttributeTypeFromMethod(method);
if (type != null && attributeName != null && Common.interfaceInheritsFrom(type, RawEntity.class)
&& type.getAnnotation(Polymorphic.class) == null) {
DDLForeignKey key = new DDLForeignKey();
key.setField(attributeName);
key.setTable(nameConverter.getName((Class extends RawEntity>>) type));
key.setForeignField(Common.getPrimaryKeyField((Class extends RawEntity>>) type, fieldConverter));
key.setDomesticTable(nameConverter.getName(clazz));
back.add(key);
}
}
return back.toArray(new DDLForeignKey[back.size()]);
}
private static DDLIndex[] parseIndexes(DatabaseProvider provider, TableNameConverter nameConverter, FieldNameConverter fieldConverter,
Class extends RawEntity>> clazz) {
Set back = new LinkedHashSet();
String tableName = nameConverter.getName(clazz);
for (Method method : clazz.getMethods()) {
String attributeName = fieldConverter.getName(method);
AnnotationDelegate annotations = Common.getAnnotationDelegate(fieldConverter, method);
if (Common.isAccessor(method) || Common.isMutator(method)) {
Indexed indexedAnno = annotations.getAnnotation(Indexed.class);
Class> type = Common.getAttributeTypeFromMethod(method);
if (indexedAnno != null || (type != null && Common.interfaceInheritsFrom(type, RawEntity.class))) {
DDLIndex index = new DDLIndex();
index.setField(attributeName);
index.setTable(tableName);
index.setType(getSQLTypeFromMethod(provider.getTypeManager(), type, method, annotations));
back.add(index);
}
}
}
for (Class> superInterface : clazz.getInterfaces()) {
if (!superInterface.equals(RawEntity.class) && !superInterface.isAnnotationPresent(Polymorphic.class)) {
back.addAll(Arrays.asList(parseIndexes(provider, nameConverter, fieldConverter, (Class extends RawEntity>>) superInterface)));
}
}
return back.toArray(new DDLIndex[back.size()]);
}
private static Object convertStringDefaultValue(String value, TypeInfo> type, Method method)
{
if (value == null)
{
return null;
}
if (!type.getSchemaProperties().isDefaultValueAllowed())
{
throw new ActiveObjectsConfigurationException("Default value is not allowed for database type " +
type.getSchemaProperties().getSqlTypeName());
}
try
{
Object ret = type.getLogicalType().parseDefault(value);
if (ret == null)
{
throw new ActiveObjectsConfigurationException("Default value cannot be empty").forMethod(method);
}
return ret;
}
catch (IllegalArgumentException e)
{
throw new ActiveObjectsConfigurationException(e.getMessage());
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy