HHH-14338 use SESSION prefix instead of MODULE for temp tables on HSQLDB

+ more cleanups of HSQLDialect
This commit is contained in:
Gavin 2022-12-23 21:42:26 +01:00 committed by Gavin King
parent 9dbfc29749
commit 4946e8ca45
1 changed files with 23 additions and 85 deletions

View File

@ -10,17 +10,9 @@ import java.sql.DatabaseMetaData;
import java.sql.SQLException; import java.sql.SQLException;
import java.sql.Types; import java.sql.Types;
import org.hibernate.LockMode;
import org.hibernate.dialect.function.CommonFunctionFactory; import org.hibernate.dialect.function.CommonFunctionFactory;
import org.hibernate.dialect.identity.HSQLIdentityColumnSupport; import org.hibernate.dialect.identity.HSQLIdentityColumnSupport;
import org.hibernate.dialect.identity.IdentityColumnSupport; import org.hibernate.dialect.identity.IdentityColumnSupport;
import org.hibernate.dialect.lock.LockingStrategy;
import org.hibernate.dialect.lock.OptimisticForceIncrementLockingStrategy;
import org.hibernate.dialect.lock.OptimisticLockingStrategy;
import org.hibernate.dialect.lock.PessimisticForceIncrementLockingStrategy;
import org.hibernate.dialect.lock.PessimisticReadSelectLockingStrategy;
import org.hibernate.dialect.lock.PessimisticWriteSelectLockingStrategy;
import org.hibernate.dialect.lock.SelectLockingStrategy;
import org.hibernate.dialect.pagination.LimitHandler; import org.hibernate.dialect.pagination.LimitHandler;
import org.hibernate.dialect.pagination.OffsetFetchLimitHandler; import org.hibernate.dialect.pagination.OffsetFetchLimitHandler;
import org.hibernate.dialect.sequence.HSQLSequenceSupport; import org.hibernate.dialect.sequence.HSQLSequenceSupport;
@ -36,11 +28,9 @@ import org.hibernate.engine.jdbc.env.spi.NameQualifierSupport;
import org.hibernate.engine.spi.SessionFactoryImplementor; import org.hibernate.engine.spi.SessionFactoryImplementor;
import org.hibernate.exception.spi.TemplatedViolatedConstraintNameExtractor; import org.hibernate.exception.spi.TemplatedViolatedConstraintNameExtractor;
import org.hibernate.exception.spi.ViolatedConstraintNameExtractor; import org.hibernate.exception.spi.ViolatedConstraintNameExtractor;
import org.hibernate.internal.CoreMessageLogger;
import org.hibernate.internal.util.JdbcExceptionHelper; import org.hibernate.internal.util.JdbcExceptionHelper;
import org.hibernate.metamodel.mapping.EntityMappingType; import org.hibernate.metamodel.mapping.EntityMappingType;
import org.hibernate.metamodel.spi.RuntimeModelCreationContext; import org.hibernate.metamodel.spi.RuntimeModelCreationContext;
import org.hibernate.persister.entity.Lockable;
import org.hibernate.query.spi.QueryEngine; import org.hibernate.query.spi.QueryEngine;
import org.hibernate.query.sqm.CastType; import org.hibernate.query.sqm.CastType;
import org.hibernate.query.sqm.IntervalType; import org.hibernate.query.sqm.IntervalType;
@ -63,8 +53,6 @@ import org.hibernate.tool.schema.extract.internal.SequenceInformationExtractorHS
import org.hibernate.tool.schema.extract.spi.SequenceInformationExtractor; import org.hibernate.tool.schema.extract.spi.SequenceInformationExtractor;
import org.hibernate.type.spi.TypeConfiguration; import org.hibernate.type.spi.TypeConfiguration;
import org.jboss.logging.Logger;
import jakarta.persistence.TemporalType; import jakarta.persistence.TemporalType;
import static org.hibernate.exception.spi.TemplatedViolatedConstraintNameExtractor.extractUsingTemplate; import static org.hibernate.exception.spi.TemplatedViolatedConstraintNameExtractor.extractUsingTemplate;
@ -79,10 +67,6 @@ import static org.hibernate.type.SqlTypes.NCLOB;
* @author Fred Toussi * @author Fred Toussi
*/ */
public class HSQLDialect extends Dialect { public class HSQLDialect extends Dialect {
private static final CoreMessageLogger LOG = Logger.getMessageLogger(
CoreMessageLogger.class,
HSQLDialect.class.getName()
);
private static final DatabaseVersion MINIMUM_VERSION = DatabaseVersion.make( 2, 6, 1 ); private static final DatabaseVersion MINIMUM_VERSION = DatabaseVersion.make( 2, 6, 1 );
private final UniqueDelegate uniqueDelegate = new CreateTableUniqueDelegate(this); private final UniqueDelegate uniqueDelegate = new CreateTableUniqueDelegate(this);
@ -146,7 +130,7 @@ public class HSQLDialect extends Dialect {
public void initializeFunctionRegistry(QueryEngine queryEngine) { public void initializeFunctionRegistry(QueryEngine queryEngine) {
super.initializeFunctionRegistry( queryEngine ); super.initializeFunctionRegistry( queryEngine );
CommonFunctionFactory functionFactory = new CommonFunctionFactory(queryEngine); CommonFunctionFactory functionFactory = new CommonFunctionFactory( queryEngine );
// AVG by default uses the input type, so we possibly need to cast the argument type, hence a special function // AVG by default uses the input type, so we possibly need to cast the argument type, hence a special function
functionFactory.avg_castingNonDoubleArguments( this, SqlAstNodeRenderingMode.DEFAULT ); functionFactory.avg_castingNonDoubleArguments( this, SqlAstNodeRenderingMode.DEFAULT );
@ -355,9 +339,9 @@ public class HSQLDialect extends Dialect {
return OffsetFetchLimitHandler.INSTANCE; return OffsetFetchLimitHandler.INSTANCE;
} }
// Note : HSQLDB actually supports [IF EXISTS] before AND after the <tablename> // Note: HSQLDB actually supports IF EXISTS before AND after the table name.
// But as CASCADE has to be AFTER IF EXISTS in case it's after the tablename, // But as CASCADE has to be after IF EXISTS in case it's after the table name,
// We put the IF EXISTS before the tablename to be able to add CASCADE after. // we put the IF EXISTS before the table name to be able to add CASCADE after.
@Override @Override
public boolean supportsIfExistsAfterTableName() { public boolean supportsIfExistsAfterTableName() {
return false; return false;
@ -375,7 +359,6 @@ public class HSQLDialect extends Dialect {
@Override @Override
public String getQuerySequencesString() { public String getQuerySequencesString() {
// this assumes schema support, which is present in 1.8.0 and later...
return "select * from information_schema.sequences"; return "select * from information_schema.sequences";
} }
@ -394,11 +377,8 @@ public class HSQLDialect extends Dialect {
return EXTRACTOR_20; return EXTRACTOR_20;
} }
/**
* HSQLDB 2.0 messages have changed
* messages may be localized - therefore use the common, non-locale element " table: "
*/
private static final ViolatedConstraintNameExtractor EXTRACTOR_20 = private static final ViolatedConstraintNameExtractor EXTRACTOR_20 =
// messages may be localized, therefore use the common, non-locale element " table: "
new TemplatedViolatedConstraintNameExtractor( sqle -> { new TemplatedViolatedConstraintNameExtractor( sqle -> {
switch ( JdbcExceptionHelper.extractErrorCode( sqle ) ) { switch ( JdbcExceptionHelper.extractErrorCode( sqle ) ) {
case -8: case -8:
@ -415,44 +395,33 @@ public class HSQLDialect extends Dialect {
@Override @Override
public String getSelectClauseNullString(int sqlType, TypeConfiguration typeConfiguration) { public String getSelectClauseNullString(int sqlType, TypeConfiguration typeConfiguration) {
String literal;
switch ( sqlType ) { switch ( sqlType ) {
case Types.LONGVARCHAR: case Types.LONGVARCHAR:
case Types.VARCHAR: case Types.VARCHAR:
case Types.CHAR: case Types.CHAR:
literal = "cast(null as varchar(100))"; return "cast(null as varchar(100))";
break;
case Types.LONGVARBINARY: case Types.LONGVARBINARY:
case Types.VARBINARY: case Types.VARBINARY:
case Types.BINARY: case Types.BINARY:
literal = "cast(null as varbinary(100))"; return "cast(null as varbinary(100))";
break;
case Types.CLOB: case Types.CLOB:
literal = "cast(null as clob)"; return "cast(null as clob)";
break;
case Types.BLOB: case Types.BLOB:
literal = "cast(null as blob)"; return "cast(null as blob)";
break;
case Types.DATE: case Types.DATE:
literal = "cast(null as date)"; return "cast(null as date)";
break;
case Types.TIMESTAMP: case Types.TIMESTAMP:
case Types.TIMESTAMP_WITH_TIMEZONE: case Types.TIMESTAMP_WITH_TIMEZONE:
literal = "cast(null as timestamp)"; return "cast(null as timestamp)";
break;
case Types.BOOLEAN: case Types.BOOLEAN:
literal = "cast(null as boolean)"; return "cast(null as boolean)";
break;
case Types.BIT: case Types.BIT:
literal = "cast(null as bit)"; return "cast(null as bit)";
break;
case Types.TIME: case Types.TIME:
literal = "cast(null as time)"; return "cast(null as time)";
break;
default: default:
literal = "cast(null as int)"; return "cast(null as int)";
} }
return literal;
} }
@Override @Override
@ -475,11 +444,11 @@ public class HSQLDialect extends Dialect {
// can happen in the middle of a transaction // can happen in the middle of a transaction
return new LocalTemporaryTableMutationStrategy( return new LocalTemporaryTableMutationStrategy(
// With HSQLDB 2.0, the table name is qualified with MODULE to assist the drop // With HSQLDB 2.0, the table name is qualified with SESSION to assist the drop
// statement (in-case there is a global name beginning with HT_) // statement (in-case there is a global name beginning with HT_)
TemporaryTable.createIdTable( TemporaryTable.createIdTable(
rootEntityDescriptor, rootEntityDescriptor,
basename -> "MODULE." + TemporaryTable.ID_TABLE_PREFIX + basename, basename -> "session." + TemporaryTable.ID_TABLE_PREFIX + basename,
this, this,
runtimeModelCreationContext runtimeModelCreationContext
), ),
@ -502,11 +471,11 @@ public class HSQLDialect extends Dialect {
// can happen in the middle of a transaction // can happen in the middle of a transaction
return new LocalTemporaryTableInsertStrategy( return new LocalTemporaryTableInsertStrategy(
// With HSQLDB 2.0, the table name is qualified with MODULE to assist the drop // With HSQLDB 2.0, the table name is qualified with SESSION to assist the drop
// statement (in-case there is a global name beginning with HT_) // statement (in-case there is a global name beginning with HT_)
TemporaryTable.createEntityTable( TemporaryTable.createEntityTable(
rootEntityDescriptor, rootEntityDescriptor,
name -> "MODULE." + TemporaryTable.ENTITY_TABLE_PREFIX + name, name -> "session." + TemporaryTable.ENTITY_TABLE_PREFIX + name,
this, this,
runtimeModelCreationContext runtimeModelCreationContext
), ),
@ -536,14 +505,6 @@ public class HSQLDialect extends Dialect {
// current timestamp support ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ // current timestamp support ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
/**
* HSQLDB 1.8.x requires CALL CURRENT_TIMESTAMP but this should not
* be treated as a callable statement. It is equivalent to
* "select current_timestamp from dual" in some databases.
* HSQLDB 2.0 also supports VALUES CURRENT_TIMESTAMP
* <p>
* {@inheritDoc}
*/
@Override @Override
public boolean supportsCurrentTimestampSelection() { public boolean supportsCurrentTimestampSelection() {
return true; return true;
@ -556,30 +517,7 @@ public class HSQLDialect extends Dialect {
@Override @Override
public String getCurrentTimestampSelectString() { public String getCurrentTimestampSelectString() {
return "call current_timestamp"; return "values current_timestamp";
}
/**
* For HSQLDB 2.0, this is a copy of the base class implementation.
* For HSQLDB 1.8, only READ_UNCOMMITTED is supported.
* <p>
* {@inheritDoc}
*/
@Override
public LockingStrategy getLockingStrategy(Lockable lockable, LockMode lockMode) {
switch (lockMode) {
case PESSIMISTIC_FORCE_INCREMENT:
return new PessimisticForceIncrementLockingStrategy(lockable, lockMode);
case PESSIMISTIC_WRITE:
return new PessimisticWriteSelectLockingStrategy(lockable, lockMode);
case PESSIMISTIC_READ:
return new PessimisticReadSelectLockingStrategy(lockable, lockMode);
case OPTIMISTIC:
return new OptimisticLockingStrategy(lockable, lockMode);
case OPTIMISTIC_FORCE_INCREMENT:
return new OptimisticForceIncrementLockingStrategy(lockable, lockMode);
}
return new SelectLockingStrategy( lockable, lockMode );
} }
@Override @Override
@ -616,7 +554,7 @@ public class HSQLDialect extends Dialect {
@Override @Override
public boolean supportsTupleDistinctCounts() { public boolean supportsTupleDistinctCounts() {
// from v. 2.2.9 is added support for COUNT(DISTINCT ...) with multiple arguments // 2.2.9 is added support for COUNT(DISTINCT ...) with multiple arguments
return true; return true;
} }
@ -697,8 +635,8 @@ public class HSQLDialect extends Dialect {
@Override @Override
public IdentifierHelper buildIdentifierHelper(IdentifierHelperBuilder builder, DatabaseMetaData dbMetaData) public IdentifierHelper buildIdentifierHelper(IdentifierHelperBuilder builder, DatabaseMetaData dbMetaData)
throws SQLException { throws SQLException {
builder.setAutoQuoteInitialUnderscore(true); builder.setAutoQuoteInitialUnderscore( true );
return super.buildIdentifierHelper(builder, dbMetaData); return super.buildIdentifierHelper( builder, dbMetaData );
} }
@Override @Override