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

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

There is a newer version: 6.0.0-beta-1
Show newest version
/*
 * Firebird Open Source JavaEE Connector - 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.ng.FbExceptionBuilder;
import org.firebirdsql.jdbc.parser.StatementParser;
import org.firebirdsql.logging.Logger;
import org.firebirdsql.logging.LoggerFactory;

import java.sql.SQLException;
import java.sql.Statement;
import java.util.Collections;
import java.util.EnumSet;
import java.util.Locale;
import java.util.Set;

/**
 * Factory to create appropriate {@code GeneratedKeysSupport} based on the configuration, Firebird version and parser
 * availability.
 *
 * @author Mark Rotteveel
 * @since 4.0
 */
final class GeneratedKeysSupportFactory {

    static final String GENERATED_KEYS_FUNCTIONALITY_NOT_AVAILABLE =
            "Generated keys functionality not available, most likely cause: ";
    static final String REASON_NO_RETURNING_SUPPORT =
            "This version of Firebird does not support retrieving generated keys (support was added in Firebird 2.0)";
    static final String REASON_EXPLICITLY_DISABLED = "disabled through connection property";
    private static final String REASON_NO_ANTLR = "antlr-runtime not available on classpath";
    private static final String GENERATED_KEYS_ENABLED_DEFAULT = "default";
    private static final String GENERATED_KEYS_DISABLED = "disabled";
    private static final String GENERATED_KEYS_IGNORED = "ignored";

    private GeneratedKeysSupportFactory() {
        // No instances
    }

