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

org.firebirdsql.jdbc.FBDatabaseMetaData Maven / Gradle / Ivy

There is a newer version: 6.0.0-beta-1
Show newest version
/*
 * Firebird Open Source JDBC Driver
 *
 * Distributable under LGPL license.
 * You may obtain a copy of the License at http://www.gnu.org/copyleft/lgpl.html
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * LGPL License for more details.
 *
 * This file was created by members of the firebird development team.
 * All individual contributions remain the Copyright (C) of those
 * individuals.  Contributors to this file are either listed here or
 * can be obtained from a source control history command.
 *
 * All rights reserved.
 */
package org.firebirdsql.jdbc;

import org.firebirdsql.gds.JaybirdErrorCodes;
import org.firebirdsql.gds.impl.GDSFactory;
import org.firebirdsql.gds.impl.GDSHelper;
import org.firebirdsql.gds.impl.GDSType;
import org.firebirdsql.gds.ng.FbExceptionBuilder;
import org.firebirdsql.gds.ng.LockCloseable;
import org.firebirdsql.gds.ng.OdsVersion;
import org.firebirdsql.jaybird.Version;
import org.firebirdsql.jdbc.InternalTransactionCoordinator.MetaDataTransactionCoordinator;
import org.firebirdsql.jdbc.escape.FBEscapedFunctionHelper;
import org.firebirdsql.jdbc.metadata.*;
import org.firebirdsql.util.FirebirdSupportInfo;
import org.firebirdsql.util.InternalApi;

import java.io.Serial;
import java.nio.charset.StandardCharsets;
import java.sql.*;
import java.util.*;

import static java.lang.System.Logger.Level.DEBUG;
import static java.lang.System.Logger.Level.WARNING;
import static java.util.Collections.emptyList;
import static org.firebirdsql.util.FirebirdSupportInfo.supportInfoFor;

