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

com.avaje.ebeaninternal.server.ddl.CreateTableVisitor Maven / Gradle / Ivy

package com.avaje.ebeaninternal.server.ddl;

import java.util.ArrayList;
import java.util.HashSet;
import java.util.Set;
import java.util.logging.Logger;

import com.avaje.ebean.config.dbplatform.DbDdlSyntax;
import com.avaje.ebean.config.dbplatform.DbType;
import com.avaje.ebeaninternal.server.deploy.BeanDescriptor;
import com.avaje.ebeaninternal.server.deploy.BeanProperty;
import com.avaje.ebeaninternal.server.deploy.BeanPropertyAssocOne;
import com.avaje.ebeaninternal.server.deploy.BeanPropertyCompound;
import com.avaje.ebeaninternal.server.deploy.CompoundUniqueContraint;
import com.avaje.ebeaninternal.server.deploy.InheritInfo;
import com.avaje.ebeaninternal.server.deploy.parse.SqlReservedWords;

/**
 * Used to generated the create table DDL script.
 */
public class CreateTableVisitor extends AbstractBeanVisitor {
	
	protected static final Logger logger = Logger.getLogger(CreateTableVisitor.class.getName());
	
	final DdlGenContext ctx;
	
	final PropertyVisitor pv;
	
	final DbDdlSyntax ddl;
	
	final int columnNameWidth;

	// avoid writing columns twice, e.g. when used in associations with insertable=false and updateable=false
	private final Set wroteColumns = new HashSet();

    private ArrayList checkConstraints = new ArrayList();

    private ArrayList uniqueConstraints = new ArrayList();

    private String table;
    private String schema;
    
	public Set getWroteColumns() {
		return wroteColumns;
	}
	

	public CreateTableVisitor(DdlGenContext ctx) {
		this.ctx = ctx;
		this.ddl = ctx.getDdlSyntax();
		this.columnNameWidth = ddl.getColumnNameWidth();
		this.pv = new CreateTableColumnVisitor(this, ctx);
	}
	
	public boolean isDbColumnWritten(String dbColumn) {
		// Columns are not case sensitive - user lower case
		// as e.g. @JoinColumn(s) may use a different case
	    return wroteColumns.contains(dbColumn.toLowerCase());
	}
	
	public void addDbColumnWritten(String dbColumn){
		// Column names are case insensitive
	    wroteColumns.add(dbColumn.toLowerCase());
	}

	/**
	 * Write the table name including a check for SQL reserved words.
	 */
	protected void writeTableName(BeanDescriptor descriptor) {
		
		String tableName = descriptor.getBaseTable();
		int dotPos = tableName.lastIndexOf('.');
		if (dotPos > -1){
			schema = tableName.substring(0, dotPos);
			table = tableName.substring(dotPos+1);
		} else {
			table = tableName;
		}
		
		if (SqlReservedWords.isKeyword(table)) {
			logger.warning("Table name ["+table+"] is a suspected SQL reserved word for bean "+descriptor.getFullName());
		}

		ctx.write(tableName);		
	}

	protected String getTable() {
	    return table;
	}
	
	protected String getSchema() {
	    return schema;
	}
	
	/**
	 * Write the column name including a check for the SQL reserved words.
	 * @param p
	 */
	protected void writeColumnName(String columnName, BeanProperty p) {
		
        addDbColumnWritten(columnName);

		if (SqlReservedWords.isKeyword(columnName)) {
			String propName = p == null ? "(Unknown)" : p.getFullBeanName();
			logger.warning("Column name ["+columnName+"] is a suspected SQL reserved word for property "+propName);
		}

		ctx.write("  ").write(columnName, columnNameWidth).write(" ");
	}
	
