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

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

There is a newer version: 4.0.10.java8
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.GDSException;
import org.firebirdsql.gds.JaybirdErrorCodes;
import org.firebirdsql.gds.impl.GDSFactory;
import org.firebirdsql.gds.impl.GDSType;
import org.firebirdsql.gds.ng.FbExceptionBuilder;
import org.firebirdsql.jca.FBManagedConnectionFactory;
import org.firebirdsql.logging.Logger;
import org.firebirdsql.logging.LoggerFactory;

import javax.resource.ResourceException;
import java.io.UnsupportedEncodingException;
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.SoftReference;
import java.net.URLDecoder;
import java.sql.*;
import java.util.Map;
import java.util.Properties;
import java.util.StringTokenizer;
import java.util.concurrent.ConcurrentHashMap;

/**
 * The Jaybird JDBC Driver implementation for the Firebird database.
 *
 * @author David Jencks
 * @author Mark Rotteveel
 */
public class FBDriver implements FirebirdDriver {

    private final static Logger log;

    public static final String CHARSET = "charSet";
    public static final String USE_TRANSLATION = "useTranslation";
    public static final String USER = "user";
    public static final String USER_NAME = "user_name";
    public static final String PASSWORD = "password";
    public static final String DATABASE = "database";
    public static final String BLOB_BUFFER_LENGTH = "blob_buffer_length";
    public static final String TPB_MAPPING = "tpb_mapping";

    private static final String URL_CHARSET = "UTF-8";
    
    /*
     * @todo implement the default subject for the
     * standard connection.
     */

    private final Map> mcfToDataSourceMap =
            new ConcurrentHashMap<>();
    private final ReferenceQueue dataSourceReferenceQueue = new ReferenceQueue<>();
    private final Object createDataSourceLock = new Object();

    static {
        log = LoggerFactory.getLogger(FBDriver.class);
        try {
            DriverManager.registerDriver(new FBDriver());
        } catch (Exception ex) {
            log.error("Could not register with driver manager", ex);
        }
    }