    /**
     * Create generated keys support based on the provided config and database metadata.
     * 

* The supported values for {@code generatedKeysEnabled} are *

*
{@code default}
*
Support all query types supported by Firebird version (if none are supported, will throw * {@code SQLFeatureNotSupportedException} for attempts to use generated keys facility, except * {@code Statement.RETURN_NO_GENERATED_KEYS})
*
{@code disabled}
*
Explicitly disabled support, will throw {@code SQLFeatureNotSupportedException} for attempts to use * generated keys facility, except {@code Statement.RETURN_NO_GENERATED_KEYS})
*
{@code ignored}
*
Ignore generated keys support. Any attempt to use generated keys will be ignored and executed as a * normal non-generated keys statement.
*
Comma-separated list of query types
*
Only enable support for the specified query types. All other query types are handled as non-generated * keys statement
*
*

*

* The last option is a comma-separated list of query types to enable for support. Primary use-case is to enable * only {@code insert} to avoid the singleton limitation for the other DML types. Other example * {@code insert,update} enables only insert and update. Supported query type values (case insensitive) are: *

    *
  • {@code insert}
  • *
  • {@code update}
  • *
  • {@code delete}
  • *
  • {@code update_or_insert}
  • *
  • {@code merge}
  • *
*

*

* Supplying a list of query types will only enable the query types that are specified in the list (if supported by * the connected Firebird version). Unknown query types are ignored. If all specified query types are unknown or not * supported, then this behaves as {@code ignored}. NOTE: For Firebird 1.5 and earlier (unsupported) this will - * same as for {@code default} - behave as {@code disabled} and attempts to use generated keys (except * {@code Statement.RETURN_NO_GENERATED_KEYS}) will always throw a {@code SQLFeatureNotSupportedException}. *

* * @param generatedKeysEnabled * Generated keys enabled value (case insensitive: {@code disabled}, {@code ignored}, * {@code default} / {@code null} / empty string, or a list of query types to enable * @param fbDatabaseMetaData * Database metadata object * @return Appropriate generated keys support determined by config value, Firebird feature support and availability * of ANTLR on the the classpath. * @throws SQLException * if a database access error occurs while determining feature support */ static GeneratedKeysSupport createFor(String generatedKeysEnabled, FirebirdDatabaseMetaData fbDatabaseMetaData) throws SQLException { String normalizedConfigValue = generatedKeysEnabled != null && !generatedKeysEnabled.isEmpty() ? generatedKeysEnabled.toLowerCase(Locale.ROOT) : GENERATED_KEYS_ENABLED_DEFAULT; if (GENERATED_KEYS_DISABLED.equals(normalizedConfigValue)) { return DisabledGeneratedKeysSupport.EXPLICITLY_DISABLED; } else if (GENERATED_KEYS_IGNORED.equals(normalizedConfigValue)) { return IgnoredGeneratedKeysSupport.INSTANCE; } else if (!isGeneratedKeysSupportLoaded()) { return DisabledGeneratedKeysSupport.PARSER_NOT_LOADED; } else { Set returningSupport = GeneratedKeysSupport.QueryType.returningSupportForVersion( fbDatabaseMetaData.getDatabaseMajorVersion(), fbDatabaseMetaData.getDatabaseMinorVersion()); if (returningSupport.isEmpty()) { return new DisabledGeneratedKeysSupport(REASON_NO_RETURNING_SUPPORT); } else if (GENERATED_KEYS_ENABLED_DEFAULT.equals(normalizedConfigValue)) { return new DefaultGeneratedKeysSupport(ParserHolder.PARSER, fbDatabaseMetaData, returningSupport); } else { Set enabledTypes = getEnabledTypes(normalizedConfigValue, returningSupport); if (enabledTypes.isEmpty()) { return IgnoredGeneratedKeysSupport.INSTANCE; } return new DefaultGeneratedKeysSupport(ParserHolder.PARSER, fbDatabaseMetaData, enabledTypes); } } } private static Set getEnabledTypes(String normalizedConfigValue, Set returningSupport) { String[] queryTypeNames = normalizedConfigValue.split(","); EnumSet enabledTypes = EnumSet.noneOf(GeneratedKeysSupport.QueryType.class); // Given the limited number of values of queryType, double iteration is acceptable for (String queryTypeName : queryTypeNames) { for (GeneratedKeysSupport.QueryType queryType : returningSupport) { if (queryType.matches(queryTypeName.trim())) { enabledTypes.add(queryType); } } } return enabledTypes; } /** * Indicates if generated keys support has been loaded and available for use. *

* This method returns {@code false} when the antlr-runtime is not on the classpath or the {@link StatementParser} * implementation could not be loaded for other reasons. *

* * @return {@code true} if generated keys can be used in the driver (assuming the Firebird version supports it) */ static boolean isGeneratedKeysSupportLoaded() { return ParserHolder.PARSER != null; } /** * The default generated keys support, either based on the capabilities of Firebird or on a reduced set based on * configuration. * * @since 4.0 */ private static final class DefaultGeneratedKeysSupport implements GeneratedKeysSupport { private final StatementParser parser; private final FirebirdDatabaseMetaData fbDatabaseMetaData; private final Set supportedQueryTypes; private DefaultGeneratedKeysSupport(StatementParser parser, FirebirdDatabaseMetaData fbDatabaseMetaData, Set supportedQueryTypes) { assert !supportedQueryTypes.contains(QueryType.UNSUPPORTED) : "supportedQueryTypes should not contain UNSUPPORTED"; assert !supportedQueryTypes.isEmpty() : "At least one query type should be present"; this.parser = parser; this.fbDatabaseMetaData = fbDatabaseMetaData; this.supportedQueryTypes = supportedQueryTypes; } @Override public Query buildQuery(String sql, int autoGeneratedKeys) throws SQLException { switch (autoGeneratedKeys) { case Statement.NO_GENERATED_KEYS: return GeneratedKeysQueryBuilder .create(parser, sql, supportedQueryTypes) .forNoGeneratedKeysOption(); case Statement.RETURN_GENERATED_KEYS: return GeneratedKeysQueryBuilder .create(parser, sql, supportedQueryTypes) .forReturnGeneratedKeysOption(fbDatabaseMetaData); default: throw new FbExceptionBuilder() .nonTransientException(JaybirdErrorCodes.jb_invalidGeneratedKeysOption) .toFlatSQLException(); } } @Override public Query buildQuery(String sql, int[] columnIndexes) throws SQLException { return GeneratedKeysQueryBuilder .create(parser, sql, supportedQueryTypes) .forColumnsByIndex(columnIndexes, fbDatabaseMetaData); } @Override public Query buildQuery(String sql, String[] columnNames) throws SQLException { return GeneratedKeysQueryBuilder .create(parser, sql, supportedQueryTypes) .forColumnsByName(columnNames); } @Override public Set supportedQueryTypes() { return Collections.unmodifiableSet(supportedQueryTypes); } @Override public boolean supportsGetGeneratedKeys() { return true; } } /** * Disabled generated keys support. *

* Throws a {@code SQLFeatureNotSupportedException} with a configured reason for most options except * using {@code autoGeneratedKeys} with value {@link Statement#NO_GENERATED_KEYS}. *

* * @since 4.0 */ private static final class DisabledGeneratedKeysSupport implements GeneratedKeysSupport { private static final DisabledGeneratedKeysSupport EXPLICITLY_DISABLED = new DisabledGeneratedKeysSupport(REASON_EXPLICITLY_DISABLED); private static final DisabledGeneratedKeysSupport PARSER_NOT_LOADED = new DisabledGeneratedKeysSupport(REASON_NO_ANTLR); private final String reasonDisabled; public DisabledGeneratedKeysSupport(String reasonDisabled) { this.reasonDisabled = reasonDisabled; } @Override public Query buildQuery(String sql, int autoGeneratedKeys) throws SQLException { switch (autoGeneratedKeys) { case Statement.NO_GENERATED_KEYS: return new Query(false, sql); case Statement.RETURN_GENERATED_KEYS: throw disabled(); default: throw new FbExceptionBuilder() .nonTransientException(JaybirdErrorCodes.jb_invalidGeneratedKeysOption) .toFlatSQLException(); } } @Override public Query buildQuery(String sql, int[] columnIndexes) throws SQLException { throw disabled(); } @Override public Query buildQuery(String sql, String[] columnNames) throws SQLException { throw disabled(); } @Override public Set supportedQueryTypes() { return Collections.emptySet(); } @Override public boolean supportsGetGeneratedKeys() { return false; } private SQLException disabled() { return FbExceptionBuilder.forException(JaybirdErrorCodes.jb_generatedKeysSupportNotAvailable) .messageParameter(reasonDisabled) .toFlatSQLException(); } } /** * Ignores generated keys support and processes queries without producing a generated keys result set. *

* This ignores requests for generated keys handling and acts as if no query types support generated keys at all. * This allows executing all queries normally without support for {@code getGeneratedKeys}. This option will * yield errors when executing statements with a {@code RETURNING} clause with {@code executeUpdate} or * {@code executeLargeUpdate}. *

* * @since 4.0 */ private static final class IgnoredGeneratedKeysSupport implements GeneratedKeysSupport { private static final IgnoredGeneratedKeysSupport INSTANCE = new IgnoredGeneratedKeysSupport(); @Override public Query buildQuery(String sql, int autoGeneratedKeys) { // NOTE: Violates JDBC spec: allow values other than NO_GENERATED_KEYS and RETURN_GENERATED_KEYS for ignored return createNonGeneratedKeysQuery(sql); } @Override public Query buildQuery(String sql, int[] columnIndexes) { return createNonGeneratedKeysQuery(sql); } @Override public Query buildQuery(String sql, String[] columnNames) { return createNonGeneratedKeysQuery(sql); } @Override public Set supportedQueryTypes() { return Collections.emptySet(); } @Override public boolean supportsGetGeneratedKeys() { return false; } private Query createNonGeneratedKeysQuery(String sql) { return new Query(false, sql); } } /** * Initialization-on-demand depending on classloading behavior specified in JLS 12.4 */ private static final class ParserHolder { private static final StatementParser PARSER; static { // Attempt to load statement parser StatementParser temp = null; try { temp = (StatementParser) Class.forName("org.firebirdsql.jdbc.parser.StatementParserImpl") .getDeclaredConstructor() .newInstance(); } catch (Throwable ex) { // Unable to load class of parser implementation, antlr4-runtime not in path Logger log = LoggerFactory.getLogger(GeneratedKeysSupport.class); String message = "Unable to load generated key parser. " + GENERATED_KEYS_FUNCTIONALITY_NOT_AVAILABLE + REASON_NO_ANTLR; log.error(message + ": " + ex + "; see debug level for stacktrace"); log.debug(message, ex); } finally { PARSER = temp; } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy