org.sqlite.jdbc3.JDBC3DatabaseMetaData Maven / Gradle / Ivy
package org.sqlite.jdbc3;
import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.DatabaseMetaData;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.SQLFeatureNotSupportedException;
import java.sql.Statement;
import java.sql.Struct;
import java.sql.Types;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors;
import org.sqlite.SQLiteConnection;
import org.sqlite.core.CoreDatabaseMetaData;
import org.sqlite.core.CoreStatement;
import org.sqlite.jdbc3.JDBC3DatabaseMetaData.ImportedKeyFinder.ForeignKey;
import org.sqlite.util.Logger;
import org.sqlite.util.LoggerFactory;
import org.sqlite.util.QueryUtils;
import org.sqlite.util.StringUtils;
public abstract class JDBC3DatabaseMetaData extends CoreDatabaseMetaData {
private static String driverName;
private static String driverVersion;
static {
try (InputStream sqliteJdbcPropStream =
JDBC3DatabaseMetaData.class
.getClassLoader()
.getResourceAsStream("sqlite-jdbc.properties")) {
if (sqliteJdbcPropStream == null) {
throw new IOException("Cannot load sqlite-jdbc.properties from jar");
}
final Properties sqliteJdbcProp = new Properties();
sqliteJdbcProp.load(sqliteJdbcPropStream);
driverName = sqliteJdbcProp.getProperty("name");
driverVersion = sqliteJdbcProp.getProperty("version");
} catch (Exception e) {
// Default values
driverName = "SQLite JDBC";
driverVersion = "3.0.0-UNKNOWN";
}
}
protected JDBC3DatabaseMetaData(SQLiteConnection conn) {
super(conn);
}
/** @see java.sql.DatabaseMetaData#getConnection() */
public Connection getConnection() {
return conn;
}
/** @see java.sql.DatabaseMetaData#getDatabaseMajorVersion() */
public int getDatabaseMajorVersion() throws SQLException {
return Integer.parseInt(conn.libversion().split("\\.")[0]);
}
/** @see java.sql.DatabaseMetaData#getDatabaseMinorVersion() */
public int getDatabaseMinorVersion() throws SQLException {
return Integer.parseInt(conn.libversion().split("\\.")[1]);
}
/** @see java.sql.DatabaseMetaData#getDriverMajorVersion() */
public int getDriverMajorVersion() {
return Integer.parseInt(driverVersion.split("\\.")[0]);
}
/** @see java.sql.DatabaseMetaData#getDriverMinorVersion() */
public int getDriverMinorVersion() {
return Integer.parseInt(driverVersion.split("\\.")[1]);
}
/** @see java.sql.DatabaseMetaData#getJDBCMajorVersion() */
public int getJDBCMajorVersion() {
return 4;
}
/** @see java.sql.DatabaseMetaData#getJDBCMinorVersion() */
public int getJDBCMinorVersion() {
return 2;
}
/** @see java.sql.DatabaseMetaData#getDefaultTransactionIsolation() */
public int getDefaultTransactionIsolation() {
return Connection.TRANSACTION_SERIALIZABLE;
}
/** @see java.sql.DatabaseMetaData#getMaxBinaryLiteralLength() */
public int getMaxBinaryLiteralLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxCatalogNameLength() */
public int getMaxCatalogNameLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxCharLiteralLength() */
public int getMaxCharLiteralLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxColumnNameLength() */
public int getMaxColumnNameLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxColumnsInGroupBy() */
public int getMaxColumnsInGroupBy() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxColumnsInIndex() */
public int getMaxColumnsInIndex() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxColumnsInOrderBy() */
public int getMaxColumnsInOrderBy() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxColumnsInSelect() */
public int getMaxColumnsInSelect() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxColumnsInTable() */
public int getMaxColumnsInTable() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxConnections() */
public int getMaxConnections() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxCursorNameLength() */
public int getMaxCursorNameLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxIndexLength() */
public int getMaxIndexLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxProcedureNameLength() */
public int getMaxProcedureNameLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxRowSize() */
public int getMaxRowSize() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxSchemaNameLength() */
public int getMaxSchemaNameLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxStatementLength() */
public int getMaxStatementLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxStatements() */
public int getMaxStatements() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxTableNameLength() */
public int getMaxTableNameLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxTablesInSelect() */
public int getMaxTablesInSelect() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getMaxUserNameLength() */
public int getMaxUserNameLength() {
return 0;
}
/** @see java.sql.DatabaseMetaData#getResultSetHoldability() */
public int getResultSetHoldability() {
return ResultSet.CLOSE_CURSORS_AT_COMMIT;
}
/** @see java.sql.DatabaseMetaData#getSQLStateType() */
public int getSQLStateType() {
return DatabaseMetaData.sqlStateSQL99;
}
/** @see java.sql.DatabaseMetaData#getDatabaseProductName() */
public String getDatabaseProductName() {
return "SQLite";
}
/** @see java.sql.DatabaseMetaData#getDatabaseProductVersion() */
public String getDatabaseProductVersion() throws SQLException {
return conn.libversion();
}
/** @see java.sql.DatabaseMetaData#getDriverName() */
public String getDriverName() {
return driverName;
}
/** @see java.sql.DatabaseMetaData#getDriverVersion() */
public String getDriverVersion() {
return driverVersion;
}
/** @see java.sql.DatabaseMetaData#getExtraNameCharacters() */
public String getExtraNameCharacters() {
return "";
}
/** @see java.sql.DatabaseMetaData#getCatalogSeparator() */
public String getCatalogSeparator() {
return ".";
}
/** @see java.sql.DatabaseMetaData#getCatalogTerm() */
public String getCatalogTerm() {
return "catalog";
}
/** @see java.sql.DatabaseMetaData#getSchemaTerm() */
public String getSchemaTerm() {
return "schema";
}
/** @see java.sql.DatabaseMetaData#getProcedureTerm() */
public String getProcedureTerm() {
return "not_implemented";
}
/** @see java.sql.DatabaseMetaData#getSearchStringEscape() */
public String getSearchStringEscape() {
return "\\";
}
/** @see java.sql.DatabaseMetaData#getIdentifierQuoteString() */
public String getIdentifierQuoteString() {
return "\"";
}
/**
* @see java.sql.DatabaseMetaData#getSQLKeywords()
* @see SQLite Keywords
*/
public String getSQLKeywords() {
return "ABORT,ACTION,AFTER,ANALYZE,ATTACH,AUTOINCREMENT,BEFORE,"
+ "CASCADE,CONFLICT,DATABASE,DEFERRABLE,DEFERRED,DESC,DETACH,"
+ "EXCLUSIVE,EXPLAIN,FAIL,GLOB,IGNORE,INDEX,INDEXED,INITIALLY,INSTEAD,ISNULL,"
+ "KEY,LIMIT,NOTNULL,OFFSET,PLAN,PRAGMA,QUERY,"
+ "RAISE,REGEXP,REINDEX,RENAME,REPLACE,RESTRICT,"
+ "TEMP,TEMPORARY,TRANSACTION,VACUUM,VIEW,VIRTUAL";
}
/** @see java.sql.DatabaseMetaData#getNumericFunctions() */
public String getNumericFunctions() {
return "";
}
/** @see java.sql.DatabaseMetaData#getStringFunctions() */
public String getStringFunctions() {
return "";
}
/** @see java.sql.DatabaseMetaData#getSystemFunctions() */
public String getSystemFunctions() {
return "";
}
/** @see java.sql.DatabaseMetaData#getTimeDateFunctions() */
public String getTimeDateFunctions() {
return "DATE,TIME,DATETIME,JULIANDAY,STRFTIME";
}
/** @see java.sql.DatabaseMetaData#getURL() */
public String getURL() {
return conn.getUrl();
}
/** @see java.sql.DatabaseMetaData#getUserName() */
public String getUserName() {
return null;
}
/** @see java.sql.DatabaseMetaData#allProceduresAreCallable() */
public boolean allProceduresAreCallable() {
return false;
}
/** @see java.sql.DatabaseMetaData#allTablesAreSelectable() */
public boolean allTablesAreSelectable() {
return true;
}
/** @see java.sql.DatabaseMetaData#dataDefinitionCausesTransactionCommit() */
public boolean dataDefinitionCausesTransactionCommit() {
return false;
}
/** @see java.sql.DatabaseMetaData#dataDefinitionIgnoredInTransactions() */
public boolean dataDefinitionIgnoredInTransactions() {
return false;
}
/** @see java.sql.DatabaseMetaData#doesMaxRowSizeIncludeBlobs() */
public boolean doesMaxRowSizeIncludeBlobs() {
return false;
}
/** @see java.sql.DatabaseMetaData#deletesAreDetected(int) */
public boolean deletesAreDetected(int type) {
return false;
}
/** @see java.sql.DatabaseMetaData#insertsAreDetected(int) */
public boolean insertsAreDetected(int type) {
return false;
}
/** @see java.sql.DatabaseMetaData#isCatalogAtStart() */
public boolean isCatalogAtStart() {
return true;
}
/** @see java.sql.DatabaseMetaData#locatorsUpdateCopy() */
public boolean locatorsUpdateCopy() {
return false;
}
/** @see java.sql.DatabaseMetaData#nullPlusNonNullIsNull() */
public boolean nullPlusNonNullIsNull() {
return true;
}
/** @see java.sql.DatabaseMetaData#nullsAreSortedAtEnd() */
public boolean nullsAreSortedAtEnd() {
return !nullsAreSortedAtStart();
}
/** @see java.sql.DatabaseMetaData#nullsAreSortedAtStart() */
public boolean nullsAreSortedAtStart() {
return true;
}
/** @see java.sql.DatabaseMetaData#nullsAreSortedHigh() */
public boolean nullsAreSortedHigh() {
return true;
}
/** @see java.sql.DatabaseMetaData#nullsAreSortedLow() */
public boolean nullsAreSortedLow() {
return !nullsAreSortedHigh();
}
/** @see java.sql.DatabaseMetaData#othersDeletesAreVisible(int) */
public boolean othersDeletesAreVisible(int type) {
return false;
}
/** @see java.sql.DatabaseMetaData#othersInsertsAreVisible(int) */
public boolean othersInsertsAreVisible(int type) {
return false;
}
/** @see java.sql.DatabaseMetaData#othersUpdatesAreVisible(int) */
public boolean othersUpdatesAreVisible(int type) {
return false;
}
/** @see java.sql.DatabaseMetaData#ownDeletesAreVisible(int) */
public boolean ownDeletesAreVisible(int type) {
return false;
}
/** @see java.sql.DatabaseMetaData#ownInsertsAreVisible(int) */
public boolean ownInsertsAreVisible(int type) {
return false;
}
/** @see java.sql.DatabaseMetaData#ownUpdatesAreVisible(int) */
public boolean ownUpdatesAreVisible(int type) {
return false;
}
/** @see java.sql.DatabaseMetaData#storesLowerCaseIdentifiers() */
public boolean storesLowerCaseIdentifiers() {
return false;
}
/** @see java.sql.DatabaseMetaData#storesLowerCaseQuotedIdentifiers() */
public boolean storesLowerCaseQuotedIdentifiers() {
return false;
}
/** @see java.sql.DatabaseMetaData#storesMixedCaseIdentifiers() */
public boolean storesMixedCaseIdentifiers() {
return true;
}
/** @see java.sql.DatabaseMetaData#storesMixedCaseQuotedIdentifiers() */
public boolean storesMixedCaseQuotedIdentifiers() {
return false;
}
/** @see java.sql.DatabaseMetaData#storesUpperCaseIdentifiers() */
public boolean storesUpperCaseIdentifiers() {
return false;
}
/** @see java.sql.DatabaseMetaData#storesUpperCaseQuotedIdentifiers() */
public boolean storesUpperCaseQuotedIdentifiers() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsAlterTableWithAddColumn() */
public boolean supportsAlterTableWithAddColumn() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsAlterTableWithDropColumn() */
public boolean supportsAlterTableWithDropColumn() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsANSI92EntryLevelSQL() */
public boolean supportsANSI92EntryLevelSQL() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsANSI92FullSQL() */
public boolean supportsANSI92FullSQL() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsANSI92IntermediateSQL() */
public boolean supportsANSI92IntermediateSQL() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsBatchUpdates() */
public boolean supportsBatchUpdates() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsCatalogsInDataManipulation() */
public boolean supportsCatalogsInDataManipulation() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsCatalogsInIndexDefinitions() */
public boolean supportsCatalogsInIndexDefinitions() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsCatalogsInPrivilegeDefinitions() */
public boolean supportsCatalogsInPrivilegeDefinitions() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsCatalogsInProcedureCalls() */
public boolean supportsCatalogsInProcedureCalls() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsCatalogsInTableDefinitions() */
public boolean supportsCatalogsInTableDefinitions() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsColumnAliasing() */
public boolean supportsColumnAliasing() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsConvert() */
public boolean supportsConvert() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsConvert(int, int) */
public boolean supportsConvert(int fromType, int toType) {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsCorrelatedSubqueries() */
public boolean supportsCorrelatedSubqueries() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsDataDefinitionAndDataManipulationTransactions() */
public boolean supportsDataDefinitionAndDataManipulationTransactions() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsDataManipulationTransactionsOnly() */
public boolean supportsDataManipulationTransactionsOnly() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsDifferentTableCorrelationNames() */
public boolean supportsDifferentTableCorrelationNames() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsExpressionsInOrderBy() */
public boolean supportsExpressionsInOrderBy() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsMinimumSQLGrammar() */
public boolean supportsMinimumSQLGrammar() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsCoreSQLGrammar() */
public boolean supportsCoreSQLGrammar() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsExtendedSQLGrammar() */
public boolean supportsExtendedSQLGrammar() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsLimitedOuterJoins() */
public boolean supportsLimitedOuterJoins() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsFullOuterJoins() */
public boolean supportsFullOuterJoins() throws SQLException {
String[] version = conn.libversion().split("\\.");
return Integer.parseInt(version[0]) >= 3 && Integer.parseInt(version[1]) >= 39;
}
/** @see java.sql.DatabaseMetaData#supportsGetGeneratedKeys() */
public boolean supportsGetGeneratedKeys() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsGroupBy() */
public boolean supportsGroupBy() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsGroupByBeyondSelect() */
public boolean supportsGroupByBeyondSelect() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsGroupByUnrelated() */
public boolean supportsGroupByUnrelated() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsIntegrityEnhancementFacility() */
public boolean supportsIntegrityEnhancementFacility() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsLikeEscapeClause() */
public boolean supportsLikeEscapeClause() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsMixedCaseIdentifiers() */
public boolean supportsMixedCaseIdentifiers() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsMixedCaseQuotedIdentifiers() */
public boolean supportsMixedCaseQuotedIdentifiers() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsMultipleOpenResults() */
public boolean supportsMultipleOpenResults() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsMultipleResultSets() */
public boolean supportsMultipleResultSets() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsMultipleTransactions() */
public boolean supportsMultipleTransactions() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsNamedParameters() */
public boolean supportsNamedParameters() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsNonNullableColumns() */
public boolean supportsNonNullableColumns() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsOpenCursorsAcrossCommit() */
public boolean supportsOpenCursorsAcrossCommit() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsOpenCursorsAcrossRollback() */
public boolean supportsOpenCursorsAcrossRollback() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsOpenStatementsAcrossCommit() */
public boolean supportsOpenStatementsAcrossCommit() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsOpenStatementsAcrossRollback() */
public boolean supportsOpenStatementsAcrossRollback() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsOrderByUnrelated() */
public boolean supportsOrderByUnrelated() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsOuterJoins() */
public boolean supportsOuterJoins() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsPositionedDelete() */
public boolean supportsPositionedDelete() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsPositionedUpdate() */
public boolean supportsPositionedUpdate() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsResultSetConcurrency(int, int) */
public boolean supportsResultSetConcurrency(int t, int c) {
return t == ResultSet.TYPE_FORWARD_ONLY && c == ResultSet.CONCUR_READ_ONLY;
}
/** @see java.sql.DatabaseMetaData#supportsResultSetHoldability(int) */
public boolean supportsResultSetHoldability(int h) {
return h == ResultSet.CLOSE_CURSORS_AT_COMMIT;
}
/** @see java.sql.DatabaseMetaData#supportsResultSetType(int) */
public boolean supportsResultSetType(int t) {
return t == ResultSet.TYPE_FORWARD_ONLY;
}
/** @see java.sql.DatabaseMetaData#supportsSavepoints() */
public boolean supportsSavepoints() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsSchemasInDataManipulation() */
public boolean supportsSchemasInDataManipulation() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsSchemasInIndexDefinitions() */
public boolean supportsSchemasInIndexDefinitions() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsSchemasInPrivilegeDefinitions() */
public boolean supportsSchemasInPrivilegeDefinitions() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsSchemasInProcedureCalls() */
public boolean supportsSchemasInProcedureCalls() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsSchemasInTableDefinitions() */
public boolean supportsSchemasInTableDefinitions() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsSelectForUpdate() */
public boolean supportsSelectForUpdate() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsStatementPooling() */
public boolean supportsStatementPooling() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsStoredProcedures() */
public boolean supportsStoredProcedures() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsSubqueriesInComparisons() */
public boolean supportsSubqueriesInComparisons() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsSubqueriesInExists() */
public boolean supportsSubqueriesInExists() {
return true;
} // TODO: check
/** @see java.sql.DatabaseMetaData#supportsSubqueriesInIns() */
public boolean supportsSubqueriesInIns() {
return true;
} // TODO: check
/** @see java.sql.DatabaseMetaData#supportsSubqueriesInQuantifieds() */
public boolean supportsSubqueriesInQuantifieds() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsTableCorrelationNames() */
public boolean supportsTableCorrelationNames() {
return false;
}
/** @see java.sql.DatabaseMetaData#supportsTransactionIsolationLevel(int) */
public boolean supportsTransactionIsolationLevel(int level) {
return level == Connection.TRANSACTION_SERIALIZABLE;
}
/** @see java.sql.DatabaseMetaData#supportsTransactions() */
public boolean supportsTransactions() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsUnion() */
public boolean supportsUnion() {
return true;
}
/** @see java.sql.DatabaseMetaData#supportsUnionAll() */
public boolean supportsUnionAll() {
return true;
}
/** @see java.sql.DatabaseMetaData#updatesAreDetected(int) */
public boolean updatesAreDetected(int type) {
return false;
}
/** @see java.sql.DatabaseMetaData#usesLocalFilePerTable() */
public boolean usesLocalFilePerTable() {
return false;
}
/** @see java.sql.DatabaseMetaData#usesLocalFiles() */
public boolean usesLocalFiles() {
return true;
}
/** @see java.sql.DatabaseMetaData#isReadOnly() */
public boolean isReadOnly() throws SQLException {
return conn.isReadOnly();
}
/**
* @see java.sql.DatabaseMetaData#getAttributes(java.lang.String, java.lang.String,
* java.lang.String, java.lang.String)
*/
public ResultSet getAttributes(String c, String s, String t, String a) throws SQLException {
if (getAttributes == null) {
getAttributes =
conn.prepareStatement(
"select null as TYPE_CAT, null as TYPE_SCHEM, "
+ "null as TYPE_NAME, null as ATTR_NAME, null as DATA_TYPE, "
+ "null as ATTR_TYPE_NAME, null as ATTR_SIZE, null as DECIMAL_DIGITS, "
+ "null as NUM_PREC_RADIX, null as NULLABLE, null as REMARKS, null as ATTR_DEF, "
+ "null as SQL_DATA_TYPE, null as SQL_DATETIME_SUB, null as CHAR_OCTET_LENGTH, "
+ "null as ORDINAL_POSITION, null as IS_NULLABLE, null as SCOPE_CATALOG, "
+ "null as SCOPE_SCHEMA, null as SCOPE_TABLE, null as SOURCE_DATA_TYPE limit 0;");
}
return getAttributes.executeQuery();
}
/**
* @see java.sql.DatabaseMetaData#getBestRowIdentifier(java.lang.String, java.lang.String,
* java.lang.String, int, boolean)
*/
public ResultSet getBestRowIdentifier(String c, String s, String t, int scope, boolean n)
throws SQLException {
if (getBestRowIdentifier == null) {
getBestRowIdentifier =
conn.prepareStatement(
"select null as SCOPE, null as COLUMN_NAME, "
+ "null as DATA_TYPE, null as TYPE_NAME, null as COLUMN_SIZE, "
+ "null as BUFFER_LENGTH, null as DECIMAL_DIGITS, null as PSEUDO_COLUMN limit 0;");
}
return getBestRowIdentifier.executeQuery();
}
/**
* @see java.sql.DatabaseMetaData#getColumnPrivileges(java.lang.String, java.lang.String,
* java.lang.String, java.lang.String)
*/
public ResultSet getColumnPrivileges(String c, String s, String t, String colPat)
throws SQLException {
if (getColumnPrivileges == null) {
getColumnPrivileges =
conn.prepareStatement(
"select null as TABLE_CAT, null as TABLE_SCHEM, "
+ "null as TABLE_NAME, null as COLUMN_NAME, null as GRANTOR, null as GRANTEE, "
+ "null as PRIVILEGE, null as IS_GRANTABLE limit 0;");
}
return getColumnPrivileges.executeQuery();
}
// Column type patterns
protected static final Pattern TYPE_INTEGER = Pattern.compile(".*(INT|BOOL).*");
protected static final Pattern TYPE_VARCHAR = Pattern.compile(".*(CHAR|CLOB|TEXT|BLOB).*");
protected static final Pattern TYPE_FLOAT = Pattern.compile(".*(REAL|FLOA|DOUB|DEC|NUM).*");
/**
* @see java.sql.DatabaseMetaData#getColumns(java.lang.String, java.lang.String,
* java.lang.String, java.lang.String)
*/
public ResultSet getColumns(String c, String s, String tblNamePattern, String colNamePattern)
throws SQLException {
// get the list of tables matching the pattern (getTables)
// create a Matrix Cursor for each of the tables
// create a merge cursor from all the Matrix Cursors
// and return the columname and type from:
// "PRAGMA table_xinfo(tablename)"
// which returns data like this:
// sqlite> PRAGMA lastyear.table_info(gross_sales);
// cid|name|type|notnull|dflt_value|pk
// 0|year|INTEGER|0|'2006'|0
// 1|month|TEXT|0||0
// 2|monthlygross|REAL|0||0
// 3|sortcol|INTEGER|0||0
// sqlite>
// and then make the cursor have these columns
// TABLE_CAT String => table catalog (may be null)
// TABLE_SCHEM String => table schema (may be null)
// TABLE_NAME String => table name
// COLUMN_NAME String => column name
// DATA_TYPE int => SQL type from java.sql.Types
// TYPE_NAME String => Data source dependent type name, for a UDT the type name is
// fully qualified
// COLUMN_SIZE int => column size.
// BUFFER_LENGTH is not used.
// DECIMAL_DIGITS int => the number of fractional digits. Null is returned for data
// types where DECIMAL_DIGITS is not applicable.
// NUM_PREC_RADIX int => Radix (typically either 10 or 2)
// NULLABLE int => is NULL allowed.
// columnNoNulls - might not allow NULL values
// columnNullable - definitely allows NULL values
// columnNullableUnknown - nullability unknown
// REMARKS String => comment describing column (may be null)
// COLUMN_DEF String => default value for the column, which should be interpreted as
// a string when the value is enclosed in single quotes (may be null)
// SQL_DATA_TYPE int => unused
// SQL_DATETIME_SUB int => unused
// CHAR_OCTET_LENGTH int => for char types the maximum number of bytes in the column
// ORDINAL_POSITION int => index of column in table (starting at 1)
// IS_NULLABLE String => ISO rules are used to determine the nullability for a
// column.
// YES --- if the parameter can include NULLs
// NO --- if the parameter cannot include NULLs
// empty string --- if the nullability for the parameter is unknown
// SCOPE_CATALOG String => catalog of table that is the scope of a reference
// attribute
// (null if DATA_TYPE isn't REF)
// SCOPE_SCHEMA String => schema of table that is the scope of a reference attribute
// (null if the DATA_TYPE isn't REF)
// SCOPE_TABLE String => table name that this the scope of a reference attribute
// (null if the DATA_TYPE isn't REF)
// SOURCE_DATA_TYPE short => source type of a distinct type or user-generated Ref
// type, SQL type from java.sql.Types (null if DATA_TYPE isn't DISTINCT or user-generated
// REF)
// IS_AUTOINCREMENT String => Indicates whether this column is auto incremented
// YES --- if the column is auto incremented
// NO --- if the column is not auto incremented
// empty string --- if it cannot be determined whether the column is auto incremented
// parameter is unknown
// IS_GENERATEDCOLUMN String => Indicates whether this column is auto incremented
// YES --- if the column is generated
// NO --- if the column is not generated
// empty string --- if it cannot be determined whether the column is auto incremented
// parameter is unknown
checkOpen();
StringBuilder sql = new StringBuilder(700);
sql.append("select null as TABLE_CAT, null as TABLE_SCHEM, tblname as TABLE_NAME, ")
.append(
"cn as COLUMN_NAME, ct as DATA_TYPE, tn as TYPE_NAME, colSize as COLUMN_SIZE, ")
.append(
"2000000000 as BUFFER_LENGTH, colDecimalDigits as DECIMAL_DIGITS, 10 as NUM_PREC_RADIX, ")
.append("colnullable as NULLABLE, null as REMARKS, colDefault as COLUMN_DEF, ")
.append(
"0 as SQL_DATA_TYPE, 0 as SQL_DATETIME_SUB, 2000000000 as CHAR_OCTET_LENGTH, ")
.append(
"ordpos as ORDINAL_POSITION, (case colnullable when 0 then 'NO' when 1 then 'YES' else '' end)")
.append(" as IS_NULLABLE, null as SCOPE_CATALOG, null as SCOPE_SCHEMA, ")
.append("null as SCOPE_TABLE, null as SOURCE_DATA_TYPE, ")
.append(
"(case colautoincrement when 0 then 'NO' when 1 then 'YES' else '' end) as IS_AUTOINCREMENT, ")
.append(
"(case colgenerated when 0 then 'NO' when 1 then 'YES' else '' end) as IS_GENERATEDCOLUMN from (");
boolean colFound = false;
ResultSet rs = null;
try {
// Get all tables implied by the input
rs = getTables(c, s, tblNamePattern, null);
while (rs.next()) {
String tableName = rs.getString(3);
boolean isAutoIncrement;
Statement statColAutoinc = conn.createStatement();
ResultSet rsColAutoinc = null;
try {
statColAutoinc = conn.createStatement();
rsColAutoinc =
statColAutoinc.executeQuery(
"SELECT LIKE('%autoincrement%', LOWER(sql)) FROM sqlite_schema "
+ "WHERE LOWER(name) = LOWER('"
+ escape(tableName)
+ "') AND TYPE IN ('table', 'view')");
rsColAutoinc.next();
isAutoIncrement = rsColAutoinc.getInt(1) == 1;
} finally {
if (rsColAutoinc != null) {
try {
rsColAutoinc.close();
} catch (Exception e) {
LogHolder.logger.error(() -> "Could not close ResultSet", e);
}
}
if (statColAutoinc != null) {
try {
statColAutoinc.close();
} catch (Exception e) {
LogHolder.logger.error(() -> "Could not close statement", e);
}
}
}
// For each table, get the column info and build into overall SQL
String pragmaStatement = "PRAGMA table_xinfo('" + escape(tableName) + "')";
try (Statement colstat = conn.createStatement();
ResultSet rscol = colstat.executeQuery(pragmaStatement)) {
for (int i = 0; rscol.next(); i++) {
String colName = rscol.getString(2);
String colType = rscol.getString(3);
String colNotNull = rscol.getString(4);
String colDefault = rscol.getString(5);
boolean isPk = "1".equals(rscol.getString(6));
String colHidden = rscol.getString(7);
int colNullable = 2;
if (colNotNull != null) {
colNullable = colNotNull.equals("0") ? 1 : 0;
}
if (colFound) {
sql.append(" union all ");
}
colFound = true;
// default values
int iColumnSize = 2000000000;
int iDecimalDigits = 10;
/*
* improved column types
* ref https://www.sqlite.org/datatype3.html - 2.1 Determination Of Column Affinity
* plus some degree of artistic-license applied
*/
colType = colType == null ? "TEXT" : colType.toUpperCase();
int colAutoIncrement = 0;
if (isPk && isAutoIncrement) {
colAutoIncrement = 1;
}
int colJavaType;
// rule #1 + boolean
if (TYPE_INTEGER.matcher(colType).find()) {
colJavaType = Types.INTEGER;
// there are no decimal digits
iDecimalDigits = 0;
} else if (TYPE_VARCHAR.matcher(colType).find()) {
colJavaType = Types.VARCHAR;
// there are no decimal digits
iDecimalDigits = 0;
} else if (TYPE_FLOAT.matcher(colType).find()) {
colJavaType = Types.FLOAT;
} else {
// catch-all
colJavaType = Types.VARCHAR;
}
// try to find an (optional) length/dimension of the column
int iStartOfDimension = colType.indexOf('(');
if (iStartOfDimension > 0) {
// find end of dimension
int iEndOfDimension = colType.indexOf(')', iStartOfDimension);
if (iEndOfDimension > 0) {
String sInteger, sDecimal;
// check for two values (integer part, fraction) divided by
// comma
int iDimensionSeparator = colType.indexOf(',', iStartOfDimension);
if (iDimensionSeparator > 0) {
sInteger =
colType.substring(
iStartOfDimension + 1, iDimensionSeparator);
sDecimal =
colType.substring(
iDimensionSeparator + 1, iEndOfDimension);
}
// only a single dimension
else {
sInteger =
colType.substring(
iStartOfDimension + 1, iEndOfDimension);
sDecimal = null;
}
// try to parse the values
try {
int iInteger = Integer.parseUnsignedInt(sInteger);
// parse decimals?
if (sDecimal != null) {
iDecimalDigits = Integer.parseUnsignedInt(sDecimal);
// columns size equals sum of integer and decimal part
// of dimension
iColumnSize = iInteger + iDecimalDigits;
} else {
// no decimals
iDecimalDigits = 0;
// columns size equals dimension
iColumnSize = iInteger;
}
} catch (NumberFormatException ex) {
// just ignore invalid dimension formats here
}
}
// "TYPE_NAME" (colType) is without the length/ dimension
colType = colType.substring(0, iStartOfDimension).trim();
}
int colGenerated = 0;
if ("2".equals(colHidden) || "3".equals(colHidden)) {
colGenerated = 1;
}
sql.append("select ")
.append(i + 1)
.append(" as ordpos, ")
.append(colNullable)
.append(" as colnullable,")
.append(colJavaType)
.append(" as ct, ")
.append(iColumnSize)
.append(" as colSize, ")
.append(iDecimalDigits)
.append(" as colDecimalDigits, ")
.append("'")
.append(tableName)
.append("' as tblname, ")
.append("'")
.append(escape(colName))
.append("' as cn, ")
.append("'")
.append(escape(colType))
.append("' as tn, ")
.append(quote(colDefault == null ? null : escape(colDefault)))
.append(" as colDefault,")
.append(colAutoIncrement)
.append(" as colautoincrement,")
.append(colGenerated)
.append(" as colgenerated");
if (colNamePattern != null) {
sql.append(" where upper(cn) like upper('")
.append(escape(colNamePattern))
.append("') ESCAPE '")
.append(getSearchStringEscape())
.append("'");
}
}
}
}
} finally {
if (rs != null) {
try {
rs.close();
} catch (Exception e) {
LogHolder.logger.error(() -> "Could not close ResultSet", e);
}
}
}
if (colFound) {
sql.append(") order by TABLE_SCHEM, TABLE_NAME, ORDINAL_POSITION;");
} else {
sql.append(
"select null as ordpos, null as colnullable, null as ct, null as colsize, null as colDecimalDigits, null as tblname, null as cn, null as tn, null as colDefault, null as colautoincrement, null as colgenerated) limit 0;");
}
Statement stat = conn.createStatement();
return ((CoreStatement) stat).executeQuery(sql.toString(), true);
}
/**
* @see java.sql.DatabaseMetaData#getCrossReference(java.lang.String, java.lang.String,
* java.lang.String, java.lang.String, java.lang.String, java.lang.String)
*/
public ResultSet getCrossReference(
String pc, String ps, String pt, String fc, String fs, String ft) throws SQLException {
if (pt == null) {
return getExportedKeys(fc, fs, ft);
}
if (ft == null) {
return getImportedKeys(pc, ps, pt);
}
String query =
"select "
+ quote(pc)
+ " as PKTABLE_CAT, "
+ quote(ps)
+ " as PKTABLE_SCHEM, "
+ quote(pt)
+ " as PKTABLE_NAME, "
+ "'' as PKCOLUMN_NAME, "
+ quote(fc)
+ " as FKTABLE_CAT, "
+ quote(fs)
+ " as FKTABLE_SCHEM, "
+ quote(ft)
+ " as FKTABLE_NAME, "
+ "'' as FKCOLUMN_NAME, -1 as KEY_SEQ, 3 as UPDATE_RULE, 3 as DELETE_RULE, '' as FK_NAME, '' as PK_NAME, "
+ DatabaseMetaData.importedKeyInitiallyDeferred
+ " as DEFERRABILITY limit 0 ";
return ((CoreStatement) conn.createStatement()).executeQuery(query, true);
}
/** @see java.sql.DatabaseMetaData#getSchemas() */
public ResultSet getSchemas() throws SQLException {
if (getSchemas == null) {
getSchemas =
conn.prepareStatement(
"select null as TABLE_SCHEM, null as TABLE_CATALOG limit 0;");
}
return getSchemas.executeQuery();
}
/** @see java.sql.DatabaseMetaData#getCatalogs() */
public ResultSet getCatalogs() throws SQLException {
if (getCatalogs == null) {
getCatalogs = conn.prepareStatement("select null as TABLE_CAT limit 0;");
}
return getCatalogs.executeQuery();
}
/**
* @see java.sql.DatabaseMetaData#getPrimaryKeys(java.lang.String, java.lang.String,
* java.lang.String)
*/
public ResultSet getPrimaryKeys(String c, String s, String table) throws SQLException {
PrimaryKeyFinder pkFinder = new PrimaryKeyFinder(table);
String[] columns = pkFinder.getColumns();
Statement stat = conn.createStatement();
StringBuilder sql = new StringBuilder(512);
sql.append("select null as TABLE_CAT, null as TABLE_SCHEM, '")
.append(escape(table))
.append("' as TABLE_NAME, cn as COLUMN_NAME, ks as KEY_SEQ, pk as PK_NAME from (");
if (columns == null) {
sql.append("select null as cn, null as pk, 0 as ks) limit 0;");
return ((CoreStatement) stat).executeQuery(sql.toString(), true);
}
String pkName = pkFinder.getName();
if (pkName != null) {
pkName = "'" + pkName + "'";
}
for (int i = 0; i < columns.length; i++) {
if (i > 0) sql.append(" union ");
sql.append("select ")
.append(pkName)
.append(" as pk, '")
.append(escape(unquoteIdentifier(columns[i])))
.append("' as cn, ")
.append(i + 1)
.append(" as ks");
}
return ((CoreStatement) stat).executeQuery(sql.append(") order by cn;").toString(), true);
}
private static final Map RULE_MAP = new HashMap<>();
static {
RULE_MAP.put("NO ACTION", DatabaseMetaData.importedKeyNoAction);
RULE_MAP.put("CASCADE", DatabaseMetaData.importedKeyCascade);
RULE_MAP.put("RESTRICT", DatabaseMetaData.importedKeyRestrict);
RULE_MAP.put("SET NULL", DatabaseMetaData.importedKeySetNull);
RULE_MAP.put("SET DEFAULT", DatabaseMetaData.importedKeySetDefault);
}
/**
* @see java.sql.DatabaseMetaData#getExportedKeys(java.lang.String, java.lang.String,
* java.lang.String)
*/
public ResultSet getExportedKeys(String catalog, String schema, String table)
throws SQLException {
PrimaryKeyFinder pkFinder = new PrimaryKeyFinder(table);
String[] pkColumns = pkFinder.getColumns();
Statement stat = conn.createStatement();
catalog = (catalog != null) ? quote(catalog) : null;
schema = (schema != null) ? quote(schema) : null;
StringBuilder exportedKeysQuery = new StringBuilder(512);
String target = null;
int count = 0;
if (pkColumns != null) {
// retrieve table list
ArrayList tableList;
try (ResultSet rs =
stat.executeQuery("select name from sqlite_schema where type = 'table'")) {
tableList = new ArrayList<>();
while (rs.next()) {
String tblname = rs.getString(1);
tableList.add(tblname);
if (tblname.equalsIgnoreCase(table)) {
// get the correct case as in the database
// (not uppercase nor lowercase)
target = tblname;
}
}
}
// find imported keys for each table
for (String tbl : tableList) {
final ImportedKeyFinder impFkFinder = new ImportedKeyFinder(tbl);
List fkNames = impFkFinder.getFkList();
for (ForeignKey foreignKey : fkNames) {
String PKTabName = foreignKey.getPkTableName();
if (PKTabName == null || !PKTabName.equalsIgnoreCase(target)) {
continue;
}
for (int j = 0; j < foreignKey.getColumnMappingCount(); j++) {
int keySeq = j + 1;
String[] columnMapping = foreignKey.getColumnMapping(j);
String PKColName = columnMapping[1];
PKColName = (PKColName == null) ? "" : PKColName;
String FKColName = columnMapping[0];
FKColName = (FKColName == null) ? "" : FKColName;
boolean usePkName = false;
for (String pkColumn : pkColumns) {
if (pkColumn != null && pkColumn.equalsIgnoreCase(PKColName)) {
usePkName = true;
break;
}
}
String pkName =
(usePkName && pkFinder.getName() != null) ? pkFinder.getName() : "";
exportedKeysQuery
.append(count > 0 ? " union all select " : "select ")
.append(keySeq)
.append(" as ks, '")
.append(escape(tbl))
.append("' as fkt, '")
.append(escape(FKColName))
.append("' as fcn, '")
.append(escape(PKColName))
.append("' as pcn, '")
.append(escape(pkName))
.append("' as pkn, ")
.append(RULE_MAP.get(foreignKey.getOnUpdate()))
.append(" as ur, ")
.append(RULE_MAP.get(foreignKey.getOnDelete()))
.append(" as dr, ");
String fkName = foreignKey.getFkName();
if (fkName != null) {
exportedKeysQuery.append("'").append(escape(fkName)).append("' as fkn");
} else {
exportedKeysQuery.append("'' as fkn");
}
count++;
}
}
}
}
boolean hasImportedKey = (count > 0);
StringBuilder sql = new StringBuilder(512);
sql.append("select ")
.append(catalog)
.append(" as PKTABLE_CAT, ")
.append(schema)
.append(" as PKTABLE_SCHEM, ")
.append(quote(target))
.append(" as PKTABLE_NAME, ")
.append(hasImportedKey ? "pcn" : "''")
.append(" as PKCOLUMN_NAME, ")
.append(catalog)
.append(" as FKTABLE_CAT, ")
.append(schema)
.append(" as FKTABLE_SCHEM, ")
.append(hasImportedKey ? "fkt" : "''")
.append(" as FKTABLE_NAME, ")
.append(hasImportedKey ? "fcn" : "''")
.append(" as FKCOLUMN_NAME, ")
.append(hasImportedKey ? "ks" : "-1")
.append(" as KEY_SEQ, ")
.append(hasImportedKey ? "ur" : "3")
.append(" as UPDATE_RULE, ")
.append(hasImportedKey ? "dr" : "3")
.append(" as DELETE_RULE, ")
.append(hasImportedKey ? "fkn" : "''")
.append(" as FK_NAME, ")
.append(hasImportedKey ? "pkn" : "''")
.append(" as PK_NAME, ")
.append(DatabaseMetaData.importedKeyInitiallyDeferred) // FIXME: Check for pragma
// foreign_keys = true ?
.append(" as DEFERRABILITY ");
if (hasImportedKey) {
sql.append("from (")
.append(exportedKeysQuery)
.append(") ORDER BY FKTABLE_CAT, FKTABLE_SCHEM, FKTABLE_NAME, KEY_SEQ");
} else {
sql.append("limit 0");
}
return ((CoreStatement) stat).executeQuery(sql.toString(), true);
}
private StringBuilder appendDummyForeignKeyList(StringBuilder sql) {
sql.append("select -1 as ks, '' as ptn, '' as fcn, '' as pcn, ")
.append(DatabaseMetaData.importedKeyNoAction)
.append(" as ur, ")
.append(DatabaseMetaData.importedKeyNoAction)
.append(" as dr, ")
.append(" '' as fkn, ")
.append(" '' as pkn ")
.append(") limit 0;");
return sql;
}
/**
* @see java.sql.DatabaseMetaData#getImportedKeys(java.lang.String, java.lang.String,
* java.lang.String)
*/
public ResultSet getImportedKeys(String catalog, String schema, String table)
throws SQLException {
ResultSet rs;
Statement stat = conn.createStatement();
StringBuilder sql = new StringBuilder(700);
sql.append("select ")
.append(quote(catalog))
.append(" as PKTABLE_CAT, ")
.append(quote(schema))
.append(" as PKTABLE_SCHEM, ")
.append("ptn as PKTABLE_NAME, pcn as PKCOLUMN_NAME, ")
.append(quote(catalog))
.append(" as FKTABLE_CAT, ")
.append(quote(schema))
.append(" as FKTABLE_SCHEM, ")
.append(quote(table))
.append(" as FKTABLE_NAME, ")
.append(
"fcn as FKCOLUMN_NAME, ks as KEY_SEQ, ur as UPDATE_RULE, dr as DELETE_RULE, fkn as FK_NAME, pkn as PK_NAME, ")
.append(DatabaseMetaData.importedKeyInitiallyDeferred)
.append(" as DEFERRABILITY from (");
// Use a try catch block to avoid "query does not return ResultSet" error
try {
rs = stat.executeQuery("pragma foreign_key_list('" + escape(table) + "');");
} catch (SQLException e) {
sql = appendDummyForeignKeyList(sql);
return ((CoreStatement) stat).executeQuery(sql.toString(), true);
}
final ImportedKeyFinder impFkFinder = new ImportedKeyFinder(table);
List fkNames = impFkFinder.getFkList();
int i = 0;
for (; rs.next(); i++) {
int keySeq = rs.getInt(2) + 1;
int keyId = rs.getInt(1);
String PKTabName = rs.getString(3);
String FKColName = rs.getString(4);
String PKColName = rs.getString(5);
String pkName = null;
try {
PrimaryKeyFinder pkFinder = new PrimaryKeyFinder(PKTabName);
pkName = pkFinder.getName();
if (PKColName == null) {
PKColName = pkFinder.getColumns()[0];
}
} catch (SQLException ignored) {
}
String updateRule = rs.getString(6);
String deleteRule = rs.getString(7);
if (i > 0) {
sql.append(" union all ");
}
String fkName = null;
if (fkNames.size() > keyId) fkName = fkNames.get(keyId).getFkName();
sql.append("select ")
.append(keySeq)
.append(" as ks,")
.append("'")
.append(escape(PKTabName))
.append("' as ptn, '")
.append(escape(FKColName))
.append("' as fcn, '")
.append(escape(PKColName))
.append("' as pcn,")
.append("case '")
.append(escape(updateRule))
.append("'")
.append(" when 'NO ACTION' then ")
.append(DatabaseMetaData.importedKeyNoAction)
.append(" when 'CASCADE' then ")
.append(DatabaseMetaData.importedKeyCascade)
.append(" when 'RESTRICT' then ")
.append(DatabaseMetaData.importedKeyRestrict)
.append(" when 'SET NULL' then ")
.append(DatabaseMetaData.importedKeySetNull)
.append(" when 'SET DEFAULT' then ")
.append(DatabaseMetaData.importedKeySetDefault)
.append(" end as ur, ")
.append("case '")
.append(escape(deleteRule))
.append("'")
.append(" when 'NO ACTION' then ")
.append(DatabaseMetaData.importedKeyNoAction)
.append(" when 'CASCADE' then ")
.append(DatabaseMetaData.importedKeyCascade)
.append(" when 'RESTRICT' then ")
.append(DatabaseMetaData.importedKeyRestrict)
.append(" when 'SET NULL' then ")
.append(DatabaseMetaData.importedKeySetNull)
.append(" when 'SET DEFAULT' then ")
.append(DatabaseMetaData.importedKeySetDefault)
.append(" end as dr, ")
.append(fkName == null ? "''" : quote(fkName))
.append(" as fkn, ")
.append(pkName == null ? "''" : quote(pkName))
.append(" as pkn");
}
rs.close();
if (i == 0) {
sql = appendDummyForeignKeyList(sql);
} else {
sql.append(") ORDER BY PKTABLE_CAT, PKTABLE_SCHEM, PKTABLE_NAME, KEY_SEQ;");
}
return ((CoreStatement) stat).executeQuery(sql.toString(), true);
}
/**
* @see java.sql.DatabaseMetaData#getIndexInfo(java.lang.String, java.lang.String,
* java.lang.String, boolean, boolean)
*/
public ResultSet getIndexInfo(String c, String s, String table, boolean u, boolean approximate)
throws SQLException {
ResultSet rs;
Statement stat = conn.createStatement();
StringBuilder sql = new StringBuilder(500);
// define the column header
// this is from the JDBC spec, it is part of the driver protocol
sql.append("select null as TABLE_CAT, null as TABLE_SCHEM, '")
.append(escape(table))
.append(
"' as TABLE_NAME, un as NON_UNIQUE, null as INDEX_QUALIFIER, n as INDEX_NAME, ")
.append(Integer.toString(DatabaseMetaData.tableIndexOther))
.append(" as TYPE, op as ORDINAL_POSITION, ")
.append(
"cn as COLUMN_NAME, null as ASC_OR_DESC, 0 as CARDINALITY, 0 as PAGES, null as FILTER_CONDITION from (");
// this always returns a result set now, previously threw exception
rs = stat.executeQuery("pragma index_list('" + escape(table) + "');");
ArrayList> indexList = new ArrayList<>();
while (rs.next()) {
indexList.add(new ArrayList<>());
indexList.get(indexList.size() - 1).add(rs.getString(2));
indexList.get(indexList.size() - 1).add(rs.getInt(3));
}
rs.close();
if (indexList.size() == 0) {
// if pragma index_list() returns no information, use this null block
sql.append("select null as un, null as n, null as op, null as cn) limit 0;");
return ((CoreStatement) stat).executeQuery(sql.toString(), true);
} else {
// loop over results from pragma call, getting specific info for each index
Iterator> indexIterator = indexList.iterator();
ArrayList
© 2015 - 2025 Weber Informatics LLC | Privacy Policy