    /**
     * Attempts to make a database connection to the given URL.
     * The driver should return "null" if it realizes it is the wrong kind
     * of driver to connect to the given URL.  This will be common, as when
     * the JDBC driver manager is asked to connect to a given URL it passes
     * the URL to each loaded driver in turn.
     * 

* The driver should raise a SQLException if it is the right * driver to connect to the given URL, but has trouble connecting to * the database. *

*

* The java.util.Properties argument can be used to passed arbitrary * string tag/value pairs as connection arguments. * Normally at least "user" and "password" properties should be * included in the Properties. *

* * @param url * the URL of the database to which to connect * @param info * a list of arbitrary string tag/value pairs as * connection arguments. Normally at least a "user" and * "password" property should be included. * @return a Connection object that represents a * connection to the URL * @throws SQLException * if a database access error occurs */ public Connection connect(String url, final Properties info) throws SQLException { if (url == null) { throw new SQLException("url is null"); } final GDSType type = GDSFactory.getTypeForProtocol(url); if (type == null) { return null; } final Properties mergedProperties = mergeProperties(url, info); final Map normalizedInfo = FBDriverPropertyManager.normalize(mergedProperties); try { int qMarkIndex = url.indexOf('?'); if (qMarkIndex != -1) { url = url.substring(0, qMarkIndex); } FBManagedConnectionFactory mcf = new FBManagedConnectionFactory(type); String databaseURL = GDSFactory.getDatabasePath(type, url); mcf.setDatabase(databaseURL); for (Map.Entry entry : normalizedInfo.entrySet()) { mcf.setNonStandardProperty(entry.getKey(), entry.getValue()); } FBTpbMapper.processMapping(mcf, mergedProperties); mcf = mcf.canonicalize(); FBDataSource dataSource = createDataSource(mcf); return dataSource.getConnection(mcf.getUserName(), mcf.getPassword()); } catch (ResourceException | GDSException resex) { throw new FBSQLException(resex); } } private FBDataSource createDataSource(final FBManagedConnectionFactory mcf) throws ResourceException { final FBConnectionProperties cacheKey = mcf.getCacheKey(); FBDataSource dataSource = dataSourceFromCache(cacheKey); if (dataSource != null) return dataSource; synchronized (createDataSourceLock) { // Obtain again dataSource = dataSourceFromCache(cacheKey); if (dataSource == null) { dataSource = (FBDataSource) mcf.createConnectionFactory(); mcfToDataSourceMap.put(cacheKey, new SoftReference<>(dataSource, dataSourceReferenceQueue)); } } cleanDataSourceCache(); return dataSource; } /** * Removes cleared references from the {@link #mcfToDataSourceMap} cache. */ private void cleanDataSourceCache() { Reference reference; while ((reference = dataSourceReferenceQueue.poll()) != null) { mcfToDataSourceMap.values().remove(reference); } } private FBDataSource dataSourceFromCache(final FBConnectionProperties cacheKey) { final SoftReference dataSourceReference = mcfToDataSourceMap.get(cacheKey); return dataSourceReference != null ? dataSourceReference.get() : null; } public FirebirdConnection connect(FirebirdConnectionProperties properties) throws SQLException { GDSType type = GDSType.getType(properties.getType()); if (type == null) type = GDSFactory.getDefaultGDSType(); try { FBManagedConnectionFactory mcf = new FBManagedConnectionFactory(type); mcf = mcf.canonicalize(); FBDataSource dataSource = createDataSource(mcf); return (FirebirdConnection) dataSource.getConnection(mcf.getUserName(), mcf.getPassword()); } catch (ResourceException ex) { throw new FBSQLException(ex); } } public FirebirdConnectionProperties newConnectionProperties() { return new FBConnectionProperties(); } /** * Returns true if the driver thinks that it can open a connection * to the given URL. Typically drivers will return true if they * understand the subprotocol specified in the URL and false if * they don't. * * @param url * the URL of the database * @return true if this driver can connect to the given URL * @throws SQLException * if a database access error occurs */ public boolean acceptsURL(String url) throws SQLException { if (url == null) { throw new SQLException("url is null"); } for (String protocol : GDSFactory.getSupportedProtocols()) { if (url.startsWith(protocol)) return true; } return false; } /** * Gets information about the possible properties for this driver. *

The getPropertyInfo method is intended to allow a generic GUI tool to * discover what properties it should prompt a human for in order to get * enough information to connect to a database. Note that depending on * the values the human has supplied so far, additional values may become * necessary, so it may be necessary to iterate though several calls * to getPropertyInfo. * * @param url * the URL of the database to which to connect * @param info * a proposed list of tag/value pairs that will be sent on * connect open * @return an array of DriverPropertyInfo objects describing possible * properties. This array may be an empty array if no properties * are required. * @throws SQLException * if a database access error occurs * TODO check the correctness of implementation * TODO convert parameters into constants */ public DriverPropertyInfo[] getPropertyInfo(String url, Properties info) throws SQLException { return FBDriverPropertyManager.getDriverPropertyInfo(info); } /** * Gets the driver's major version number. Initially this should be 1. * * @return this driver's major version number */ public int getMajorVersion() { return 3; } /** * Gets the driver's minor version number. Initially this should be 0. * * @return this driver's minor version number */ public int getMinorVersion() { return 0; } /** * Reports whether this driver is a genuine JDBC * COMPLIANTTM driver. * A driver may only report true here if it passes the JDBC compliance * tests; otherwise it is required to return false. * * JDBC compliance requires full support for the JDBC API and full support * for SQL 92 Entry Level. It is expected that JDBC compliant drivers will * be available for all the major commercial databases. * * This method is not intended to encourage the development of non-JDBC * compliant drivers, but is a recognition of the fact that some vendors * are interested in using the JDBC API and framework for lightweight * databases that do not support full database functionality, or for * special databases such as document information retrieval where a SQL * implementation may not be feasible. */ public boolean jdbcCompliant() { return true; } public java.util.logging.Logger getParentLogger() throws SQLFeatureNotSupportedException { throw new FBDriverNotCapableException("Method getParentLogger() not supported"); } /** * Merges the properties from the JDBC URL and properties object, normalizes them to a standard name. *

* If a property with the exact same name is present in both, the property specified in the JDBC url takes * precedence. Short and long form {@code isc_dpb} properties will be merged if both are present, as will two * different (non-{@code isc_dpb}) aliases, but precedence is undefined. If a property is specified in the (short * or long) {@code isc_dpb} form and as an alias, then an exception is thrown. *

*

* The property name that is the result of normalization, is implementation specific behaviour, and might change in * a future version of Jaybird. When present, the (normalized) property `"database"` will be excluded, this also * might change in the future. *

* * @param jdbcUrl * JDBC Url * @param connectionProperties * Properties object * @return New map object with the merged and normalized connection properties * @throws SQLException * For failures to extract connection properties from {@code jdbcUrl} (URL decoding errors), or presence * of the same property under multiple aliases. * @since 3.0.10 */ public static Map normalizeProperties(String jdbcUrl, Properties connectionProperties) throws SQLException { Properties mergedProperties = mergeProperties(jdbcUrl, connectionProperties); return FBDriverPropertyManager.normalize(mergedProperties); } /** * Merges the properties from the JDBC URL and properties object. *

* If a property with the exact same name is present in both, the property specified in the JDBC url takes * precedence. *

* * @param jdbcUrl * JDBC Url * @param connectionProperties * Properties object * @return Map with connection properties * @throws SQLException * For failures to extract connection properties from {@code jdbcUrl} (URL decoding errors) */ private static Properties mergeProperties(String jdbcUrl, Properties connectionProperties) throws SQLException { Properties mergedProperties = new Properties(); if (connectionProperties != null) { for (String propertyName : connectionProperties.stringPropertyNames()) { mergedProperties.setProperty(propertyName, connectionProperties.getProperty(propertyName)); } } convertUrlParams(jdbcUrl, mergedProperties); return mergedProperties; } /** * Extract properties specified as URL parameter into the specified properties object. * * @param url * specified URL. * @param info * instance of {@link Properties} into which values should be extracted. * @throws SQLException * For failures to extract connection properties from {@code url} (URL decoding errors) */ private static void convertUrlParams(String url, Properties info) throws SQLException { if (url == null) { return; } int iQuestionMark = url.indexOf("?"); if (iQuestionMark == -1) { return; } String propString = url.substring(iQuestionMark + 1); StringTokenizer st = new StringTokenizer(propString, "&;"); while (st.hasMoreTokens()) { String propertyString = st.nextToken(); int iIs = propertyString.indexOf("="); if (iIs > -1) { String property = urlDecode(propertyString.substring(0, iIs), url); String value = urlDecode(propertyString.substring(iIs + 1), url); info.setProperty(property, value); } else { info.setProperty(urlDecode(propertyString, url), ""); } } } /** * Decodes URL encoded value, transforming exceptions to SQLExceptions * * @param encodedValue * The value to decode * @return The decoded value * @throws SQLException * If decoding fails (failures of {@link URLDecoder#decode(String, String)} */ private static String urlDecode(String encodedValue, String url) throws SQLException { try { return URLDecoder.decode(encodedValue, URL_CHARSET); } catch (RuntimeException | UnsupportedEncodingException e) { // NOTE: The UnsupportedEncodingException shouldn't occur because UTF-8 support is required in Java throw new FbExceptionBuilder() .nonTransientConnectionException(JaybirdErrorCodes.jb_invalidConnectionString) .messageParameter(url) .messageParameter(e.toString()) .cause(e) .toFlatSQLException(); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy