use of org.jaxdb.www.ddlx_0_5.xLygluGCXAA.$Schema.Table in project jaxdb by jaxdb.
the class Decompiler method createDDL.
public static Schema createDDL(final Connection connection) throws SQLException {
final DBVendor vendor = DBVendor.valueOf(connection.getMetaData());
final Decompiler decompiler = Decompiler.getDecompiler(vendor);
final DatabaseMetaData metaData = connection.getMetaData();
try (final ResultSet tableRows = metaData.getTables(null, null, null, new String[] { "TABLE" })) {
final Schema schema = new Schema();
final Map<String, List<$CheckReference>> tableNameToChecks = decompiler.getCheckConstraints(connection);
final Map<String, List<$Table.Constraints.Unique>> tableNameToUniques = decompiler.getUniqueConstraints(connection);
final Map<String, $Table.Indexes> tableNameToIndexes = decompiler.getIndexes(connection);
final Map<String, Map<String, $ForeignKeyUnary>> tableNameToForeignKeys = decompiler.getForeignKeys(connection);
final Map<String, $Column> columnNameToColumn = new HashMap<>();
final Map<Integer, $Column> columnNumberToColumn = new TreeMap<>();
final Map<String, TreeMap<Short, String>> indexNameToIndex = new HashMap<>();
final Map<String, String> indexNameToType = new HashMap<>();
final Map<String, Boolean> indexNameToUnique = new HashMap<>();
while (tableRows.next()) {
final String tableName = tableRows.getString(3);
final $Table table = new Schema.Table();
table.setName$(new $Named.Name$(tableName.toLowerCase()));
schema.addTable(table);
try (final ResultSet columnRows = metaData.getColumns(null, null, tableName, null)) {
while (columnRows.next()) {
final String columnName = columnRows.getString("COLUMN_NAME").toLowerCase();
final String typeName = columnRows.getString("TYPE_NAME");
final int columnSize = columnRows.getInt("COLUMN_SIZE");
final String _default = columnRows.getString("COLUMN_DEF");
final int index = columnRows.getInt("ORDINAL_POSITION");
final String nullable = columnRows.getString("IS_NULLABLE");
final String autoIncrement = columnRows.getString("IS_AUTOINCREMENT");
final int decimalDigits = columnRows.getInt("DECIMAL_DIGITS");
final $Column column = decompiler.makeColumn(columnName.toLowerCase(), typeName, columnSize, decimalDigits, _default, nullable.length() == 0 ? null : "YES".equals(nullable), autoIncrement.length() == 0 ? null : "YES".equals(autoIncrement));
columnNameToColumn.put(columnName, column);
columnNumberToColumn.put(index, column);
}
columnNumberToColumn.values().forEach(table::addColumn);
try (final ResultSet primaryKeyRows = metaData.getPrimaryKeys(null, null, tableName)) {
while (primaryKeyRows.next()) {
final String columnName = primaryKeyRows.getString("COLUMN_NAME").toLowerCase();
if (table.getConstraints() == null)
table.setConstraints(new $Table.Constraints());
if (table.getConstraints().getPrimaryKey() == null)
table.getConstraints().setPrimaryKey(new $Table.Constraints.PrimaryKey());
final $Table.Constraints.PrimaryKey.Column column = new $Table.Constraints.PrimaryKey.Column();
column.setName$(new $Table.Constraints.PrimaryKey.Column.Name$(columnName));
table.getConstraints().getPrimaryKey().addColumn(column);
}
}
final List<$Table.Constraints.Unique> uniques = tableNameToUniques == null ? null : tableNameToUniques.get(tableName);
if (uniques != null && uniques.size() > 0) {
if (table.getConstraints() == null)
table.setConstraints(new $Table.Constraints());
for (final $Table.Constraints.Unique unique : uniques) table.getConstraints().addUnique(unique);
}
try (final ResultSet indexRows = metaData.getIndexInfo(null, null, tableName, false, true)) {
while (indexRows.next()) {
final String columnName = indexRows.getString("COLUMN_NAME").toLowerCase();
if (columnName == null)
continue;
final String indexName = indexRows.getString("INDEX_NAME").toLowerCase();
TreeMap<Short, String> indexes = indexNameToIndex.get(indexName);
if (indexes == null)
indexNameToIndex.put(indexName, indexes = new TreeMap<>());
final short ordinalPosition = indexRows.getShort("ORDINAL_POSITION");
indexes.put(ordinalPosition, columnName);
final String type = getType(indexRows.getShort("TYPE"));
final String currentType = indexNameToType.get(indexName);
if (currentType == null)
indexNameToType.put(indexName, type);
else if (!type.equals(currentType))
throw new IllegalStateException("Expected " + type + " = " + currentType);
final boolean unique = !indexRows.getBoolean("NON_UNIQUE");
final Boolean currentUnique = indexNameToUnique.get(indexName);
if (currentUnique == null)
indexNameToUnique.put(indexName, unique);
else if (unique != currentUnique)
throw new IllegalStateException("Expected " + unique + " = " + currentType);
}
}
final $Table.Indexes indexes = tableNameToIndexes == null ? null : tableNameToIndexes.get(tableName);
if (indexes != null)
table.setIndexes(indexes);
final List<$CheckReference> checks = tableNameToChecks == null ? null : tableNameToChecks.get(tableName);
if (checks != null)
for (final $CheckReference check : checks) addCheck(columnNameToColumn.get(check.getColumn$().text()), check);
final Map<String, $ForeignKeyUnary> foreignKeys = tableNameToForeignKeys == null ? null : tableNameToForeignKeys.get(tableName);
if (foreignKeys != null)
for (final Map.Entry<String, $ForeignKeyUnary> entry : foreignKeys.entrySet()) columnNameToColumn.get(entry.getKey().toLowerCase()).setForeignKey(entry.getValue());
}
columnNameToColumn.clear();
columnNumberToColumn.clear();
indexNameToIndex.clear();
indexNameToType.clear();
}
return schema;
}
}
use of org.jaxdb.www.ddlx_0_5.xLygluGCXAA.$Schema.Table in project jaxdb by jaxdb.
the class Generator method registerColumns.
private static void registerColumns(final $Table table, final Set<? super String> tableNames, final Map<? super String, ColumnRef> columnNameToColumn) throws GeneratorExecutionException {
final String tableName = table.getName$().text();
final List<String> violations = new ArrayList<>();
String nameViolation = checkNameViolation(tableName);
if (nameViolation != null)
violations.add(nameViolation);
if (tableNames.contains(tableName))
throw new GeneratorExecutionException("Circular table dependency detected: " + tableName);
tableNames.add(tableName);
final List<$Column> columns = table.getColumn();
if (columns != null) {
for (int c = 0, len = columns.size(); c < len; ++c) {
final $Column column = columns.get(c);
final String columnName = column.getName$().text();
nameViolation = checkNameViolation(columnName);
if (nameViolation != null)
violations.add(nameViolation);
final ColumnRef existing = columnNameToColumn.get(columnName);
if (existing != null)
throw new GeneratorExecutionException("Duplicate column definition: " + tableName + "." + columnName);
columnNameToColumn.put(columnName, new ColumnRef(column, c));
}
}
if (violations.size() > 0)
violations.forEach(logger::warn);
}
use of org.jaxdb.www.ddlx_0_5.xLygluGCXAA.$Schema.Table in project jaxdb by jaxdb.
the class Generator method parse.
public LinkedHashSet<Statement> parse(final DBVendor vendor) throws GeneratorExecutionException {
final Map<String, LinkedHashSet<DropStatement>> dropTableStatements = new HashMap<>();
final Map<String, LinkedHashSet<DropStatement>> dropTypeStatements = new HashMap<>();
final Map<String, LinkedHashSet<CreateStatement>> createTableStatements = new HashMap<>();
final Schema normalized = ddlx.getNormalizedSchema();
// The following code resolves a problem with ENUM types. The DDLx is generated from merged schema, whereby the original owner
// of the ENUM type is lost. The jSQL, however, is generated from the normalized schema, where the owner of the ENUM type is
// present. The `tableNameToEnumToOwner` variable is a map for each table linking each table's ENUMs to their original owners.
final Map<String, $Table> tableNameToTable = new HashMap<>();
final Map<String, Map<String, String>> tableNameToEnumToOwner = new HashMap<String, Map<String, String>>() {
@Override
public Map<String, String> get(final Object key) {
final String tableName = (String) key;
Map<String, String> map = super.get(key);
if (map == null)
put(tableName, map = new HashMap<>());
return map;
}
};
for (final $Table table : normalized.getTable()) tableNameToTable.put(table.getName$().text(), table);
for ($Table table : normalized.getTable()) {
if (table.getAbstract$().text())
continue;
final Map<String, String> colNameToOwnerTable = tableNameToEnumToOwner.get(table.getName$().text());
do {
for (final $Column column : table.getColumn()) if (column instanceof $Enum)
colNameToOwnerTable.put(column.getName$().text(), table.getName$().text());
table = table.getExtends$() != null ? tableNameToTable.get(table.getExtends$().text()) : null;
} while (table != null);
}
final Set<String> skipTables = new HashSet<>();
final Schema merged = ddlx.getMergedSchema();
final List<$Table> tables = merged.getTable();
for (final $Table table : tables) {
if (table.getSkip$().text()) {
skipTables.add(table.getName$().text());
} else if (!table.getAbstract$().text()) {
dropTableStatements.put(table.getName$().text(), Compiler.getCompiler(vendor).dropTable(table));
dropTypeStatements.put(table.getName$().text(), Compiler.getCompiler(vendor).dropTypes(table, tableNameToEnumToOwner));
}
}
final Set<String> tableNames = new HashSet<>();
for (final $Table table : tables) if (!table.getAbstract$().text())
createTableStatements.put(table.getName$().text(), parseTable(vendor, table, tableNames, tableNameToEnumToOwner));
final LinkedHashSet<Statement> statements = new LinkedHashSet<>();
final CreateStatement createSchema = Compiler.getCompiler(vendor).createSchemaIfNotExists(merged);
if (createSchema != null)
statements.add(createSchema);
final ListIterator<$Table> listIterator = tables.listIterator(tables.size());
while (listIterator.hasPrevious()) {
final $Table table = listIterator.previous();
final String tableName = table.getName$().text();
if (!skipTables.contains(tableName) && !table.getAbstract$().text())
statements.addAll(dropTableStatements.get(tableName));
}
for (final $Table table : tables) {
final String tableName = table.getName$().text();
if (!skipTables.contains(tableName) && !table.getAbstract$().text())
statements.addAll(dropTypeStatements.get(tableName));
}
for (final $Table table : tables) {
final String tableName = table.getName$().text();
if (!skipTables.contains(tableName) && !table.getAbstract$().text())
statements.addAll(createTableStatements.get(tableName));
}
return statements;
}
use of org.jaxdb.www.ddlx_0_5.xLygluGCXAA.$Schema.Table in project jaxdb by jaxdb.
the class OracleCompiler method dropTypes.
@Override
LinkedHashSet<DropStatement> dropTypes(final $Table table, final Map<String, Map<String, String>> tableNameToEnumToOwner) {
final LinkedHashSet<DropStatement> statements = super.dropTypes(table, tableNameToEnumToOwner);
if (table.getColumn() != null) {
for (final $Column column : table.getColumn()) {
if (column instanceof $Integer) {
final $Integer type = ($Integer) column;
if (Generator.isAuto(type)) {
statements.add(new DropStatement("BEGIN EXECUTE IMMEDIATE 'DROP SEQUENCE " + q(getSequenceName(table, type)) + "'; EXCEPTION WHEN OTHERS THEN IF SQLCODE != -2289 THEN RAISE; END IF; END;"));
statements.add(new DropStatement("BEGIN EXECUTE IMMEDIATE 'DROP TRIGGER " + q(getTriggerName(table, type)) + "'; EXCEPTION WHEN OTHERS THEN IF SQLCODE != -4080 THEN RAISE; END IF; END;"));
}
}
}
}
return statements;
}
use of org.jaxdb.www.ddlx_0_5.xLygluGCXAA.$Schema.Table in project jaxdb by jaxdb.
the class Generator method getTypes.
private Type[] getTypes(final Table table, final Map<String, Table> tableNameToTable, final int depth, final Info info) throws GeneratorExecutionException {
final List<$Column> columns = table.getColumn();
final int size = columns == null ? 0 : columns.size();
final Type[] types;
if (table.getExtends$() == null) {
types = new Type[depth + size];
info.rootClassName = schemaClassName + "." + Identifiers.toClassCase(table.getName$().text());
} else {
types = getTypes(tableNameToTable.get(table.getExtends$().text()), tableNameToTable, depth + size, info);
}
if (columns != null) {
final boolean isSuperTable = depth != 0;
info.totalPrimaryCount.inc(getPrimaryColumnCount(table), isSuperTable);
for (int c = 1; c <= size; ++c) {
final $Column column = columns.get(size - c);
final Type type = getType(table, column);
if (org.jaxdb.ddlx.Generator.isAuto(column))
info.totalAutoCount.inc(1, isSuperTable);
if (type.keyForUpdate)
info.totalKeyForUpdateCount.inc(1, isSuperTable);
types[types.length - depth - c] = type;
}
}
return types;
}
Aggregations