/**
 * Comprehensive information about the database as a whole.
 * 

* This class is internal API of Jaybird. Future versions may radically change, move, or make inaccessible this type. * For the public API, refer to the {@link DatabaseMetaData} and {@link FirebirdDatabaseMetaData} interfaces. *

* * @author David Jencks * @author Mark Rotteveel */ @SuppressWarnings("RedundantThrows") @InternalApi public class FBDatabaseMetaData implements FirebirdDatabaseMetaData { private static final System.Logger log = System.getLogger(FBDatabaseMetaData.class.getName()); private final GDSHelper gdsHelper; private final FBConnection connection; private final FirebirdSupportInfo firebirdSupportInfo; private static final int STATEMENT_CACHE_SIZE = 12; private final Map statements = new LruPreparedStatementCache(STATEMENT_CACHE_SIZE); private final FirebirdVersionMetaData versionMetaData; private CatalogMetadataInfo catalogMetadataInfo; protected FBDatabaseMetaData(FBConnection c) throws SQLException { this.gdsHelper = c.getGDSHelper(); this.connection = c; firebirdSupportInfo = supportInfoFor(c); versionMetaData = FirebirdVersionMetaData.getVersionMetaDataFor(c); } @Override public void close() { try (LockCloseable ignored = connection.withLock()) { if (statements.isEmpty()) { return; } try { for (FBStatement stmt : statements.values()) { try { stmt.close(); } catch (Exception e) { log.log(WARNING, "error closing cached statements in DatabaseMetaData.close; see debug level for stacktrace"); log.log(DEBUG, "error closing cached statements in DatabaseMetaData.close", e); } } } finally { statements.clear(); } } } @Override public boolean allProceduresAreCallable() throws SQLException { //returns all procedures regardless of whether you have execute permission return false; } @Override public boolean allTablesAreSelectable() throws SQLException { //returns all tables matching criteria independent of access permissions. return false; } @Override public String getURL() throws SQLException { // TODO Think of a less complex way to obtain the url or just return null? GDSType gdsType = getGDSType(); return GDSFactory.getJdbcUrl(gdsType, gdsHelper.getConnectionProperties()); } private GDSType getGDSType() { return connection.mc.getManagedConnectionFactory().getGDSType(); } @Override public String getUserName() throws SQLException { return gdsHelper.getUserName(); } @Override public boolean isReadOnly() throws SQLException { return false;//could be true, not yetimplemented } @Override public boolean nullsAreSortedHigh() throws SQLException { // in Firebird 1.5.x NULLs are always sorted at the end // in Firebird 2.0.x NULLs are sorted low return false; } @Override public boolean nullsAreSortedLow() throws SQLException { // in Firebird 1.5.x NULLs are always sorted at the end // in Firebird 2.0.x NULLs are sorted low return gdsHelper.compareToVersion(2) >= 0; } @Override public boolean nullsAreSortedAtStart() throws SQLException { // in Firebird 1.5.x NULLs are always sorted at the end // in Firebird 2.0.x NULLs are sorted low return false; } @Override public boolean nullsAreSortedAtEnd() throws SQLException { // in Firebird 1.5.x NULLs are always sorted at the end // in Firebird 2.0.x NULLs are sorted low return gdsHelper.compareToVersion(2) < 0; } @Override public String getDatabaseProductName() throws SQLException { return gdsHelper.getDatabaseProductName(); } @Override public String getDatabaseProductVersion() throws SQLException { return gdsHelper.getDatabaseProductVersion(); } @Override public String getDriverName() throws SQLException { // Retain JCA in name for compatibility with tools that consult metadata and use this string return "Jaybird JCA/JDBC driver"; } @Override public String getDriverVersion() throws SQLException { return Version.JAYBIRD_SIMPLE_VERSION; } @Override public int getDriverMajorVersion() { return Version.JAYBIRD_MAJOR_VERSION; } @Override public int getDriverMinorVersion() { return Version.JAYBIRD_MINOR_VERSION; } @Override public boolean usesLocalFiles() throws SQLException { return false; } @Override public boolean usesLocalFilePerTable() throws SQLException { return false; } @Override public boolean supportsMixedCaseIdentifiers() throws SQLException { return false; } // TODO implement statement pooling on the server.. then in the driver @Override public boolean supportsStatementPooling() throws SQLException { return false; } @Override public boolean locatorsUpdateCopy() throws SQLException { // Firebird creates a new blob when making changes return true; } @Override public boolean storesUpperCaseIdentifiers() throws SQLException { return true; } @Override public boolean storesLowerCaseIdentifiers() throws SQLException { return false; } @Override public boolean storesMixedCaseIdentifiers() throws SQLException { return false; } @Override public boolean supportsMixedCaseQuotedIdentifiers() throws SQLException { return true; } @Override public boolean storesUpperCaseQuotedIdentifiers() throws SQLException { return false; } @Override public boolean storesLowerCaseQuotedIdentifiers() throws SQLException { return false; } @Override public boolean storesMixedCaseQuotedIdentifiers() throws SQLException { return false; } @Override public String getIdentifierQuoteString() throws SQLException { return getConnectionDialect() == 1 ? " " : "\""; } @Override public String getSQLKeywords() throws SQLException { return versionMetaData.getSqlKeywords(); } /** * {@inheritDoc} *

* NOTE: Some of the functions listed may only work on Firebird 2.1 or higher, or when equivalent UDFs * are installed. *

*/ @Override public String getNumericFunctions() throws SQLException { return collectionToCommaSeparatedList(FBEscapedFunctionHelper.getSupportedNumericFunctions()); } private static String collectionToCommaSeparatedList(Collection collection) { StringBuilder sb = new StringBuilder(); for (String item : collection) { sb.append(item); sb.append(','); } sb.setLength(sb.length() - 1); return sb.toString(); } /** * {@inheritDoc} *

* NOTE: Some of the functions listed may only work on Firebird 2.1 or higher, or when equivalent UDFs * are installed. *

*/ @Override public String getStringFunctions() throws SQLException { return collectionToCommaSeparatedList(FBEscapedFunctionHelper.getSupportedStringFunctions()); } /** * {@inheritDoc} *

* NOTE: Some of the functions listed may only work on Firebird 2.1 or higher, or when equivalent UDFs * are installed. *

*/ @Override public String getSystemFunctions() throws SQLException { return collectionToCommaSeparatedList(FBEscapedFunctionHelper.getSupportedSystemFunctions()); } /** * {@inheritDoc} *

* NOTE: Some of the functions listed may only work on Firebird 2.1 or higher, or when equivalent UDFs * are installed. *

*/ @Override public String getTimeDateFunctions() throws SQLException { return collectionToCommaSeparatedList(FBEscapedFunctionHelper.getSupportedTimeDateFunctions()); } @Override public String getSearchStringEscape() throws SQLException { return "\\"; } @Override public String getExtraNameCharacters() throws SQLException { return "$"; } @Override public boolean supportsAlterTableWithAddColumn() throws SQLException { return true; } @Override public boolean supportsAlterTableWithDropColumn() throws SQLException { return true; } @Override public boolean supportsColumnAliasing() throws SQLException { return true; } @Override public boolean nullPlusNonNullIsNull() throws SQLException { return true; } @Override public boolean supportsConvert() throws SQLException { return true; } /** * {@inheritDoc} *

* See also {@code org.firebirdsql.jdbc.escape.ConvertFunction} for caveats. *

*/ @Override public boolean supportsConvert(int fromType, int toType) throws SQLException { switch (fromType) { case JaybirdTypeCodes.DECFLOAT: if (!firebirdSupportInfo.supportsDecfloat()) { return false; } // Intentional fallthrough case Types.TINYINT: // Doesn't exist in Firebird; handled as if SMALLINT case Types.SMALLINT: case Types.INTEGER: case Types.BIGINT: case Types.FLOAT: case Types.REAL: case Types.DOUBLE: case Types.NUMERIC: case Types.DECIMAL: // Numerical values all convertible to the same types. switch (toType) { case Types.TINYINT: // Doesn't exist in Firebird; handled as if SMALLINT case Types.SMALLINT: case Types.INTEGER: case Types.BIGINT: case Types.FLOAT: case Types.REAL: case Types.DOUBLE: case Types.NUMERIC: case Types.DECIMAL: case Types.CHAR: case Types.VARCHAR: case Types.LONGVARCHAR: case Types.CLOB: case Types.NCHAR: case Types.LONGNVARCHAR: case Types.NVARCHAR: case Types.NCLOB: return true; // casting numerical values to binary types will result in ASCII bytes of string conversion, not to the // binary representation of the number (eg 1 will be converted to binary 0x31 (ASCII '1'), not 0x01) case Types.BINARY: case Types.VARBINARY: case Types.LONGVARBINARY: case Types.BLOB: return true; case JaybirdTypeCodes.DECFLOAT: return firebirdSupportInfo.supportsDecfloat(); default: return false; } case Types.CHAR: case Types.VARCHAR: case Types.LONGVARCHAR: case Types.CLOB: case Types.NCHAR: case Types.LONGNVARCHAR: case Types.NVARCHAR: case Types.NCLOB: case Types.BINARY: case Types.VARBINARY: case Types.LONGVARBINARY: case Types.BLOB: case Types.ROWID: // Internally rowid is not discernible from BINARY // String and binary values all convertible to the same types // Be aware though that casting of binary to non-string/binary will perform the same conversion as // if it is an ASCII string value. Eg the binary string value 0x31 cast to integer will be 1, not 49. switch (toType) { case Types.CHAR: case Types.VARCHAR: case Types.LONGVARCHAR: case Types.CLOB: case Types.NCHAR: case Types.LONGNVARCHAR: case Types.NVARCHAR: case Types.NCLOB: case Types.BINARY: case Types.VARBINARY: case Types.LONGVARBINARY: case Types.BLOB: return true; case Types.TINYINT: // Doesn't exist in Firebird; handled as if SMALLINT case Types.SMALLINT: case Types.INTEGER: case Types.BIGINT: case Types.FLOAT: case Types.REAL: case Types.DOUBLE: case Types.NUMERIC: case Types.DECIMAL: case Types.DATE: case Types.TIME: case Types.TIMESTAMP: return fromType != Types.ROWID; case JaybirdTypeCodes.DECFLOAT: return fromType != Types.ROWID && firebirdSupportInfo.supportsDecfloat(); case Types.BOOLEAN: return fromType != Types.ROWID && firebirdSupportInfo.supportsBoolean(); case Types.ROWID: // As size of rowid is context dependent, we can't cast to it using the convert escape return false; case Types.TIME_WITH_TIMEZONE: case Types.TIMESTAMP_WITH_TIMEZONE: return fromType != Types.ROWID && firebirdSupportInfo.supportsTimeZones(); default: return false; } case Types.DATE: switch(toType) { case Types.DATE: case Types.TIMESTAMP: return true; case Types.TIME: case Types.TIME_WITH_TIMEZONE: return false; case Types.TIMESTAMP_WITH_TIMEZONE: return firebirdSupportInfo.supportsTimeZones(); case Types.CHAR: case Types.VARCHAR: case Types.LONGVARCHAR: case Types.CLOB: case Types.NCHAR: case Types.LONGNVARCHAR: case Types.NVARCHAR: case Types.NCLOB: return true; // casting date/time values to binary types will result in ASCII bytes of string conversion case Types.BINARY: case Types.VARBINARY: case Types.LONGVARBINARY: case Types.BLOB: return true; default: return false; } case Types.TIME: switch(toType) { case Types.TIMESTAMP: case Types.TIME: return true; case Types.DATE: return false; case Types.TIME_WITH_TIMEZONE: case Types.TIMESTAMP_WITH_TIMEZONE: return firebirdSupportInfo.supportsTimeZones(); case Types.CHAR: case Types.VARCHAR: case Types.LONGVARCHAR: case Types.CLOB: case Types.NCHAR: case Types.LONGNVARCHAR: case Types.NVARCHAR: case Types.NCLOB: return true; // casting date/time values to binary types will result in ASCII bytes of string conversion case Types.BINARY: case Types.VARBINARY: case Types.LONGVARBINARY: case Types.BLOB: return true; default: return false; } case Types.TIMESTAMP: switch(toType) { case Types.TIMESTAMP: case Types.TIME: case Types.DATE: return true; case Types.TIME_WITH_TIMEZONE: case Types.TIMESTAMP_WITH_TIMEZONE: return firebirdSupportInfo.supportsTimeZones(); case Types.CHAR: case Types.VARCHAR: case Types.LONGVARCHAR: case Types.CLOB: case Types.NCHAR: case Types.LONGNVARCHAR: case Types.NVARCHAR: case Types.NCLOB: return true; // casting date/time values to binary types will result in ASCII bytes of string conversion case Types.BINARY: case Types.VARBINARY: case Types.LONGVARBINARY: case Types.BLOB: return true; default: return false; } case Types.NULL: // If a type can be cast to itself, then null can be cast to it as well return toType != Types.NULL && supportsConvert(toType, toType); case Types.BOOLEAN: if (firebirdSupportInfo.supportsBoolean()) { switch (toType) { case Types.BOOLEAN: case Types.CHAR: case Types.VARCHAR: case Types.LONGVARCHAR: case Types.CLOB: case Types.NCHAR: case Types.LONGNVARCHAR: case Types.NVARCHAR: case Types.NCLOB: return true; // casting boolean values to binary types will result in ASCII bytes of string conversion case Types.BINARY: case Types.VARBINARY: case Types.LONGVARBINARY: case Types.BLOB: return true; default: return false; } } return false; case Types.TIME_WITH_TIMEZONE: if (firebirdSupportInfo.supportsTimeZones()) { switch (toType) { case Types.TIME: case Types.TIMESTAMP: return true; case Types.DATE: return false; case Types.TIME_WITH_TIMEZONE: case Types.TIMESTAMP_WITH_TIMEZONE: return true; case Types.CHAR: case Types.VARCHAR: case Types.LONGVARCHAR: case Types.CLOB: case Types.NCHAR: case Types.LONGNVARCHAR: case Types.NVARCHAR: case Types.NCLOB: return true; // casting date/time values to binary types will result in ASCII bytes of string conversion case Types.BINARY: case Types.VARBINARY: case Types.LONGVARBINARY: case Types.BLOB: return true; default: return false; } } return false; case Types.TIMESTAMP_WITH_TIMEZONE: if (firebirdSupportInfo.supportsTimeZones()) { switch (toType) { case Types.TIME: case Types.TIMESTAMP: case Types.DATE: case Types.TIME_WITH_TIMEZONE: case Types.TIMESTAMP_WITH_TIMEZONE: return true; case Types.CHAR: case Types.VARCHAR: case Types.LONGVARCHAR: case Types.CLOB: case Types.NCHAR: case Types.LONGNVARCHAR: case Types.NVARCHAR: case Types.NCLOB: return true; // casting date/time values to binary types will result in ASCII bytes of string conversion case Types.BINARY: case Types.VARBINARY: case Types.LONGVARBINARY: case Types.BLOB: return true; default: return false; } } return false; case Types.ARRAY: // Arrays are not supported by Jaybird (and casting would be tricky anyway) return false; // Unsupported types case Types.BIT: case Types.OTHER: case Types.JAVA_OBJECT: case Types.DISTINCT: case Types.STRUCT: case Types.REF: case Types.DATALINK: case Types.SQLXML: case Types.REF_CURSOR: default: return false; } } @Override public boolean supportsTableCorrelationNames() throws SQLException { return true; } @Override public boolean supportsDifferentTableCorrelationNames() throws SQLException { return false; } @Override public boolean supportsExpressionsInOrderBy() throws SQLException { return firebirdSupportInfo.isVersionEqualOrAbove(1, 5); } @Override public boolean supportsOrderByUnrelated() throws SQLException { return true; } @Override public boolean supportsGroupBy() throws SQLException { return true; } @Override public boolean supportsGroupByUnrelated() throws SQLException { // TODO Verify return false; } @Override public boolean supportsGroupByBeyondSelect() throws SQLException { // TODO Verify return false; } @Override public boolean supportsLikeEscapeClause() throws SQLException { return true; } @Override public boolean supportsMultipleResultSets() throws SQLException { return false; } @Override public boolean supportsMultipleTransactions() throws SQLException { return true; } @Override public boolean supportsNonNullableColumns() throws SQLException { return true; } @Override public boolean supportsMinimumSQLGrammar() throws SQLException { return true; } @Override public boolean supportsCoreSQLGrammar() throws SQLException { return true; } @Override public boolean supportsExtendedSQLGrammar() throws SQLException { return true; } @Override public boolean supportsANSI92EntryLevelSQL() throws SQLException { return true; } @Override public boolean supportsANSI92IntermediateSQL() throws SQLException { // TODO Verify return false; } @Override public boolean supportsANSI92FullSQL() throws SQLException { // TODO Verify return false; } @Override public boolean supportsIntegrityEnhancementFacility() throws SQLException { return true; // rrokytskyy: yep, they call so foreign keys + cascade deletes } @Override public boolean supportsOuterJoins() throws SQLException { return true; } @Override public boolean supportsFullOuterJoins() throws SQLException { return true; } @Override public boolean supportsLimitedOuterJoins() throws SQLException { return true; } /** * {@inheritDoc} * * @return the vendor term, always {@code null} because schemas are not supported by database server (see JDBC CTS * for details). */ @Override public String getSchemaTerm() throws SQLException { return null; } @Override public String getProcedureTerm() throws SQLException { return "PROCEDURE"; } /** * {@inheritDoc} * * @return the vendor term for catalog, normally {@code null} because catalogs are not supported by database server * (see JDBC CTS for details), or {@code "PACKAGE"} when {@code useCatalogAsPackage = true} and packages are * supported. */ @Override public String getCatalogTerm() throws SQLException { return getCatalogMetadata().getCatalogTerm(); } /** * {@inheritDoc} * *

* Will report {@code true} when {@code useCatalogAsPackage = true} and packages are supported. *

*/ @Override public boolean isCatalogAtStart() throws SQLException { return getCatalogMetadata().isCatalogAtStart(); } /** * {@inheritDoc} * * @return the separator string, always {@code null} because catalogs are not supported by database server (see * JDBC CTS for details), or {@code "."} when {@code useCatalogAsPackage = true} and packages are supported. */ @Override public String getCatalogSeparator() throws SQLException { return getCatalogMetadata().getCatalogSeparator(); } @Override public boolean supportsSchemasInDataManipulation() throws SQLException { return false; } @Override public boolean supportsSchemasInProcedureCalls() throws SQLException { return false; } @Override public boolean supportsSchemasInTableDefinitions() throws SQLException { return false; } @Override public boolean supportsSchemasInIndexDefinitions() throws SQLException { return false; } @Override public boolean supportsSchemasInPrivilegeDefinitions() throws SQLException { return false; } /** * {@inheritDoc} * *

* Will report {@code true} when {@code useCatalogAsPackage = true} and packages are supported. *

*/ @Override public boolean supportsCatalogsInDataManipulation() throws SQLException { return getCatalogMetadata().supportsCatalogsInDataManipulation(); } /** * {@inheritDoc} * *

* Will report {@code true} when {@code useCatalogAsPackage = true} and packages are supported. *

*/ @Override public boolean supportsCatalogsInProcedureCalls() throws SQLException { return getCatalogMetadata().supportsCatalogsInProcedureCalls(); } @Override public boolean supportsCatalogsInTableDefinitions() throws SQLException { return getCatalogMetadata().supportsCatalogsInTableDefinitions(); } @Override public boolean supportsCatalogsInIndexDefinitions() throws SQLException { return getCatalogMetadata().supportsCatalogsInIndexDefinitions(); } @Override public boolean supportsCatalogsInPrivilegeDefinitions() throws SQLException { return getCatalogMetadata().supportsCatalogsInPrivilegeDefinitions(); } @Override public boolean supportsPositionedDelete() throws SQLException { return true; } @Override public boolean supportsPositionedUpdate() throws SQLException { return true; } @Override public boolean supportsSelectForUpdate() throws SQLException { return true; } @Override public boolean supportsStoredProcedures() throws SQLException { return true; } @Override public boolean supportsSubqueriesInComparisons() throws SQLException { return true; } @Override public boolean supportsSubqueriesInExists() throws SQLException { return true; } @Override public boolean supportsSubqueriesInIns() throws SQLException { return true; } @Override public boolean supportsSubqueriesInQuantifieds() throws SQLException { return true; } @Override public boolean supportsCorrelatedSubqueries() throws SQLException { return true; } @Override public boolean supportsUnion() throws SQLException { return true; } @Override public boolean supportsUnionAll() throws SQLException { return true; } @Override public boolean supportsOpenCursorsAcrossCommit() throws SQLException { return false;//only when commit retaining is executed I think } @Override public boolean supportsOpenCursorsAcrossRollback() throws SQLException { return false;//commit retaining only. } @Override public boolean supportsOpenStatementsAcrossCommit() throws SQLException { return true; } @Override public boolean supportsOpenStatementsAcrossRollback() throws SQLException { return true; } //---------------------------------------------------------------------- // The following group of methods exposes various limitations // based on the target database with the current driver. // Unless otherwise specified, a result of zero means there is no // limit, or the limit is not known. @Override public int getMaxBinaryLiteralLength() throws SQLException { return 0; // TODO 32764 Test (assumed on length/2 and max string literal length) } @Override public int getMaxCharLiteralLength() throws SQLException { return 32765; } /** * {@inheritDoc} * *

* NOTE: This method reports the standard maximum length, and does not take into account restrictions configured * through {@code MaxIdentifierByteLength} or {@code MaxIdentifierCharLength}. *

*/ @Override public int getMaxColumnNameLength() throws SQLException { return getMaxObjectNameLength(); } @Override public int getMaxObjectNameLength() { return versionMetaData.maxIdentifierLength(); } @Override public int getMaxColumnsInGroupBy() throws SQLException { return 0; //I don't know } @Override public int getMaxColumnsInIndex() throws SQLException { return 0; //I don't know } @Override public int getMaxColumnsInOrderBy() throws SQLException { return 0; //I don't know } @Override public int getMaxColumnsInSelect() throws SQLException { return 0; //I don't know } @Override public int getMaxColumnsInTable() throws SQLException { return 32767; // Depends on datatypes and sizes, at most 64 kbyte excluding blobs (but including blob ids) } @Override public int getMaxConnections() throws SQLException { return 0; //I don't know } @Override public int getMaxCursorNameLength() throws SQLException { return 31; } @Override public int getMaxIndexLength() throws SQLException { if (gdsHelper.compareToVersion(2) < 0) { return 252; // See http://www.firebirdsql.org/en/firebird-technical-specifications/ } else { return 0; // 1/4 of page size, maybe retrieve page size and use that? } } @Override public int getMaxSchemaNameLength() throws SQLException { return 0; //No schemas } /** * {@inheritDoc} * *

* NOTE: This method reports the standard maximum length, and does not take into account restrictions configured * through {@code MaxIdentifierByteLength} or {@code MaxIdentifierCharLength}. *

*/ @Override public int getMaxProcedureNameLength() throws SQLException { return getMaxObjectNameLength(); } /** * {@inheritDoc} * *

* Will normally report {@code 0}, but non-zero when {@code useCatalogAsPackage = true} and packages are supported. *

*/ @Override public int getMaxCatalogNameLength() throws SQLException { return getCatalogMetadata().getMaxCatalogNameLength(); } @Override public int getMaxRowSize() throws SQLException { if (gdsHelper.compareToVersion(1, 5) >= 0) return 65531; else return 0; } @Override public boolean doesMaxRowSizeIncludeBlobs() throws SQLException { return false; // Blob sizes are not included in rowsize } @Override public int getMaxStatementLength() throws SQLException { if (gdsHelper.compareToVersion(3) >= 0) { // 10 MB return 10 * 1024 * 1024; } else { // 64 KB return 64 * 1024; } } @Override public int getMaxStatements() throws SQLException { // Limited by max handles, but this includes other objects than statements return 0; } /** * {@inheritDoc} * *

* NOTE: This method reports the standard maximum length, and does not take into account restrictions configured * through {@code MaxIdentifierByteLength} or {@code MaxIdentifierCharLength}. *

*/ @Override public int getMaxTableNameLength() throws SQLException { return getMaxObjectNameLength(); } @Override public int getMaxTablesInSelect() throws SQLException { // TODO Check if there is a max return 0; } /** * {@inheritDoc} * *

* NOTE: This method reports the standard maximum length, and does not take into account restrictions configured * through {@code MaxIdentifierByteLength} or {@code MaxIdentifierCharLength}. *

*/ @Override public int getMaxUserNameLength() throws SQLException { return getMaxObjectNameLength(); } //---------------------------------------------------------------------- @Override public int getDefaultTransactionIsolation() throws SQLException { return Connection.TRANSACTION_READ_COMMITTED; } @Override public boolean supportsTransactions() throws SQLException { return true; } @Override public boolean supportsTransactionIsolationLevel(int level) throws SQLException { return switch (level) { case Connection.TRANSACTION_READ_COMMITTED, Connection.TRANSACTION_REPEATABLE_READ, Connection.TRANSACTION_SERIALIZABLE -> true; default -> false; }; } /** * {@inheritDoc} *

* Although Firebird supports both DML and DDL in transactions, it is not possible to use objects in the same * transaction that defines them. For example, it is not possible to insert into a table in the same transaction * that created it. *

*/ @Override public boolean supportsDataDefinitionAndDataManipulationTransactions() throws SQLException { return true; } @Override public boolean supportsDataManipulationTransactionsOnly() throws SQLException { return false; } /** * {@inheritDoc} *

* Read the note on {@link #supportsDataDefinitionAndDataManipulationTransactions()}. *

*/ @Override public boolean dataDefinitionCausesTransactionCommit() throws SQLException { return false; } /** * {@inheritDoc} *

* Read the note on {@link #supportsDataDefinitionAndDataManipulationTransactions()}. *

*/ @Override public boolean dataDefinitionIgnoredInTransactions() throws SQLException { return false; } /** * {@inheritDoc} * *

* By default, this method does not return procedures defined in packages. To also return procedures in packages, * set connection property {@code useCatalogAsPackage} to {@code true}. When enabled, this method has the following * differences in behaviour: *

*
    *
  • The {@code catalog} parameter will return normal and packaged procedures when {@code null}, only normal * (non-packaged) procedures when empty string ({@code ""}), and procedures from a specific package (exact * case-sensitive match!) for other non-{@code null} values
  • *
  • Column {@code PROCEDURE_CAT} for normal procedures is empty string ({@code ""}) instead of {@code null}, * for packaged procedures it is the package name
  • *
  • Column {@code SPECIFIC_NAME} for packaged procedures will report * {@code .} (normal procedures will report the same as column * {@code PROCEDURE_NAME}, the unquoted name)
  • *
*/ @Override public ResultSet getProcedures(String catalog, String schemaPattern, String procedureNamePattern) throws SQLException { return GetProcedures.create(getDbMetadataMediator()).getProcedures(catalog, procedureNamePattern); } /** * {@inheritDoc} * *

* By default, this method does not return columns of procedures defined in packages. To also return columns of * procedures in packages, set connection property {@code useCatalogAsPackage} to {@code true}. When enabled, this * method has the following differences in behaviour: *

*
    *
  • The {@code catalog} parameter will return normal and packaged procedures when {@code null}, only normal * (non-packaged) procedures when empty string ({@code ""}), and procedures from a specific package (exact * case-sensitive match!) for other non-{@code null} values
  • *
  • Column {@code PROCEDURE_CAT} for normal procedures is empty string ({@code ""}) instead of {@code null}, * for packaged procedures it is the package name
  • *
  • Column {@code SPECIFIC_NAME} for packaged procedures will report * {@code .} (normal procedures will report the same as column * {@code PROCEDURE_NAME}, the unquoted name)
  • *
*/ @Override public ResultSet getProcedureColumns(String catalog, String schemaPattern, String procedureNamePattern, String columnNamePattern) throws SQLException { return GetProcedureColumns.create(getDbMetadataMediator()) .getProcedureColumns(catalog, procedureNamePattern, columnNamePattern); } public static final String TABLE = "TABLE"; public static final String SYSTEM_TABLE = "SYSTEM TABLE"; public static final String VIEW = "VIEW"; public static final String GLOBAL_TEMPORARY = "GLOBAL TEMPORARY"; /** * {@inheritDoc} *

* Jaybird defines these additional columns: *

    *
  1. OWNER_NAME String => Owner of the table
  2. *
  3. JB_RELATION_ID Short => Value of {@code RDB$RELATIONS.RDB$RELATION_ID} of the table
  4. *
*

*/ @Override public ResultSet getTables(String catalog, String schemaPattern, String tableNamePattern, String[] types) throws SQLException { return createGetTablesInstance().getTables(tableNamePattern, types); } private GetTables createGetTablesInstance() { return GetTables.create(getDbMetadataMediator()); } @Override public ResultSet getSchemas() throws SQLException { return getSchemas(null, null); } /** * {@inheritDoc} *

* When {@code useCatalogAsPackage = true} and packages are supported, this method will return the package names in * column {@code TABLE_CAT}. *

> */ @Override public ResultSet getCatalogs() throws SQLException { return GetCatalogs.create(getDbMetadataMediator()).getCatalogs(); } @Override public ResultSet getTableTypes() throws SQLException { return createGetTablesInstance().getTableTypes(); } @Override public String[] getTableTypeNames() throws SQLException { return createGetTablesInstance().getTableTypeNames(); } /** * {@inheritDoc} * *

* Jaybird defines these additional columns: *

    *
  1. JB_IS_IDENTITY String => Indicates whether this column is an identity column (NOTE: Jaybird * specific column; retrieve by name!). * There is a subtle difference with the meaning of {@code IS_AUTOINCREMENT}. This column indicates if the column * is a true identity column. *
      *
    • YES --- if the column is an identity column
    • *
    • NO --- if the column is not an identity column
    • *
    *
  2. *
  3. JB_IDENTITY_TYPE String => Type of identity column (NOTE: Jaybird specific column; retrieve by * name!) *
      *
    • ALWAYS --- for a GENERATED ALWAYS AS IDENTITY column (not yet supported in Firebird 3!)
    • *
    • BY DEFAULT --- for a GENERATED BY DEFAULT AS IDENTITY column
    • *
    • null --- if the column is not an identity type (or the identity type is unknown)
    • *
    *
  4. *
*

*/ @Override public ResultSet getColumns(String catalog, String schemaPattern, String tableNamePattern, String columnNamePattern) throws SQLException { return GetColumns.create(getDbMetadataMediator()).getColumns(tableNamePattern, columnNamePattern); } /** * {@inheritDoc} * *

* Jaybird defines an additional column: *

    *
  1. JB_GRANTEE_TYPE String => Object type of {@code GRANTEE} (NOTE: Jaybird specific column; * retrieve by name!).
  2. *
*

*

* Privileges granted to the table as a whole are reported for each individual column. *

*

* NOTE: This implementation returns all privileges, not just applicable to the current user. It is * unclear if this complies with the JDBC requirements. This may change in the future to only return only privileges * applicable to the current user, user {@code PUBLIC} and — maybe — active roles. *

*/ @Override public ResultSet getColumnPrivileges(String catalog, String schema, String table, String columnNamePattern) throws SQLException { return GetColumnPrivileges.create(getDbMetadataMediator()).getColumnPrivileges(table, columnNamePattern); } /** * {@inheritDoc} * *

* Jaybird defines an additional column: *

    *
  1. JB_GRANTEE_TYPE String => Object type of {@code GRANTEE} (NOTE: Jaybird specific column; * retrieve by name!).
  2. *
*

*

* NOTE: This implementation returns all privileges, not just applicable to the current user. It is * unclear if this complies with the JDBC requirements. This may change in the future to only return only privileges * applicable to the current user, user {@code PUBLIC} and — maybe — active roles. *

*/ @Override public ResultSet getTablePrivileges(String catalog, String schemaPattern, String tableNamePattern) throws SQLException { return GetTablePrivileges.create(getDbMetadataMediator()).getTablePrivileges(tableNamePattern); } /** * {@inheritDoc} *

* Jaybird considers the primary key (scoped as {@code bestRowSession} as the best identifier for all scopes. * Pseudo column {@code RDB$DB_KEY} (scoped as {@code bestRowTransaction} is considered the second-best alternative * for scopes {@code bestRowTemporary} and {@code bestRowTransaction} if {@code table} has no primary key. *

*

* Jaybird currently considers {@code RDB$DB_KEY} to be {@link DatabaseMetaData#bestRowTransaction} even if the * dbkey_scope is set to 1 (session). This may change in the future. See also {@link #getRowIdLifetime()}. *

*/ @Override public ResultSet getBestRowIdentifier(String catalog, String schema, String table, int scope, boolean nullable) throws SQLException { return GetBestRowIdentifier.create(getDbMetadataMediator()) .getBestRowIdentifier(catalog, schema, table, scope, nullable); } /** * {@inheritDoc} *

* Jaybird considers both {@code RDB$DB_KEY} and {@code RDB$RECORD_VERSION} (Firebird 3 and higher) as version * columns. *

*

* Jaybird only returns pseudo-column as version columns, so 'last updated' columns updated by a trigger, * calculated columns, or other forms of change tracking are not reported by this method. *

*/ @Override public ResultSet getVersionColumns(String catalog, String schema, String table) throws SQLException { return GetVersionColumns.create(getDbMetadataMediator()) .getVersionColumns(catalog, schema, table); } /** * {@inheritDoc} *

* Jaybird defines these additional columns: *

    *
  1. JB_PK_INDEX_NAME String => Index backing the primary key
  2. *
*

*/ @Override public ResultSet getPrimaryKeys(String catalog, String schema, String table) throws SQLException { return GetPrimaryKeys.create(getDbMetadataMediator()).getPrimaryKeys(table); } /** * {@inheritDoc} *

* Jaybird defines these additional columns: *

    *
  1. JB_FK_INDEX_NAME String => Index backing the foreign key
  2. *
  3. JB_PK_INDEX_NAME String => Index backing the primary key
  4. *
*

*/ @Override public ResultSet getImportedKeys(String catalog, String schema, String table) throws SQLException { return GetImportedKeys.create(getDbMetadataMediator()).getImportedKeys(table); } /** * {@inheritDoc} *

* Jaybird defines these additional columns: *

    *
  1. JB_FK_INDEX_NAME String => Index backing the foreign key
  2. *
  3. JB_PK_INDEX_NAME String => Index backing the primary key
  4. *
*

*/ @Override public ResultSet getExportedKeys(String catalog, String schema, String table) throws SQLException { return GetExportedKeys.create(getDbMetadataMediator()).getExportedKeys(table); } /** * {@inheritDoc} *

* Jaybird defines these additional columns: *

    *
  1. JB_FK_INDEX_NAME String => Index backing the foreign key
  2. *
  3. JB_PK_INDEX_NAME String => Index backing the primary key
  4. *
*

*/ @Override public ResultSet getCrossReference( String primaryCatalog, String primarySchema, String primaryTable, String foreignCatalog, String foreignSchema, String foreignTable) throws SQLException { return GetCrossReference.create(getDbMetadataMediator()).getCrossReference(primaryTable, foreignTable); } @Override public ResultSet getTypeInfo() throws SQLException { return GetTypeInfo.create(getDbMetadataMediator()).getTypeInfo(); } /** * {@inheritDoc} *

* Implementation note: The value of {@code FILTER_CONDITION} is populated with the value of * {@code RDB$INDICES.RDB$CONDITION_SOURCE}, which includes the {@code WHERE} keyword and comments before * the {@code WHERE} keyword. This is an implementation detail which may change in the future. That is, Jaybird may * change in the future to only include the condition itself, not the {@code WHERE} keyword, and/or may remove some * or all comments. *

*/ @Override public ResultSet getIndexInfo(String catalog, String schema, String table, boolean unique, boolean approximate) throws SQLException { return GetIndexInfo.create(getDbMetadataMediator()).getIndexInfo(table, unique, approximate); } @Override public boolean supportsResultSetType(int type) throws SQLException { // TYPE_SCROLL_SENSITIVE is always downgraded to TYPE_SCROLL_INSENSITIVE, so we report false for it return switch (type) { case ResultSet.TYPE_FORWARD_ONLY, ResultSet.TYPE_SCROLL_INSENSITIVE -> true; default -> false; }; } @Override public boolean supportsResultSetConcurrency(int type, int concurrency) throws SQLException { return switch (type) { case ResultSet.TYPE_FORWARD_ONLY, ResultSet.TYPE_SCROLL_INSENSITIVE -> concurrency == ResultSet.CONCUR_READ_ONLY || concurrency == ResultSet.CONCUR_UPDATABLE; default -> false; }; } @Override public boolean ownUpdatesAreVisible(int type) throws SQLException { return ResultSet.TYPE_SCROLL_INSENSITIVE == type; } @Override public boolean ownDeletesAreVisible(int type) throws SQLException { return ResultSet.TYPE_SCROLL_INSENSITIVE == type; } @Override public boolean ownInsertsAreVisible(int type) throws SQLException { return ResultSet.TYPE_SCROLL_INSENSITIVE == type; } @Override public boolean othersUpdatesAreVisible(int type) throws SQLException { return false; } @Override public boolean othersDeletesAreVisible(int type) throws SQLException { return false; } @Override public boolean othersInsertsAreVisible(int type) throws SQLException { return false; } @Override public boolean updatesAreDetected(int type) throws SQLException { // Only updates through the result set return ResultSet.TYPE_SCROLL_INSENSITIVE == type; } @Override public boolean deletesAreDetected(int type) throws SQLException { // Only deletes through the result set return ResultSet.TYPE_SCROLL_INSENSITIVE == type; } @Override public boolean insertsAreDetected(int type) throws SQLException { // Only inserts through the result set return ResultSet.TYPE_SCROLL_INSENSITIVE == type; } @Override public boolean supportsBatchUpdates() throws SQLException { return true; } /** * {@inheritDoc} *

* UDTs are not supported by Firebird. This method will always return an empty ResultSet. *

*/ @Override public ResultSet getUDTs(String catalog, String schemaPattern, String typeNamePattern, int[] types) throws SQLException { return GetUDTs.create(getDbMetadataMediator()).getUDTs(); } @Override public Connection getConnection() throws SQLException { return connection; } /** * {@inheritDoc} *

* UDTs are not supported by Firebird. This method will always return an empty ResultSet. *

*/ @Override public ResultSet getAttributes(String catalog, String schemaPattern, String typeNamePattern, String attributeNamePattern) throws SQLException { return GetAttributes.create(getDbMetadataMediator()).getAttributes(); } @Override public boolean supportsSavepoints() throws SQLException { return firebirdSupportInfo.supportsSavepoint(); } @Override public boolean supportsNamedParameters() throws SQLException { return false; } @Override public boolean supportsMultipleOpenResults() throws SQLException { return false; } @Override public boolean supportsGetGeneratedKeys() throws SQLException { return firebirdSupportInfo.supportsInsertReturning() && connection.getGeneratedKeysSupport().supportsGetGeneratedKeys(); } /** * {@inheritDoc} *

* Supertypes are not supported by Firebird. This method will always return an empty ResultSet. *

*/ @Override public ResultSet getSuperTypes(String catalog, String schemaPattern, String tableNamePattern) throws SQLException { return GetSuperTypes.create(getDbMetadataMediator()).getSuperTypes(); } /** * {@inheritDoc} *

* Supertables are not supported by Firebird. This method will always return an empty ResultSet. *

*/ @Override public ResultSet getSuperTables(String catalog, String schemaPattern, String tableNamePattern) throws SQLException { return GetSuperTables.create(getDbMetadataMediator()).getSuperTables(); } @Override public boolean supportsResultSetHoldability(int holdability) throws SQLException { return holdability == ResultSet.CLOSE_CURSORS_AT_COMMIT || holdability == ResultSet.HOLD_CURSORS_OVER_COMMIT; } @Override public int getResultSetHoldability() throws SQLException { // TODO Retrieve default holdable connection property return ResultSet.CLOSE_CURSORS_AT_COMMIT; } @Override public int getDatabaseMajorVersion() throws SQLException { return gdsHelper.getDatabaseProductMajorVersion(); } @Override public int getDatabaseMinorVersion() throws SQLException { return gdsHelper.getDatabaseProductMinorVersion(); } @Override public int getOdsMajorVersion() throws SQLException { return gdsHelper.getCurrentDatabase().getOdsMajor(); } @Override public int getOdsMinorVersion() throws SQLException { return gdsHelper.getCurrentDatabase().getOdsMinor(); } @Override public int getDatabaseDialect() throws SQLException { return gdsHelper.getCurrentDatabase().getDatabaseDialect(); } @Override public int getConnectionDialect() throws SQLException { return gdsHelper.getCurrentDatabase().getConnectionDialect(); } /** * {@inheritDoc} *

* Firebird primarily uses SQL standard SQL states, but may occasionally use values from X/Open. *

*/ @Override public int getSQLStateType() throws SQLException { return DatabaseMetaData.sqlStateSQL; } @Override public boolean supportsStoredFunctionsUsingCallSyntax() throws SQLException { return false; } /** * {@inheritDoc} *

* The holdable result sets remain open, others are closed, but this happens before the statement is executed. *

*/ @Override public boolean autoCommitFailureClosesAllResultSets() throws SQLException { // the holdable result sets remain open, others are closed, but this // happens before the statement is executed return false; } @Override public ResultSet getClientInfoProperties() throws SQLException { return GetClientInfoProperties.create(getDbMetadataMediator()).getClientInfoProperties(); } /** * {@inheritDoc} * *

* By default, this method does not return columns of functions defined in packages. To also return columns of * functions in packages, set connection property {@code useCatalogAsPackage} to {@code true}. When enabled, this * method has the following differences in behaviour: *

*
    *
  • The {@code catalog} parameter will return normal and packaged functions when {@code null}, only normal * (non-packaged) function when empty string ({@code ""}), and functions from a specific package (exact * case-sensitive match!) for other non-{@code null} values
  • *
  • Column {@code FUNCTION_CAT} for normal functions is empty string ({@code ""}) instead of {@code null}, * for packaged functions it is the package name
  • *
  • Column {@code SPECIFIC_NAME} for packaged functions will report * {@code .} (normal functions will report the same as column * {@code FUNCTION_NAME}, the unquoted name)
  • *
*/ @Override public ResultSet getFunctionColumns(String catalog, String schemaPattern, String functionNamePattern, String columnNamePattern) throws SQLException { return GetFunctionColumns.create(getDbMetadataMediator()) .getFunctionColumns(catalog, functionNamePattern, columnNamePattern); } /** * {@inheritDoc} * *

* Jaybird defines a number of additional columns. As these are not defined in JDBC, their position may change with * revisions of JDBC. We recommend to retrieve these columns by name. The following additional columns are * available: *

    *
  1. JB_FUNCTION_SOURCE String => The source of the function (for Firebird 3+ PSQL functions only)).
  2. *
  3. JB_FUNCTION_KIND String => The kind of function, one of "UDF", "PSQL" (Firebird 3+) or * "UDR" (Firebird 3+)
  4. *
  5. JB_MODULE_NAME String => Value of {@code RDB$MODULE_NAME} (is {@code null} for PSQL)
  6. *
  7. JB_ENTRYPOINT String => Value of {@code RDB$ENTRYPOINT} (is {@code null} for PSQL)
  8. *
  9. JB_ENGINE_NAME String => Value of {@code RDB$ENGINE_NAME} (is {@code null} for UDF and PSQL)
  10. *
*

*

* By default, this method does not return functions defined in packages. To also return functions in packages, * set connection property {@code useCatalogAsPackage} to {@code true}. When enabled, this method has the following * differences in behaviour: *

*
    *
  • The {@code catalog} parameter will return normal and packaged functions when {@code null}, only normal * (non-packaged) function when empty string ({@code ""}), and functions from a specific package (exact * case-sensitive match!) for other non-{@code null} values
  • *
  • Column {@code FUNCTION_CAT} for normal functions is empty string ({@code ""}) instead of {@code null}, * for packaged functions it is the package name
  • *
  • Column {@code SPECIFIC_NAME} for packaged functions will report * {@code .} (normal functions will report the same as column * {@code FUNCTION_NAME}, the unquoted name)
  • *
*/ @Override public ResultSet getFunctions(String catalog, String schemaPattern, String functionNamePattern) throws SQLException { return GetFunctions.create(getDbMetadataMediator()).getFunctions(catalog, functionNamePattern); } @Override public ResultSet getSchemas(String catalog, String schemaPattern) throws SQLException { return GetSchemas.create(getDbMetadataMediator()).getSchemas(); } @Override public boolean isWrapperFor(Class iface) throws SQLException { return iface != null && iface.isAssignableFrom(FBDatabaseMetaData.class); } @Override public T unwrap(Class iface) throws SQLException { if (!isWrapperFor(iface)) { throw FbExceptionBuilder.forException(JaybirdErrorCodes.jb_unableToUnwrap) .messageParameter(iface != null ? iface.getName() : "(null)") .toSQLException(); } return iface.cast(this); } /** * Escapes the like wildcards and escape ({@code \_%} in the provided search string with a {@code \}. *

* Primary purpose is to escape object names with wildcards for use in metadata patterns for literal matches, but * it can also be used to escape for SQL {@code LIKE}. *

* * @param objectName * Object name to escape. * @return Object name with wildcards escaped. */ public static String escapeWildcards(String objectName) { return MetadataPattern.escapeWildcards(objectName); } @Override public ResultSet getPseudoColumns(String catalog, String schemaPattern, String tableNamePattern, String columnNamePattern) throws SQLException { return GetPseudoColumns.create(getDbMetadataMediator()).getPseudoColumns(tableNamePattern, columnNamePattern); } @Override public boolean generatedKeyAlwaysReturned() throws SQLException { // TODO Double check if this is correct return false; } @Override public String getProcedureSourceCode(String procedureName) throws SQLException { String sResult = null; String sql = "Select RDB$PROCEDURE_SOURCE From RDB$PROCEDURES Where " + "RDB$PROCEDURE_NAME = ?"; List params = new ArrayList<>(); params.add(procedureName); try (ResultSet rs = doQuery(sql, params)) { if (rs.next()) sResult = rs.getString(1); } return sResult; } @Override public String getTriggerSourceCode(String triggerName) throws SQLException { String sResult = null; String sql = "Select RDB$TRIGGER_SOURCE From RDB$TRIGGERS Where RDB$TRIGGER_NAME = ?"; List params = new ArrayList<>(); params.add(triggerName); try (ResultSet rs = doQuery(sql, params)) { if (rs.next()) sResult = rs.getString(1); } return sResult; } @Override public String getViewSourceCode(String viewName) throws SQLException { String sResult = null; String sql = "Select RDB$VIEW_SOURCE From RDB$RELATIONS Where RDB$RELATION_NAME = ?"; List params = new ArrayList<>(); params.add(viewName); try (ResultSet rs = doQuery(sql, params)) { if (rs.next()) sResult = rs.getString(1); } return sResult; } protected static byte[] getBytes(String value) { return value != null ? value.getBytes(StandardCharsets.UTF_8) : null; } private FBPreparedStatement getStatement(String sql, boolean standalone) throws SQLException { try (LockCloseable ignored = connection.withLock()) { if (!standalone) { // Check cache FBPreparedStatement cachedStatement = statements.get(sql); if (cachedStatement != null) { if (cachedStatement.isClosed()) { //noinspection resource statements.remove(sql); } else { return cachedStatement; } } } var metaDataTransactionCoordinator = new MetaDataTransactionCoordinator(connection.txCoordinator); var newStatement = new FBPreparedStatement(connection, sql, ResultSetBehavior.of( ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY, ResultSet.CLOSE_CURSORS_AT_COMMIT), metaDataTransactionCoordinator, metaDataTransactionCoordinator, true, standalone, false); if (!standalone) { statements.put(sql, newStatement); } return newStatement; } } /** * Execute an SQL query with a given set of parameters. * * @param sql * The sql statement to be used for the query * @param params * The parameters to be used in the query * @throws SQLException * if a database access error occurs */ protected ResultSet doQuery(String sql, List params) throws SQLException { return doQuery(sql, params, false); } /** * Execute an SQL query with a given set of parameters. * * @param sql * The sql statement to be used for the query * @param params * The parameters to be used in the query * @param standalone * The query to be executed is a standalone query (should not be cached and be closed asap) * @throws SQLException * if a database access error occurs */ protected ResultSet doQuery(String sql, List params, boolean standalone) throws SQLException { FBPreparedStatement s = getStatement(sql, standalone); for (int i = 0; i < params.size(); i++) { s.setString(i + 1, params.get(i)); } return s.executeMetaDataQuery(); } /** * {@inheritDoc} *

* Minimum lifetime supported by Firebird is transaction-scope, and this can be changed to session-scope with * {@code isc_dpb_dbkey_scope} set to {@code 1} (eg connection property {@code dbkey_scope=1}). This implementation, * however, will always report {@link RowIdLifetime#ROWID_VALID_TRANSACTION}. *

*/ @Override public RowIdLifetime getRowIdLifetime() throws SQLException { return RowIdLifetime.ROWID_VALID_TRANSACTION; } private CatalogMetadataInfo getCatalogMetadata() { CatalogMetadataInfo catalogMetadataInfo = this.catalogMetadataInfo; if (catalogMetadataInfo == null) { catalogMetadataInfo = this.catalogMetadataInfo = CatalogMetadataInfo.create(getDbMetadataMediator()); } return catalogMetadataInfo; } private static final int JDBC_MAJOR_VERSION = 4; private static final int JDBC_MINOR_VERSION = 3; @Override public int getJDBCMajorVersion() { return JDBC_MAJOR_VERSION; } @Override public int getJDBCMinorVersion() { return JDBC_MINOR_VERSION; } @SuppressWarnings("java:S2160") private static class LruPreparedStatementCache extends LinkedHashMap { @Serial private static final long serialVersionUID = -6600678461169652270L; private final int maxCapacity; private LruPreparedStatementCache(int maxCapacity) { super(16, 0.75f, true); this.maxCapacity = maxCapacity; } @Override protected boolean removeEldestEntry(Map.Entry eldest) { if (size() <= maxCapacity) { return false; } try { FBPreparedStatement statement = eldest.getValue(); statement.close(); } catch (Exception e) { log.log(DEBUG, "Closing eldest cached metadata statement yielded an exception; ignored", e); } return true; } } protected DbMetadataMediator getDbMetadataMediator() { return new DbMetadataMediatorImpl(); } private final class DbMetadataMediatorImpl extends DbMetadataMediator { @Override public FirebirdSupportInfo getFirebirdSupportInfo() { return firebirdSupportInfo; } @Override public ResultSet performMetaDataQuery(MetadataQuery metadataQuery) throws SQLException { return doQuery(metadataQuery.getQueryText(), metadataQuery.getParameters(), metadataQuery.isStandalone()); } @Override public FBDatabaseMetaData getMetaData() { return FBDatabaseMetaData.this; } @Override public GDSType getGDSType() { return FBDatabaseMetaData.this.getGDSType(); } @Override public boolean isUseCatalogAsPackage() { return gdsHelper.getConnectionProperties().isUseCatalogAsPackage() && firebirdSupportInfo.supportsPackages(); } @Override public Collection getClientInfoPropertyNames() { if (firebirdSupportInfo.supportsGetSetContext()) { try { return connection.getClientInfoProvider().getDefaultClientInfoPropertyNames(); } catch (SQLException e) { log.log(DEBUG, "ignored exception in getDefaultClientInfoPropertyNames, falling back to no names", e); } } return emptyList(); } @Override public OdsVersion getOdsVersion() { return gdsHelper.getCurrentDatabase().getOdsVersion(); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy