public abstract class DatabaseProvider extends Object implements Disposable
The superclass parent of all DatabaseProvider
implementations. Various implementations allow for an abstraction
around database-specific functionality (such as DDL). DatabaseProvider(s)
also handle the creation of new Connection
instances and
fully encapsulate the raw JDBC driver. Any database-specific
code should be placed in the database provider, rather than embedded
within the API logic.
This superclass contains a base-line, default implementation of most database-specific methods, thus requiring a minimum of work to implement a new database provider. For the sake of sanity (read: mine), this base-line implementation is basically specific to MySQL. Thus any DatabaseProvider implementations are really specifying the differences between the database in question and MySQL. To fully utilize the default implementations provided in this class, this fact should be kept in mind.
This class also handles the implementation details required to ensure
that only one active Connection
instance is available per thread. This is
in fact a very basic (and naive) form of connection pooling. It should
not be relied upon for performance reasons. Instead one should enable
connection pooling via the DataSource
passed to instances. You can
also use the net.java.ao.builder.EntityManagerBuilder builder to make it
easier to configure the pooling. The purpose of the thread-locked connection pooling
in this class is to satisfy transactions with external SQL statements.
Modifier and Type | Class and Description |
---|---|
protected static class |
DatabaseProvider.RenderFieldOptions |
static interface |
DatabaseProvider.SqlListener |
Modifier and Type | Field and Description |
---|---|
protected org.slf4j.Logger |
logger |
protected AtomicReference<String> |
quoteRef |
protected org.slf4j.Logger |
sqlLogger |
protected TypeManager |
typeManager |
Modifier | Constructor and Description |
---|---|
protected |
DatabaseProvider(DisposableDataSource dataSource,
String schema) |
protected |
DatabaseProvider(DisposableDataSource dataSource,
String schema,
TypeManager typeManager) |
Modifier and Type | Method and Description |
---|---|
protected String |
_getFunctionNameForField(TriggerNameConverter triggerNameConverter,
DDLTable table,
DDLField field)
Retrieves the name of the function which corresponds to the field
in question (if any).
|
protected String |
_getTriggerNameForField(TriggerNameConverter triggerNameConverter,
DDLTable table,
DDLField field)
Retrieves the name of the trigger which corresponds to the field
in question (if any).
|
protected SQLAction |
_renderDropFunctionForField(NameConverters nameConverters,
DDLTable table,
DDLField field)
Renders SQL statement(s) to drop the function which corresponds to the
specified field, if applicable, or
null otherwise. |
protected SQLAction |
_renderDropSequenceForField(NameConverters nameConverters,
DDLTable table,
DDLField field)
Renders SQL statement(s) to drop the sequence which corresponds to the
specified field, or
null if none. |
protected SQLAction |
_renderDropTriggerForField(NameConverters nameConverters,
DDLTable table,
DDLField field)
Renders SQL statement(s) to drop the trigger which corresponds to the
specified field, or
null if none. |
protected SQLAction |
_renderFunctionForField(NameConverters nameConverters,
DDLTable table,
DDLField field)
Renders the function which corresponds to the specified field, or
null if none. |
protected SQLAction |
_renderSequenceForField(NameConverters nameConverters,
DDLTable table,
DDLField field)
Renders the SQL for creating a sequence for the specified field, or
null if none. |
protected SQLAction |
_renderTriggerForField(NameConverters nameConverters,
DDLTable table,
DDLField field)
Renders the trigger which corresponds to the specified field, or
null if none. |
void |
addSqlListener(DatabaseProvider.SqlListener l) |
Connection |
commitTransaction(Connection c) |
protected String |
convertTypeToString(TypeInfo<?> type)
Converts the specified type into the database-specific DDL String
value.
|
void |
dispose()
Frees any resources held by the database provider or delegate
libraries (such as connection pools).
|
protected <T extends RawEntity<K>,K> |
executeInsertReturningKey(EntityManager manager,
Connection conn,
Class<T> entityType,
Class<K> pkType,
String pkField,
String sql,
DBParam... params)
Delegate method to execute an INSERT statement returning any auto-generated
primary key values.
|
void |
executeUpdate(Statement stmt,
CharSequence sql) |
Iterable<String> |
executeUpdateForAction(Statement stmt,
SQLAction action,
Set<String> completedStatements) |
Iterable<String> |
executeUpdatesForActions(Statement stmt,
Iterable<SQLAction> actions,
Set<String> completedStatements)
Attempt to execute a list of actions that make up a logical unit (e.g.
|
protected Iterable<DDLForeignKey> |
findForeignKeysForField(DDLTable table,
DDLField field) |
Connection |
getConnection()
Retrieves a JDBC
Connection instance which corresponds
to the database represented by the provider instance. |
protected String |
getDateFormat()
Returns the database-specific TIMESTAMP text format as defined by
the
SimpleDateFormat syntax. |
ResultSet |
getImportedKeys(Connection connection,
String tableName) |
ResultSet |
getIndexes(Connection conn,
String tableName) |
protected int |
getMaxIDLength()
Returns the maximum length for any identifier in the underlying database.
|
protected abstract Set<String> |
getReservedWords()
Retrieves the set of all reserved words for the underlying database.
|
String |
getSchema() |
ResultSet |
getSequences(Connection conn) |
ResultSet |
getTables(Connection conn)
Returns a result set of all of the tables (and associated
meta) in the database.
|
TypeManager |
getTypeManager() |
Object |
handleBlob(ResultSet res,
Class<?> type,
String field) |
void |
handleUpdateError(String sql,
SQLException e)
Tells whether this exception should be ignored when running an updated statement.
|
protected boolean |
hasIndex(IndexNameConverter indexNameConverter,
DDLIndex index) |
protected boolean |
hasIndex(String tableName,
String indexName) |
<T extends RawEntity<K>,K> |
insertBatch(EntityManager manager,
Connection conn,
Class<T> entityType,
Class<K> pkType,
String pkField,
boolean pkIdentity,
String table,
List<Map<String,Object>> rows)
Inserts a batch of rows.
|
<T extends RawEntity<K>,K> |
insertReturningKey(EntityManager manager,
Connection conn,
Class<T> entityType,
Class<K> pkType,
String pkField,
boolean pkIdentity,
String table,
DBParam... params)
Generates an INSERT statement to be used to create a new row in the
database, returning the primary key value.
|
boolean |
isCaseSensitive()
Flag indicating whether or not the underlying database uses case-sensitive
identifiers.
|
protected boolean |
isNumericType(int type)
Simple helper function used to determine of the specified JDBC
type is representitive of a numeric type.
|
protected boolean |
isSchemaNotEmpty() |
protected void |
loadQuoteString() |
protected void |
onSql(String sql) |
Object |
parseValue(int type,
String value)
Parses the database-agnostic
String value relevant to the specified SQL
type in int form (as defined by Types and returns
the Java value which corresponds. |
PreparedStatement |
preparedStatement(Connection c,
CharSequence sql) |
PreparedStatement |
preparedStatement(Connection c,
CharSequence sql,
int autoGeneratedKeys) |
PreparedStatement |
preparedStatement(Connection c,
CharSequence sql,
int resultSetType,
int resultSetConcurrency) |
String |
processID(Query.FieldMetadata fmd)
Performs any database specific post-processing on the specified
field identifier.
|
String |
processID(String id)
Performs any database specific post-processing on the specified
identifier.
|
protected String |
processOnClause(String on) |
String |
processOrderClause(String order) |
String |
processTableName(String tableName)
Processes the table name as
processID(String) but use
#shouldQuoteTableName(String) to check if the table name should
be quoted or not. |
String |
processWhereClause(String where) |
void |
putBoolean(PreparedStatement stmt,
int index,
boolean value)
Stors an SQL
BOOLEAN value in the database. |
void |
putNull(PreparedStatement stmt,
int index)
Stores an SQL
NULL value in the database. |
protected String |
querySelectFields(Query query,
TableNameConverter converter) |
protected String |
queryTableName(Query query,
TableNameConverter converter) |
String |
quote(String id) |
String |
quoteTableName(String tableName) |
void |
removeSqlListener(DatabaseProvider.SqlListener l) |
protected Iterable<SQLAction> |
renderAccessories(NameConverters nameConverters,
DDLTable table)
Generates the database-specific DDL statements required to create
all of the functions, sequences, and triggers necessary for the given table,
by calling
renderAccessoriesForField(NameConverters, DDLTable, DDLField)
for each of the table's fields. |
protected Iterable<SQLAction> |
renderAccessoriesForField(NameConverters nameConverters,
DDLTable table,
DDLField field)
Generates database-specific DDL statements required to create any functions,
sequences, or triggers required for the given field.
|
Iterable<SQLAction> |
renderAction(NameConverters nameConverters,
DDLAction action)
Top level delegating method for the process of rendering a database-agnostic
DDLAction into the database-specific SQL actions. |
protected Iterable<SQLAction> |
renderAlterTableAddColumn(NameConverters nameConverters,
DDLTable table,
DDLField field)
Generates the database-specific DDL statements required to add
a column to an existing table.
|
protected SQLAction |
renderAlterTableAddColumnStatement(NameConverters nameConverters,
DDLTable table,
DDLField field)
Generates the database-specific DDL statement for adding a column,
but not including any corresponding sequences, triggers, etc.
|
protected SQLAction |
renderAlterTableAddKey(DDLForeignKey key)
Generates the database-specific DDL statement required to add a
foreign key to a table.
|
protected Iterable<SQLAction> |
renderAlterTableChangeColumn(NameConverters nameConverters,
DDLTable table,
DDLField oldField,
DDLField field)
Generates the database-specific DDL statements required to change
the given column from its old specification to the given DDL value.
|
protected SQLAction |
renderAlterTableChangeColumnStatement(NameConverters nameConverters,
DDLTable table,
DDLField oldField,
DDLField field,
DatabaseProvider.RenderFieldOptions options)
Generates the database-specific DDL statement only for altering a table and
changing a column.
|
protected Iterable<SQLAction> |
renderAlterTableDropColumn(NameConverters nameConverters,
DDLTable table,
DDLField field)
Generates the database-specific DDL statements required to remove
the specified column from the given table.
|
protected SQLAction |
renderAlterTableDropColumnStatement(DDLTable table,
DDLField field) |
protected SQLAction |
renderAlterTableDropKey(DDLForeignKey key)
Generates the database-specific DDL statement required to remove a
foreign key from a table.
|
protected String |
renderAppend()
Generates any database-specific options which must be appended
to the end of a table definition.
|
protected String |
renderAutoIncrement()
Generates the DDL fragment required to specify an INTEGER field as
auto-incremented.
|
protected String |
renderConstraints(NameConverters nameConverters,
List<String> primaryKeys,
DDLTable table) |
protected String |
renderConstraintsForTable(UniqueNameConverter uniqueNameConverter,
DDLTable table)
Renders the foreign key constraints in database-specific DDL for
the table in question.
|
SQLAction |
renderCreateCompositeIndex(String tableName,
String indexName,
List<String> fields)
Deprecated.
Use
renderCreateIndex(IndexNameConverter, DDLIndex) for creating indexes. |
protected SQLAction |
renderCreateIndex(IndexNameConverter indexNameConverter,
DDLIndex index)
Generates the database-specific DDL statement required to create
a new index.
|
protected String |
renderDate(Date date)
Renders the provided
Date instance as a DATETIME literal
in the database-specific format. |
protected Iterable<SQLAction> |
renderDropAccessories(NameConverters nameConverters,
DDLTable table)
Generates the database-specific DDL statements required to drop
all of the functions, sequences, and triggers associated with the given table,
by calling
renderDropAccessoriesForField(NameConverters, DDLTable, DDLField)
for each of the table's fields. |
protected Iterable<SQLAction> |
renderDropAccessoriesForField(NameConverters nameConverters,
DDLTable table,
DDLField field)
Generates database-specific DDL statements required to drop any functions,
sequences, or triggers associated with the given field.
|
protected Iterable<SQLAction> |
renderDropColumnActions(NameConverters nameConverters,
DDLTable table,
DDLField field) |
protected SQLAction |
renderDropIndex(IndexNameConverter indexNameConverter,
DDLIndex index)
Generates the database-specific DDL statement required to drop
an index.
|
protected SQLAction |
renderDropTableStatement(DDLTable table)
Generates the appropriate database-specific DDL statement to
drop the specified table representation.
|
protected String |
renderField(NameConverters nameConverters,
DDLTable table,
DDLField field,
DatabaseProvider.RenderFieldOptions options)
Generates the database-specific DDL fragment required to render the
field and its associated type.
|
protected String |
renderFieldDefault(DDLTable table,
DDLField field) |
protected DatabaseProvider.RenderFieldOptions |
renderFieldOptionsInAlterColumn() |
protected Iterable<SQLAction> |
renderFields(DDLTable table,
com.google.common.base.Predicate<DDLField> filter,
com.google.common.base.Function<DDLField,Iterable<SQLAction>> render)
Deprecated.
This method will be removed on platform 6, since guava will be replaced by Java API.
|
protected Iterable<SQLAction> |
renderFields(DDLTable table,
Predicate<DDLField> filter,
Function<DDLField,Iterable<SQLAction>> render) |
protected String |
renderFieldType(DDLField field)
Renders the database-specific DDL type for the field in question.
|
protected String |
renderForeignKey(DDLForeignKey key)
Renders the specified foreign key representation into the
database-specific DDL.
|
protected SQLAction |
renderInsert(DDLTable ddlTable,
DDLValue[] ddlValues) |
String |
renderMetadataQuery(String tableName)
Render "SELECT * FROM
|
protected String |
renderPrimaryKey(String tableName,
String pkFieldName) |
String |
renderQuery(Query query,
TableNameConverter converter,
boolean count)
Top level delegating method for rendering a database-agnostic
Query object into its (potentially) database-specific
query statement. |
protected String |
renderQueryGroupBy(Query query)
Renders the GROUP BY portion of the query in the database-specific SQL
dialect.
|
protected String |
renderQueryHaving(Query query)
Renders the HAVING portion of the query in the database-specific SQL
dialect.
|
protected String |
renderQueryJoins(Query query,
TableNameConverter converter)
Renders the JOIN portion of the query in the database-specific SQL
dialect.
|
protected String |
renderQueryLimit(Query query)
Renders the LIMIT portion of the query in the database-specific SQL
dialect.
|
protected String |
renderQueryOrderBy(Query query)
Renders the ORDER BY portion of the query in the database-specific SQL
dialect.
|
protected String |
renderQuerySelect(Query query,
TableNameConverter converter,
boolean count)
Renders the SELECT portion of a given
Query instance in the
manner required by the database-specific SQL implementation. |
protected String |
renderQueryWhere(Query query)
Renders the WHERE portion of the query in the database-specific SQL
dialect.
|
protected SQLAction |
renderTable(NameConverters nameConverters,
DDLTable table)
Renders the specified table representation into the corresponding
database-specific DDL statement.
|
protected String |
renderUnique(UniqueNameConverter uniqueNameConverter,
DDLTable table,
DDLField field)
Renders the
UNIQUE constraint as defined by the
database-specific DDL syntax. |
protected String |
renderValue(Object value)
Renders the given Java instance in a database-specific way.
|
void |
rollbackTransaction(Connection c) |
protected void |
setPostConnectionProperties(Connection conn)
Called to make any post-creation modifications to a new
Connection instance. |
void |
setQueryResultSetProperties(ResultSet res,
Query query)
Allows the provider to set database-specific options on a
ResultSet instance prior to its use by the library. |
void |
setQueryStatementProperties(Statement stmt,
Query query)
Allows the provider to set database-specific options on a
Statement instance prior to its usage in a SELECT
query. |
String |
shorten(String id) |
protected boolean |
shouldQuoteID(String id)
Determines whether or not the specified identifier should be quoted
before transmission to the underlying database.
|
protected boolean |
shouldQuoteTableName(String tableName)
Determines whether or not the table name should be quoted
before transmission to the underlying database.
|
Connection |
startTransaction() |
String |
withSchema(String tableName) |
protected final org.slf4j.Logger logger
protected final org.slf4j.Logger sqlLogger
protected final TypeManager typeManager
protected AtomicReference<String> quoteRef
protected DatabaseProvider(DisposableDataSource dataSource, String schema, TypeManager typeManager)
protected DatabaseProvider(DisposableDataSource dataSource, String schema)
public TypeManager getTypeManager()
public String getSchema()
protected void loadQuoteString()
public String renderMetadataQuery(String tableName)
protected String renderAutoIncrement()
Generates the DDL fragment required to specify an INTEGER field as
auto-incremented. For databases which do not support such flags (which
is just about every database exception MySQL), ""
is an
acceptable return value. This method should never return null
as it would cause the field rendering method to throw a NullPointerException
.
public final Iterable<SQLAction> renderAction(NameConverters nameConverters, DDLAction action)
DDLAction
into the database-specific SQL actions. If the action is to
create an entity, then each of the SQL actions may have a corresponding undo
action to roll back creation of the entity if necessary.nameConverters
- action
- The database-agnostic action to render.renderTable(net.java.ao.schema.NameConverters, net.java.ao.schema.ddl.DDLTable)
,
#renderFunctions(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable)
,
#renderTriggers(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.SequenceNameConverter, net.java.ao.schema.ddl.DDLTable)
,
#renderSequences(net.java.ao.schema.SequenceNameConverter, net.java.ao.schema.ddl.DDLTable)
,
#renderDropTriggers(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable)
,
#renderDropFunctions(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable)
,
#renderDropSequences(net.java.ao.schema.SequenceNameConverter, net.java.ao.schema.ddl.DDLTable)
,
renderDropTableActions(net.java.ao.schema.NameConverters, net.java.ao.schema.ddl.DDLTable)
,
renderAlterTableAddColumn(net.java.ao.schema.NameConverters, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
,
renderAlterTableChangeColumn(net.java.ao.schema.NameConverters, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField, net.java.ao.schema.ddl.DDLField)
,
renderDropColumnActions(net.java.ao.schema.NameConverters, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
,
renderAlterTableAddKey(DDLForeignKey)
,
renderAlterTableDropKey(DDLForeignKey)
protected Iterable<SQLAction> renderDropColumnActions(NameConverters nameConverters, DDLTable table, DDLField field)
public String renderQuery(Query query, TableNameConverter converter, boolean count)
Top level delegating method for rendering a database-agnostic
Query
object into its (potentially) database-specific
query statement. This method invokes the various renderQuery*
methods to construct its output, thus it is doubtful that any subclasses
will have to override it. Rather, one of the delegate methods
should be considered.
An example of a database-specific query rendering would be the
following Query
:
Query.select().from(Person.class).limit(10)
On MySQL, this would render to SELECT id FROM people LIMIT 10
However, on SQL Server, this same Query would render as
SELECT TOP 10 id FROM people
query
- The database-agnostic Query object to be rendered in a
potentially database-specific way.converter
- Used to convert Entity
classes into table names.count
- If true
, render the Query as a SELECT COUNT(*)
rather than a standard field-data query.renderQuerySelect(Query, TableNameConverter, boolean)
,
renderQueryJoins(Query, TableNameConverter)
,
renderQueryWhere(Query)
,
renderQueryGroupBy(Query)
,
renderQueryOrderBy(Query)
,
renderQueryLimit(Query)
public Object parseValue(int type, String value)
Parses the database-agnostic String
value relevant to the specified SQL
type in int
form (as defined by Types
and returns
the Java value which corresponds. This method is completely database-agnostic, as are
all of all of its delegate methods.
WARNING: This method is being considered for removal to another
class (perhaps TypeManager
?) as it is not a database-specific function and thus
confuses the purpose of this class. Do not rely upon it heavily. (better yet, don't rely on it
at all from external code. It's not designed to be part of the public API)
type
- The JDBC integer type of the database field against which to parse the
value.value
- The database-agnostic String value to parse into a proper Java object
with respect to the specified SQL type.public void setQueryStatementProperties(Statement stmt, Query query) throws SQLException
Allows the provider to set database-specific options on a
Statement
instance prior to its usage in a SELECT
query. This is to allow things like emulation of the
LIMIT feature on databases which don't support it within
the SQL implementation.
This method is only called on SELECTs.
stmt
- The instance against which the properties
should be set.query
- The query which is being executed against
the statement instance.SQLException
public void setQueryResultSetProperties(ResultSet res, Query query) throws SQLException
ResultSet
instance prior to its use by the library.
This allows for features such as row offsetting even on
databases that don't support it (such as Oracle, Derby,
etc).res
- The ResultSet
to modify.query
- The query instance which was run to produce
the result set.SQLException
public ResultSet getTables(Connection conn) throws SQLException
Returns a result set of all of the tables (and associated
meta) in the database. The fields of the result set must
correspond with those specified in the
DatabaseMetaData#getTables(String, String, String, String[])
method. In fact, the default implementation merely calls
this method passing (null, null, "", null)
.
For databases (such as PostgreSQL) where this is unsuitable,
different parameters can be specified to the getTables
method in the override, or an entirely new implementation
written, as long as the result set corresponds in fields to
the JDBC spec.
conn
- The connection to use in retrieving the database tables.SQLException
DatabaseMetaData.getTables(String, String, String, String[])
public ResultSet getSequences(Connection conn) throws SQLException
SQLException
public ResultSet getIndexes(Connection conn, String tableName) throws SQLException
SQLException
public ResultSet getImportedKeys(Connection connection, String tableName) throws SQLException
SQLException
protected String renderQuerySelect(Query query, TableNameConverter converter, boolean count)
Renders the SELECT portion of a given Query
instance in the
manner required by the database-specific SQL implementation. Usually,
this is as simple as "SELECT id FROM table"
or "SELECT DISTINCT
* FROM table"
. However, some databases require the limit and offset
parameters to be specified as part of the SELECT clause. For example,
on HSQLDB, a Query for the "id" field limited to 10 rows would render
SELECT like this: SELECT TOP 10 id FROM table
.
There is usually no need to call this method directly. Under normal
operations it functions as a delegate for renderQuery(Query, TableNameConverter, boolean)
.
query
- The Query instance from which to determine the SELECT properties.converter
- The name converter to allow conversion of the query entity
interface into a proper table name.count
- Whether or not the query should be rendered as a SELECT COUNT(*)
.protected final String queryTableName(Query query, TableNameConverter converter)
protected final String querySelectFields(Query query, TableNameConverter converter)
protected String renderQueryJoins(Query query, TableNameConverter converter)
Renders the JOIN portion of the query in the database-specific SQL dialect. Very few databases deviate from the standard in this matter, thus the default implementation is usually sufficient.
An example return value: " JOIN table1 ON table.id = table1.value"
There is usually no need to call this method directly. Under normal
operations it functions as a delegate for renderQuery(Query, TableNameConverter, boolean)
.
query
- The Query instance from which to determine the JOIN properties.converter
- The name converter to allow conversion of the query entity
interface into a proper table name.protected String renderQueryWhere(Query query)
Renders the WHERE portion of the query in the database-specific SQL dialect. Very few databases deviate from the standard in this matter, thus the default implementation is usually sufficient.
An example return value: " WHERE name = ? OR age < 20"
There is usually no need to call this method directly. Under normal
operations it functions as a delegate for renderQuery(Query, TableNameConverter, boolean)
.
query
- The Query instance from which to determine the WHERE properties.protected String renderQueryGroupBy(Query query)
Renders the GROUP BY portion of the query in the database-specific SQL dialect. Very few databases deviate from the standard in this matter, thus the default implementation is usually sufficient.
An example return value: " GROUP BY name"
There is usually no need to call this method directly. Under normal
operations it functions as a delegate for renderQuery(Query, TableNameConverter, boolean)
.
query
- The Query instance from which to determine the GROUP BY properties.protected String renderQueryHaving(Query query)
Renders the HAVING portion of the query in the database-specific SQL dialect. Very few databases deviate from the standard in this matter, thus the default implementation is usually sufficient.
An example return value: " HAVING COUNT(name) > 2"
There is usually no need to call this method directly. Under normal
operations it functions as a delegate for renderQuery(Query, TableNameConverter, boolean)
.
query
- The Query instance from which to determine the HAVING properties.protected String renderQueryOrderBy(Query query)
Renders the ORDER BY portion of the query in the database-specific SQL dialect. Very few databases deviate from the standard in this matter, thus the default implementation is usually sufficient.
An example return value: " ORDER BY name ASC"
There is usually no need to call this method directly. Under normal
operations it functions as a delegate for renderQuery(Query, TableNameConverter, boolean)
.
query
- The Query instance from which to determine the ORDER BY properties.protected String renderQueryLimit(Query query)
Renders the LIMIT portion of the query in the database-specific SQL dialect. There is wide variety in database implementations of this particular SQL clause. In fact, many database do not support it at all.
Unfortunately, we live in the real world of proprietary database implementations that requires us to use database specific keywords or semantics to achieve these outcomes. Appropriate methods should be overridden in such cases.
An example return value: " LIMIT 10,2"
There is usually no need to call this method directly. Under normal
operations it functions as a delegate for renderQuery(Query, TableNameConverter, boolean)
.
query
- The Query instance from which to determine the LIMIT properties.public final Connection getConnection() throws SQLException
Retrieves a JDBC Connection
instance which corresponds
to the database represented by the provider instance. This Connection
can be used to execute arbitrary JDBC operations against the database.
Also, this is the method used by the whole of ActiveObjects itself to
get database connections when required.
All Connection
instances are pooled internally by thread.
Thus, there is never more than one connection per thread. This is
necessary to allow arbitrary JDBC operations within a transaction
without breaking transaction integrity. Developers using this method
should bear this fact in mind and consider the Connection
instance immutable. The only exception is if one is absolutely
certain that the JDBC code in question is not being executed within
a transaction.
Despite the fact that there is only a single connection per thread,
the Connection
instances returned from this method should
still be treated as bona fide JDBC connections. They can and
should be closed when their usage is complete. This is
especially important when actual connection pooling is in use and
non-disposal of connections can lead to a crash as the connection
pool runs out of resources. The developer need not concern themselves
with the single-connection-per-thread issue when closing the connection
as the call to close()
will be intercepted and ignored
if necessary.
SQLException
public final Connection startTransaction() throws SQLException
SQLException
public final Connection commitTransaction(Connection c) throws SQLException
SQLException
public final void rollbackTransaction(Connection c) throws SQLException
SQLException
public void dispose()
dispose
in interface Disposable
protected void setPostConnectionProperties(Connection conn) throws SQLException
Connection
instance. This is used for databases
such as Derby which require the schema to be set after
the connection is created.conn
- The connection to modify according to the database
requirements.SQLException
protected String renderConstraintsForTable(UniqueNameConverter uniqueNameConverter, DDLTable table)
uniqueNameConverter
- table
- The database-agnostic DDL representation of the table
in question.renderForeignKey(DDLForeignKey)
protected String renderConstraints(NameConverters nameConverters, List<String> primaryKeys, DDLTable table)
protected String renderForeignKey(DDLForeignKey key)
DDLForeignKey#getFKName()
value otherwise migrations will no longer function appropriately.key
- The database-agnostic foreign key representation.protected String convertTypeToString(TypeInfo<?> type)
DatabaseType#getDefaultName()
method. Subclass implementations should be sure to make a super
call in order to ensure that both default naming and future special
cases are handled appropriately.type
- The type instance to convert to a DDL string.protected final SQLAction renderTable(NameConverters nameConverters, DDLTable table)
DatabaseProvider
methods for functions such as
field rendering, foreign key rendering, etc.nameConverters
- table
- The database-agnostic table representation.protected SQLAction renderDropTableStatement(DDLTable table)
"DROP TABLE tablename"
. This is suitable
for every database that I am aware of. Any dependent database
objects (such as triggers, functions, etc) must be rendered in
one of the other delegate methods (such as renderDropTriggers(DDLTable)
).table
- The table representation which is to be dropped.protected final Iterable<SQLAction> renderAccessories(NameConverters nameConverters, DDLTable table)
Generates the database-specific DDL statements required to create
all of the functions, sequences, and triggers necessary for the given table,
by calling renderAccessoriesForField(NameConverters, DDLTable, DDLField)
for each of the table's fields. Each returned SQLAction
has a
correspondingundo action
that deletes
the corresponding function, sequence, or trigger.
nameConverters
- table
- The table for which the objects must be generated.SQLAction
s.protected final Iterable<SQLAction> renderDropAccessories(NameConverters nameConverters, DDLTable table)
Generates the database-specific DDL statements required to drop
all of the functions, sequences, and triggers associated with the given table,
by calling renderDropAccessoriesForField(NameConverters, DDLTable, DDLField)
for each of the table's fields.
nameConverters
- table
- The table for which the objects must be dropped.SQLAction
s.protected Iterable<SQLAction> renderAccessoriesForField(NameConverters nameConverters, DDLTable table, DDLField field)
SQLAction
should have a corresponding undo action
that deletes
the corresponding function, sequence, or trigger. The default implementation returns
an empty list.nameConverters
- table
- field
- SQLAction
sprotected Iterable<SQLAction> renderDropAccessoriesForField(NameConverters nameConverters, DDLTable table, DDLField field)
nameConverters
- table
- field
- SQLAction
s@Deprecated protected final Iterable<SQLAction> renderFields(DDLTable table, com.google.common.base.Predicate<DDLField> filter, com.google.common.base.Function<DDLField,Iterable<SQLAction>> render)
protected final Iterable<SQLAction> renderFields(DDLTable table, Predicate<DDLField> filter, Function<DDLField,Iterable<SQLAction>> render)
protected Iterable<SQLAction> renderAlterTableAddColumn(NameConverters nameConverters, DDLTable table, DDLField field)
@OnUpdate
annotation, chances are there will be a trigger and possibly a
function along with the ALTER statement. These "extra"
functions are properly ordered and will only be appended if
their values are not null
. Because of this, very
few database providers will need to override this method.
Each SQLAction
should have a corresponding undo action;
these will be executed in reverse order if the action needs to
be rolled back.
nameConverters
- table
- The table which should receive the new column.field
- The column to add to the specified table.#renderFunctionForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
,
#renderTriggerForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.SequenceNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
protected SQLAction renderAlterTableAddColumnStatement(NameConverters nameConverters, DDLTable table, DDLField field)
nameConverters
- table
- The table which should receive the new column.field
- The column to add to the specified table.protected Iterable<SQLAction> renderAlterTableChangeColumn(NameConverters nameConverters, DDLTable table, DDLField oldField, DDLField field)
Generates the database-specific DDL statements required to change the given column from its old specification to the given DDL value. This method will also generate the appropriate statements to remove old triggers and functions, as well as add new ones according to the requirements of the new field definition.
The default implementation of this method functions in the manner specified by the MySQL database. Some databases will have to perform more complicated actions (such as dropping and re-adding the field) in order to satesfy the same use-case. Such databases should print a warning to stderr to ensure that the end-developer is aware of such restrictions.
Thus, the specification for this method allows for data loss. Nevertheless, if the database supplies a mechanism to accomplish the task without data loss, it should be applied.
For maximum flexibility, the default implementation of this method
only deals with the dropping and addition of functions and triggers.
The actual generation of the ALTER TABLE statement is done in the
renderAlterTableChangeColumnStatement(net.java.ao.schema.NameConverters, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField, net.java.ao.schema.ddl.DDLField, net.java.ao.DatabaseProvider.RenderFieldOptions)
method.
nameConverters
- table
- The table containing the column to change.oldField
- The old column definition.field
- The new column definition (defining the resultant DDL). @return An array of DDL statements to be executed.#getTriggerNameForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
,
#getFunctionNameForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
,
#renderFunctionForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
,
#renderTriggerForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.SequenceNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
protected DatabaseProvider.RenderFieldOptions renderFieldOptionsInAlterColumn()
protected SQLAction renderAlterTableChangeColumnStatement(NameConverters nameConverters, DDLTable table, DDLField oldField, DDLField field, DatabaseProvider.RenderFieldOptions options)
renderAlterTableChangeColumn(net.java.ao.schema.NameConverters, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField, net.java.ao.schema.ddl.DDLField)
method,
for which it is a primary delegate. The default implementation of this
method functions according to the MySQL specification.nameConverters
- table
- The table containing the column to change.oldField
- The old column definition.field
- The new column definition (defining the resultant DDL).options
- renderField(net.java.ao.schema.NameConverters, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField, net.java.ao.DatabaseProvider.RenderFieldOptions)
protected Iterable<SQLAction> renderAlterTableDropColumn(NameConverters nameConverters, DDLTable table, DDLField field)
nameConverters
- table
- The table from which to drop the column.field
- The column definition to remove from the table.#getTriggerNameForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
,
#getFunctionNameForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
protected SQLAction renderAlterTableDropColumnStatement(DDLTable table, DDLField field)
protected SQLAction renderAlterTableAddKey(DDLForeignKey key)
null
value returned.key
- The foreign key to be added. As this instance contains
all necessary data (such as domestic table, field, etc), no
additional parameters are required.null
.renderForeignKey(DDLForeignKey)
protected SQLAction renderAlterTableDropKey(DDLForeignKey key)
null
value returned. This method assumes that the
renderForeignKey(DDLForeignKey)
method properly names
the foreign key according to the DDLForeignKey.getFKName()
method.key
- The foreign key to be removed. As this instance contains
all necessary data (such as domestic table, field, etc), no
additional parameters are required.null
.protected SQLAction renderCreateIndex(IndexNameConverter indexNameConverter, DDLIndex index)
null
returned.indexNameConverter
- index
- The index to create. This single instance contains all
of the data necessary to create the index, thus no separate
parameters (such as a DDLTable
) are required.@Deprecated public SQLAction renderCreateCompositeIndex(String tableName, String indexName, List<String> fields)
renderCreateIndex(IndexNameConverter, DDLIndex)
for creating indexes.tableName
- The name of the database tableindexName
- The name of the new indexfields
- List of fields that make up the indexprotected SQLAction renderDropIndex(IndexNameConverter indexNameConverter, DDLIndex index)
null
returned.indexNameConverter
- index
- The index to drop. This single instance contains all
of the data necessary to drop the index, thus no separate
parameters (such as a DDLTable
) are required.null
.protected boolean hasIndex(IndexNameConverter indexNameConverter, DDLIndex index)
protected String renderAppend()
Generates any database-specific options which must be appended to the end of a table definition. The only database I am aware of which requires this is MySQL. For example:
CREATE TABLE test ( id INTEGER NOT NULL AUTO_INCREMENT, name VARCHAR(45), PRIMARY KEY(id) ) ENGINE=InnoDB;
The "ENGINE=InnoDB
" clause is what is returned by
this method. The default implementation simply returns
null
, signifying that no append should be rendered.
null
protected final String renderField(NameConverters nameConverters, DDLTable table, DDLField field, DatabaseProvider.RenderFieldOptions options)
Generates the database-specific DDL fragment required to render the
field and its associated type. This includes all field attributes,
such as @NotNull
, @AutoIncrement
(if
supported by the database at the field level) and so on. Sample
return value:
name VARCHAR(255) DEFAULT "Skye" NOT NULL
Certain databases don't allow defined precision for certain types
(such as Derby and INTEGER). The logic for whether or not to render
precision should not be within this method, but delegated to the
#considerPrecision(DDLField)
method.
Almost all functionality within this method is delegated to other methods within the implementation. As such, it is almost never necessary to override this method directly. An exception to this would be a database like PostgreSQL which requires a different type for auto-incremented fields.
nameConverters
- table
- field
- The field to be rendered.options
- protected String renderValue(Object value)
Calendar
,
Boolean
(which is always rendered as 0/1), functions,
null
and numbers. All other values are rendered (by
default) as 'value.toString()'
(the String value
enclosed within single quotes). Implementations are encouraged to
override this method as necessary.value
- The Java instance to be rendered as a database literal.renderDate(Date)
protected String renderDate(Date date)
Date
instance as a DATETIME literal
in the database-specific format. The return value should not
be enclosed within quotes, as this is accomplished within other
functions when rendering is required. This method is actually a
boiler-plate usage of the SimpleDateFormat
class, using the
date format defined within the getDateFormat()
method.date
- The time instance to be rendered.protected String renderUnique(UniqueNameConverter uniqueNameConverter, DDLTable table, DDLField field)
UNIQUE
constraint as defined by the
database-specific DDL syntax. This method is a delegate of other, more
complex methods such as renderField(net.java.ao.schema.NameConverters, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField, net.java.ao.DatabaseProvider.RenderFieldOptions)
. The default
implementation just returns UNIQUE
. Implementations may
override this method to return an empty String
if the database
in question does not support the constraint.uniqueNameConverter
- table
- field
- UNIQUE
.protected String getDateFormat()
SimpleDateFormat
syntax. This format should include
the time down to the second (or even more precise, if allowed by
the database). The default implementation returns the format for
MySQL, which is: yyyy-MM-dd HH:mm:ss
protected String renderFieldType(DDLField field)
convertTypeToString(TypeInfo)
method, passing the field type. Thus, it is rarely necessary
(if ever) to override this method. It may be deprecated in a
future release.field
- The field which contains the type to be rendered.public Object handleBlob(ResultSet res, Class<?> type, String field) throws SQLException
SQLException
protected String _getTriggerNameForField(TriggerNameConverter triggerNameConverter, DDLTable table, DDLField field)
null
should be returned.
This function is to allow for databases which require the use of
triggers on a field to allow for certain functionality (like
ON UPDATE). The default implementation returns null
.triggerNameConverter
- table
- The table which contains the field for which a trigger
may or may not exist.field
- The field for which a previous migration may have
created a trigger.null
if none.#renderTriggerForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.SequenceNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
protected SQLAction _renderTriggerForField(NameConverters nameConverters, DDLTable table, DDLField field)
null
if none. This is to allow for databases which
require the use of triggers to provide functionality such as ON
UPDATE. The default implementation returns null
.nameConverters
- table
- The table containing the field for which a trigger
may need to be rendered.field
- The field for which the trigger should be rendered,
if any. @return A database-specific DDL statement creating a trigger for
the field in question, or null
.#getTriggerNameForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
protected SQLAction _renderDropTriggerForField(NameConverters nameConverters, DDLTable table, DDLField field)
null
if none.nameConverters
- table
- The table containing the field for which a trigger
may need to be rendered.field
- The field for which the trigger should be rendered,
if any. @return A database-specific DDL statement to drop a trigger for
the field in question, or null
.protected String _getFunctionNameForField(TriggerNameConverter triggerNameConverter, DDLTable table, DDLField field)
null
should be returned.
This method is to allow for databases which require the use of
explicitly created functions which correspond to triggers (e.g.
PostgreSQL). Few providers will need to override the default
implementation of this method, which returns null
.triggerNameConverter
- table
- The table which contains the field for which a function
may or may not exist.field
- The field for which a previous migration may have
created a function.null
if none.protected SQLAction _renderFunctionForField(NameConverters nameConverters, DDLTable table, DDLField field)
null
if none. This is to allow for databases which
require the use of triggers and explicitly created functions to
provide functionality such as ON UPDATE (e.g. PostgreSQL). The
default implementation returns null
.nameConverters
- table
- The table containing the field for which a function
may need to be rendered.field
- The field for which the function should be rendered,
if any.null
.#getFunctionNameForField(net.java.ao.schema.TriggerNameConverter, net.java.ao.schema.ddl.DDLTable, net.java.ao.schema.ddl.DDLField)
protected SQLAction _renderDropFunctionForField(NameConverters nameConverters, DDLTable table, DDLField field)
null
otherwise.nameConverters
- table
- The table containing the field for which a function
may need to be rendered.field
- The field for which the function should be rendered,
if any.null
.protected SQLAction _renderSequenceForField(NameConverters nameConverters, DDLTable table, DDLField field)
null
if none. The default implementation returns null
.nameConverters
- table
- The table containing the field for which a sequence
may need to be rendered.field
- The field for which the sequence should be rendered,
if any.null
.protected SQLAction _renderDropSequenceForField(NameConverters nameConverters, DDLTable table, DDLField field)
null
if none.public <T extends RawEntity<K>,K> K insertReturningKey(EntityManager manager, Connection conn, Class<T> entityType, Class<K> pkType, String pkField, boolean pkIdentity, String table, DBParam... params) throws SQLException
Generates an INSERT statement to be used to create a new row in the
database, returning the primary key value. This method also invokes
the delegate method, #executeInsertReturningKey(EntityManager, java.sql.Connection, Class, String, String, DBParam...)
passing the appropriate parameters and query. This method is required
because some databases do not support the JDBC parameter
RETURN_GENERATED_KEYS
(such as HSQLDB and PostgreSQL).
Also, some databases (such as MS SQL Server) require odd tricks to
support explicit value passing to auto-generated fields. This method
should take care of any extra queries or odd SQL generation required
to implement both auto-generated primary key returning, as well as
explicit primary key value definition.
Overriding implementations of this method should be sure to use the
Connection
instance passed to the method, not a new
instance generated using the getConnection()
method. This is
because this method is in fact a delegate invoked by EntityManager
as part of the entity creation process and may be part of a transaction,
a bulk creation or some more complicated operation. Both optimization
and usage patterns on the API dictate that the specified connection
instance be used. Implementations may assume that the given connection
instance is never null
.
The default implementation of this method should be sufficient for any
fully compliant ANSI SQL database with a properly implemented JDBC
driver. Note that this method should not not actually execute
the SQL it generates, but pass it on to the #executeInsertReturningKey(EntityManager, java.sql.Connection, Class, String, String, DBParam...)
method, allowing for functional delegation and better extensibility.
However, this method may execute any additional statements required to
prepare for the INSERTion (as in the case of MS SQL Server which requires
some config parameters to be set on the database itself prior to INSERT).
manager
- The EntityManager
which was used to dispatch
the INSERT in question.conn
- The connection to be used in the eventual execution of the
generated SQL statement.entityType
- The Java class of the entity.pkType
- The Java type of the primary key value. Can be used to
perform a linear search for a specified primary key value in the
params
list. The return value of the method must be of
the same type.pkField
- The database field which is the primary key for the
table in question. Can be used to perform a linear search for a
specified primary key value in the params
list.pkIdentity
- Flag indicating whether or not the primary key field
is auto-incremented by the database (IDENTITY field).table
- The name of the table into which the row is to be INSERTed.params
- A varargs array of parameters to be passed to the
INSERT statement. This may include a specified value for the
primary key.SQLException
- If the INSERT fails in the delegate method, or
if any additional statements fail with an exception.#executeInsertReturningKey(EntityManager, java.sql.Connection, Class, String, String, DBParam...)
public <T extends RawEntity<K>,K> void insertBatch(EntityManager manager, Connection conn, Class<T> entityType, Class<K> pkType, String pkField, boolean pkIdentity, String table, List<Map<String,Object>> rows) throws SQLException
manager
- The EntityManager
which was used to dispatch
the INSERT in question.conn
- The connection to be used in the eventual execution of the
generated SQL statement.entityType
- The Java class of the entity.pkType
- The Java type of the primary key value.pkField
- The database field which is the primary key for the
table in question.pkIdentity
- Flag indicating whether or not the primary key field
is auto-incremented by the database (IDENTITY field).table
- The name of the table into which the rows is to be INSERTed.rows
- A list of rows to be INSERTed. A row is represented as a map from column name to its value.
All rows must have the same columns.SQLException
protected <T extends RawEntity<K>,K> K executeInsertReturningKey(EntityManager manager, Connection conn, Class<T> entityType, Class<K> pkType, String pkField, String sql, DBParam... params) throws SQLException
Delegate method to execute an INSERT statement returning any auto-generated
primary key values. This method is primarily designed to be called as a delegate
from the #insertReturningKey(EntityManager, Connection, Class, String, boolean, String, DBParam...)
method. The idea behind this method is to allow custom implementations to
override this method to potentially execute other statements (such as getting the
next value in a sequence) rather than the default implementaiton which uses the
JDBC constant, RETURN_GENERATED_KEYS
. Any database which has a
fully-implemented JDBC driver should have no problems with the default
implementation of this method.
Part of the design behind splitting insertReturningKey
and
executeInsertReturningKey
is so that logic for generating the actual
INSERT statement need not be duplicated throughout the code and in custom
implementations providing trivial changes to the default algorithm. This method
should avoid actually generating SQL if at all possible.
This method should iterate through the passed DBParam(s)
to
ensure that no primary key value was explicitly specified. If one was, it
should be used in leiu of one which is auto-generated by the database. Also,
it is this value which should be returned if specified, rather than the value
which would have been generated or null
. As such, this method
should always return exactly the value of the primary key field in the row which
was just inserted, regardless of what that value may be.
In cases where the database mechanism for getting the next primary key value
is not thread safe, this method should be declared synchronized
,
or some thread synchronization technique employed. Unfortunately, it is not
always possible to ensure that no other INSERT could (potentially) "steal" the
expected value out from under the algorithm. Such scenarios are to be avoided
when possible, but the algorithm need not take extremely escoteric concurrency
cases into account. (see the HSQLDB provider for an example of such a
less-than-thorough asynchronous algorithm)
IMPORTANT: The INSERT Statement
must use the specified
connection, rather than a new one retrieved from getConnection()
or
equivalent. This is because the INSERT may be part of a bulk insertion, a
transaction, or possibly another such operation. It is also important to note
that this method should not close the connection. Doing so could cause the
entity creation algorithm to fail at a higher level up the stack.
manager
- The EntityManager
which was used to dispatch
the INSERT in question.conn
- The database connection to use in executing the INSERT statement.entityType
- The Java class of the entity.pkType
- The Java class type of the primary key field (for use both in
searching the params
as well as performing value conversion
of auto-generated DB values into proper Java instances).pkField
- The database field which is the primary key for the
table in question. Can be used to perform a linear search for a
specified primary key value in the params
list.params
- A varargs array of parameters to be passed to the
INSERT statement. This may include a specified value for the
primary key. @throws SQLException If the INSERT fails in the delegate method, or
if any additional statements fail with an exception.SQLException
#insertReturningKey(EntityManager, Connection, Class, String, boolean, String, DBParam...)
public void putNull(PreparedStatement stmt, int index) throws SQLException
NULL
value in the database. This method
is required due to the fact that not all JDBC drivers handle NULLs
in the same fashion. The default implementation calls PreparedStatement.setNull(int, int)
,
retrieving parameter type from metadata. Databases which require a
different implementation (e.g. PostgreSQL) should override this method.stmt
- The statement in which to store the NULL
value.index
- The index of the parameter which should be assigned NULL
.SQLException
public void putBoolean(PreparedStatement stmt, int index, boolean value) throws SQLException
BOOLEAN
value in the database. Most databases
handle differences in BOOLEAN
semantics within their JDBC
driver(s). However, some do not implement the PreparedStatement.setBoolean(int, boolean)
method correctly. To work around this defect, any database providers
for such databases should override this method to store boolean values in
the relevant fashion.stmt
- The statement in which to store the BOOLEAN
value.index
- The index of the parameter which should be assigned.value
- The value to be stored in the relevant field.SQLException
protected boolean isNumericType(int type)
Number
. The default implementation
should be suitable for every conceivable use-case.type
- The JDBC type which is to be tested.true
if the specified type represents a numeric
type, otherwise false
.public final String processID(String id)
Performs any database specific post-processing on the specified identifier. This usually means quoting reserved words, though it could potentially encompass other tasks. This method is called with unbelievable frequency and thus must return extremely quickly.
The default implementation checks two factors: max identifier
length and whether or not it represents a reserved word in the
underlying database. If the identifier exceeds the maximum ID
length for the database in question, the excess text will be
hashed against the hash code for the whole and concatenated with
the leading remainder. The shouldQuoteID(String)
method
is utilitized to determine whether or not the identifier in question
should be quoted. For most databases, this involves checking a set
of reserved words, but the method is flexible enough to allow more
complex "reservations rules" (such as those required by Oracle).
If the identifier is reserved in any way, the database-specific
quoting string will be retrieved from DatabaseMetaData
and
used to enclose the identifier. This method cannot simply quote all
identifiers by default due to the way that some databases (such as
HSQLDB and PostgreSQL) attach extra significance to quoted fields.
The general assurance of this method is that for any input identifier, this method will return a correspondingly-unique identifier which is guaranteed to be valid within the underlying database.
id
- The identifier to process.getMaxIDLength()
,
shouldQuoteID(String)
public final String processID(Query.FieldMetadata fmd)
Performs any database specific post-processing on the specified field identifier. This usually means quoting reserved words, though it could potentially encompass other tasks. This method is called with unbelievable frequency and thus must return extremely quickly. This method will quote both the column name as well as the alias, if required.
The default implementation checks two factors: max identifier
length and whether or not it represents a reserved word in the
underlying database. If the identifier exceeds the maximum ID
length for the database in question, the excess text will be
hashed against the hash code for the whole and concatenated with
the leading remainder. The shouldQuoteID(String)
method
is utilitized to determine whether or not the identifier in question
should be quoted. For most databases, this involves checking a set
of reserved words, but the method is flexible enough to allow more
complex "reservations rules" (such as those required by Oracle).
If the identifier is reserved in any way, the database-specific
quoting string will be retrieved from DatabaseMetaData
and
used to enclose the identifier. This method cannot simply quote all
identifiers by default due to the way that some databases (such as
HSQLDB and PostgreSQL) attach extra significance to quoted fields.
The general assurance of this method is that for any input identifier, this method will return a correspondingly-unique identifier which is guaranteed to be valid within the underlying database.
fmd
- The metadata about the field, including the parsed alias and aggregate function (if any)getMaxIDLength()
,
shouldQuoteID(String)
public final String processTableName(String tableName)
processID(String)
but use
#shouldQuoteTableName(String) to check if the table name should
be quoted or not.tableName
- The table name to process.getMaxIDLength()
,
shouldQuoteTableName(String)
protected final boolean isSchemaNotEmpty()
protected boolean shouldQuoteID(String id)
getReservedWords()
. Databases with more complicated
rules regarding quoting should provide a custom implementation of this
method.id
- The identifier to check against the quoting rules.true
if the specified identifier is invalid under
the relevant quoting rules, otherwise false
.protected boolean shouldQuoteTableName(String tableName)
shouldQuoteID(String)
but can be
overridden by subclasses.tableName
- The table name to check against the quoting rules.true
if the table name is invalid under
the relevant quoting rules, otherwise false
.protected int getMaxIDLength()
Integer.MAX_VALUE
.protected abstract Set<String> getReservedWords()
Set
instance returned from this
method should guarentee O(1) lookup times, otherwise ORM performance
will suffer greatly.public boolean isCaseSensitive()
SchemaReader
utility. The default value is true
. Note that databases which
support both case-sensetive and case-insensetive identifiers (like MySQL) should
return true
for better all-around compatibility.true
if identifiers are case-sensetive,
false
otherwise.public void handleUpdateError(String sql, SQLException e) throws SQLException
sql
- e
- the SQLException
that occured.SQLException
- throws the SQLException if it should not be ignored.public final PreparedStatement preparedStatement(Connection c, CharSequence sql) throws SQLException
SQLException
public final PreparedStatement preparedStatement(Connection c, CharSequence sql, int autoGeneratedKeys) throws SQLException
SQLException
public final PreparedStatement preparedStatement(Connection c, CharSequence sql, int resultSetType, int resultSetConcurrency) throws SQLException
SQLException
public final void executeUpdate(Statement stmt, CharSequence sql) throws SQLException
SQLException
public final Iterable<String> executeUpdatesForActions(Statement stmt, Iterable<SQLAction> actions, Set<String> completedStatements) throws SQLException
provider
- stmt
- A JDBC Statement that will be reused for all updatesactions
- A list of SQLAction
s to executecompletedStatements
- A set of SQL statements that should not be executed if we encounter the same one again.
This is necessary because our schema diff logic is not as smart as it could be, so it may
tell us, for instance, to create an index for a new column even though the statements for
creating the column also included creation of the index.SQLException
public final Iterable<String> executeUpdateForAction(Statement stmt, SQLAction action, Set<String> completedStatements) throws SQLException
SQLException
public final void addSqlListener(DatabaseProvider.SqlListener l)
public final void removeSqlListener(DatabaseProvider.SqlListener l)
protected final void onSql(String sql)
protected Iterable<DDLForeignKey> findForeignKeysForField(DDLTable table, DDLField field)
Copyright © 2007–2022 Atlassian. All rights reserved.