	/**
	 * Build a check constraint for the property if required.
	 * 

* Typically check constraint based on Enum mapping values. *

*/ protected void addCheckConstraint(BeanProperty p, String prefix, String constraintExpression) { if (p != null && constraintExpression != null){ // build constraint clause String s = "constraint "+getConstraintName(prefix, p)+" "+constraintExpression; // add to list as we render all check constraints just prior to primary key checkConstraints.add(s); } } protected String getConstraintName(String prefix, BeanProperty p) { return prefix + table + "_" + p.getDbColumn(); } protected void addUniqueConstraint(String constraintExpression) { uniqueConstraints.add(constraintExpression); } protected void addCheckConstraint(String constraintExpression) { checkConstraints.add(constraintExpression); } protected void addCheckConstraint(BeanProperty p) { addCheckConstraint(p,"ck_", p.getDbConstraintExpression()); } public boolean visitBean(BeanDescriptor descriptor) { wroteColumns.clear(); if (!descriptor.isInheritanceRoot()){ return false; } ctx.write("create table "); writeTableName(descriptor); ctx.write(" (").writeNewLine(); InheritInfo inheritInfo = descriptor.getInheritInfo(); if (inheritInfo != null && inheritInfo.isRoot()){ String discColumn = inheritInfo.getDiscriminatorColumn(); int discType = inheritInfo.getDiscriminatorType(); int discLength = inheritInfo.getDiscriminatorLength(); DbType dbType = ctx.getDbTypeMap().get(discType); String discDbType = dbType.renderType(discLength, 0); writeColumnName(discColumn, null); ctx.write(discDbType); ctx.write(" not null,"); ctx.writeNewLine(); } return true; } public void visitBeanEnd(BeanDescriptor descriptor) { visitInheritanceProperties(descriptor, pv); if (checkConstraints.size() > 0){ for (String checkConstraint : checkConstraints) { ctx.write(" ").write(checkConstraint).write(",").writeNewLine(); } checkConstraints = new ArrayList(); } if (uniqueConstraints.size() > 0){ for (String constraint : uniqueConstraints) { ctx.write(" ").write(constraint).write(",").writeNewLine(); } uniqueConstraints = new ArrayList(); } CompoundUniqueContraint[] compoundUniqueConstraints = descriptor.getCompoundUniqueConstraints(); if (compoundUniqueConstraints != null){ String table = descriptor.getBaseTable(); for (int i = 0; i < compoundUniqueConstraints.length; i++) { String constraint = createUniqueConstraint(table, i, compoundUniqueConstraints[i]); ctx.write(" ").write(constraint).write(",").writeNewLine(); } } BeanProperty[] ids = descriptor.propertiesId(); if (ids.length == 0){ // No comma + new line ctx.removeLast().removeLast(); } else if (ids.length > 1 || ddl.isInlinePrimaryKeyConstraint()) { // The Primary Key constraint was inlined with the column // ... No comma + new line ctx.removeLast().removeLast(); } else { // Add the primay key constraint String pkName = ddl.getPrimaryKeyName(table); ctx.write(" constraint ").write(pkName).write(" primary key ("); VisitorUtil.visit(ids, new AbstractPropertyVisitor() { @Override public void visitEmbeddedScalar(BeanProperty p, BeanPropertyAssocOne embedded) { ctx.write(p.getDbColumn()).write(", "); } @Override public void visitScalar(BeanProperty p) { ctx.write(p.getDbColumn()).write(", "); } @Override public void visitCompoundScalar(BeanPropertyCompound compound, BeanProperty p) { ctx.write(p.getDbColumn()).write(", "); } }); // remove the last comma, end of PK ctx.removeLast().write(")"); } // end of table ctx.write(")").writeNewLine(); ctx.write(";").writeNewLine().writeNewLine(); ctx.flush(); } private String createUniqueConstraint(String table, int idx, CompoundUniqueContraint uc) { String uqConstraintName = "uq_"+table+"_"+(idx+1) ; StringBuilder sb = new StringBuilder(); sb.append("constraint ") .append(uqConstraintName) .append(" unique ("); String[] columns = uc.getColumns(); for (int i = 0; i < columns.length; i++) { if (i > 0){ sb.append(","); } sb.append(columns[i]); } sb.append(")"); return sb.toString(); } public void visitBeanDescriptorEnd() { ctx.write(");").writeNewLine().writeNewLine(); } public PropertyVisitor visitProperty(BeanProperty p) { return pv; } public void visitBegin() { } public void visitEnd() { ctx.addIntersectionCreateTables(); ctx.flush(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy