org.hsqldb.jdbc.JDBCDriver Maven / Gradle / Ivy
/* Copyright (c) 2001-2011, The HSQL Development Group
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* Neither the name of the HSQL Development Group nor the names of its
* contributors may be used to endorse or promote products derived from this
* software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL HSQL DEVELOPMENT GROUP, HSQLDB.ORG,
* OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.hsqldb.jdbc;
import java.sql.Connection;
import java.sql.Driver;
import java.sql.DriverManager;
import java.sql.DriverPropertyInfo;
import java.sql.SQLException;
import java.util.Properties;
import org.hsqldb.DatabaseURL;
import org.hsqldb.error.ErrorCode;
import org.hsqldb.persist.HsqlDatabaseProperties;
import org.hsqldb.persist.HsqlProperties;
// fredt@users 20011220 - patch 1.7.0 by fredt
// new version numbering scheme
// fredt@users 20020320 - patch 1.7.0 - JDBC 2 support and error trapping
// JDBC 2 methods can now be called from jdk 1.1.x - see javadoc comments
// fredt@users 20030528 - patch 1.7.2 suggested by Gerhard Hiller - support for properties in URL
// boucherb@users 20051207 - patch 1.8.x initial JDBC 4.0 support work
/**
* Provides the java.sql.Driver interface implementation required by
* the JDBC specification.
*
* The Java SQL framework allows for multiple database drivers.
*
* The DriverManager will try to load as many drivers as it can find and
* then for any given connection request, it will ask each driver in turn
* to try to connect to the target URL.
*
* The application developer will normally not need to call any function of
* the Driver directly. All required calls are made by the DriverManager.
*
*
*
* HSQLDB-Specific Information:
*
* When the HSQL Database Engine Driver class is loaded, it creates an
* instance of itself and register it with the DriverManager. This means
* that a user can load and register the HSQL Database Engine driver by
* calling:
*
* Class.forName("org.hsqldb.jdbc.JDBCDriver")
*
*
* For detailed information about how to obtain HSQLDB JDBC Connections,
* please see {@link org.hsqldb.jdbc.JDBCConnection JDBCConnection}.
*
*
*
* JRE 1.1.x Notes:
*
* In general, JDBC 2 support requires Java 1.2 and above, and JDBC3 requires
* Java 1.4 and above. In HSQLDB, support for methods introduced in different
* versions of JDBC depends on the JDK version used for compiling and building
* HSQLDB.
*
* Since 1.7.0, it is possible to build the product so that
* all JDBC 2 methods can be called while executing under the version 1.1.x
* Java Runtime EnvironmentTM.
* However, in addition to this technique requiring explicit casts to the
* org.hsqldb.jdbc.* classes, some of the method calls also require
* int
values that are defined only in the JDBC 2 or greater
* version of the {@link java.sql.ResultSet ResultSet} interface. For this
* reason, when the product is compiled under JDK 1.1.x, these values are
* defined in {@link org.hsqldb.jdbc.JDBCResultSet JDBCResultSet}.
*
* In a JRE 1.1.x environment, calling JDBC 2 methods that take or return the
* JDBC2-only ResultSet
values can be achieved by referring
* to them in parameter specifications and return value comparisons,
* respectively, as follows:
*
*
* JDBCResultSet.FETCH_FORWARD
* JDBCResultSet.TYPE_FORWARD_ONLY
* JDBCResultSet.TYPE_SCROLL_INSENSITIVE
* JDBCResultSet.CONCUR_READ_ONLY
* // etc.
*
*
* However, please note that code written to use HSQLDB JDBC 2 features under
* JDK 1.1.x will not be compatible for use with other JDBC 2 drivers. Please
* also note that this feature is offered solely as a convenience to developers
* who must work under JDK 1.1.x due to operating constraints, yet wish to
* use some of the more advanced features available under the JDBC 2
* specification.
*
*
*
*
* JDBC 4.0 notes:
*
* Starting with JDBC 4.0 (JDK 1.6), the DriverManager
methods
* getConnection
and getDrivers
have been
* enhanced to support the Java Standard Edition Service Provider mechanism.
* When built under a Java runtime that supports JDBC 4.0, HSQLDB distribution
* jars containing the Driver implementation also include the file
* META-INF/services/java.sql.Driver
. This file contains the fully
* qualified class name ('org.hsqldb.jdbc.JDBCDriver') of the HSQLDB implementation
* of java.sql.Driver
.
*
* Hence, under JDBC 4.0 or greater, applications no longer need to explictly
* load the HSQLDB JDBC driver using Class.forName()
. Of course,
* existing programs which do load JDBC drivers using
* Class.forName()
will continue to work without modification.
*
*
* @author Campbell Boucher-Burnet (boucherb@users dot sourceforge.net)
* @author Fred Toussi (fredt@users dot sourceforge.net)
* @version 2.2.9
* @revised JDK 1.7, HLSQLDB 2.0.1
*
* * Returns "null" if this 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 throws an 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 pass
* arbitrary string tag/value pairs as connection arguments.
* Normally at least "user" and "password" properties should be
* included in the Properties
object.
* * *
HSQLDB-Specific Information:
* * For the HSQL Database Engine, at least "user" and * "password" properties should be included in the Properties.
* * From version 1.7.1, two optional properties are supported:
* *
-
*
get_column_name
(default true) - if set to false, * a ResultSetMetaData.getColumnName() call will return the user * defined label (getColumnLabel()) instead of the column * name.
* * This property is available in order to achieve * compatibility with certain non-HSQLDB JDBC driver * implementations.
*
* strict_md
if set to true, some ResultSetMetaData * methods return more strict values for compatibility * reasons.
*
*
* From version 1.8.0.x, strict_md
is deprecated (ignored)
* because metadata reporting is always strict (JDBC-compliant), and
* three new optional properties are supported:
* *
-
*
ifexits
(default false) - when true, an exception * is raised when attempting to connect to an in-process * file: or mem: scheme database instance if it has not yet been * created. When false, an in-process file: or mem: scheme * database instance is created automatically if it has not yet * been created. This property does not apply to requests for * network or res: (i.e. files_in_jar) scheme connections.- * *
shutdown
(default false) - when true, the * the target database mimics the behaviour of 1.7.1 and older * versions. When the last connection to a database is closed, * the database is automatically shut down. The property takes * effect only when the first connection is made to the database. * This means the connection that opens the database. It has no * effect if used with subsequent, simultaneous connections.
* * This command has two uses. One is for test suites, where * connections to the database are made from one JVM context, * immediately followed by another context. The other use is for * applications where it is not easy to configure the environment * to shutdown the database. Examples reported by users include * web application servers, where the closing of the last * connection coincides with the web application being shut down. *
*
* default_schema
- backwards compatibility feature. * To be used for clients written before HSQLDB schema support. * Denotes whether to use the default schema when a schema * qualifier is not included in a database object's SQL identifier * character sequence. Also affects the semantics of * DatabaseMetaData calls that supply null-valued schemaNamePattern * parameter values.
*
Connection
object that represents a
* connection to the URL
* @exception SQLException if a database access error occurs
*/
public Connection connect(String url,
Properties info) throws SQLException {
if (url.regionMatches(true, 0, DatabaseURL.S_URL_INTERNAL, 0,
DatabaseURL.S_URL_INTERNAL.length())) {
JDBCConnection conn = (JDBCConnection) threadConnection.get();
if (conn == null) {
return null;
}
return conn;
}
return getConnection(url, info);
}
/**
* The static equivalent of the connect(String,Properties)
* method.
*
* @param url the URL of the database to which to connect
* @param info a list of arbitrary string tag/value pairs as connection
* arguments including at least at a "user" and a "password" property
* @throws java.sql.SQLException if a database access error occurs
* @return a Connection
object that represents a
* connection to the URL
*/
//#ifdef JAVA6
@SuppressWarnings("deprecation")
//#endif JAVA6
public static Connection getConnection(String url,
Properties info) throws SQLException {
final HsqlProperties props = DatabaseURL.parseURL(url, true, false);
if (props == null) {
// supposed to be an HSQLDB driver url but has errors
throw JDBCUtil.invalidArgument();
} else if (props.isEmpty()) {
// is not an HSQLDB driver url
return null;
}
long timeout = 0;
if (info != null) {
timeout = HsqlProperties.getIntegerProperty(info, "loginTimeout", 0);
}
props.addProperties(info);
if (timeout == 0) {
timeout = DriverManager.getLoginTimeout();
}
// @todo: maybe impose some sort of sane restriction
// on network connections regarless of user
// specification?
if (timeout == 0) {
// no timeout restriction
return new JDBCConnection(props);
}
String connType = props.getProperty("connection_type");
if (DatabaseURL.isInProcessDatabaseType(connType)) {
return new JDBCConnection(props);
}
// @todo: Better: ThreadPool? HsqlTimer with callback?
final JDBCConnection[] conn = new JDBCConnection[1];
final SQLException[] ex = new SQLException[1];
Thread t = new Thread() {
public void run() {
try {
conn[0] = new JDBCConnection(props);
} catch (SQLException se) {
ex[0] = se;
}
}
};
t.start();
final long start = System.currentTimeMillis();
try {
t.join(1000 * timeout);
} catch (InterruptedException ie) {
}
try {
// PRE:
// deprecated, but should be ok, since neither
// the HSQLClientConnection or the HTTPClientConnection
// constructor will ever hold monitors on objects in
// an inconsistent state, such that damaged objects
// become visible to other threads with the
// potential of arbitrary behavior.
t.stop();
} catch (Exception e) {
} finally {
try {
t.setContextClassLoader(null);
} catch (Throwable th) {
}
}
if (ex[0] != null) {
throw ex[0];
}
if (conn[0] != null) {
return conn[0];
}
throw JDBCUtil.sqlException(ErrorCode.X_08501);
}
/**
* 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
*/
// fredt@users - patch 1.7.0 - allow mixedcase url's
public boolean acceptsURL(String url) {
if (url == null) {
return false;
}
if (url.regionMatches(true, 0, DatabaseURL.S_URL_PREFIX, 0,
DatabaseURL.S_URL_PREFIX.length())) {
return true;
}
if (url.regionMatches(true, 0, DatabaseURL.S_URL_INTERNAL, 0,
DatabaseURL.S_URL_INTERNAL.length())) {
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.
* * *
HSQLDB-Specific Information:
* * HSQLDB uses the values submitted in info to set the value for * each DriverPropertyInfo object returned. It does not use the default * value that it would use for the property if the value is null.
* *
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.
* * *
HSQLDB-Specific Information:
* * HSQLDB 2.0 is aimed to be compliant with JDBC 4.0 specification. * It supports SQL 92 Entry Level and beyond. *
true
if this driver is JDBC Compliant;
* false
otherwise
*/
public boolean jdbcCompliant() {
return true;
}
//------------------------- JDBC 4.1 -----------------------------------
/**
* Return the parent Logger of all the Loggers used by this driver. This
* should be the Logger farthest from the root Logger that is
* still an ancestor of all of the Loggers used by this driver. Configuring
* this Logger will affect all of the log messages generated by the driver.
* In the worst case, this may be the root Logger.
*
* @return the parent Logger for this driver
* @throws SQLFeatureNotSupportedException if the driver does not use java.util.logging.
* @since JDK 1.7 M11 2010/09/10 (b123), HSQLDB 2.0.1
*/
//#ifdef JAVA6
public java.util.logging
.Logger getParentLogger() throws java.sql
.SQLFeatureNotSupportedException {
throw (java.sql.SQLFeatureNotSupportedException) JDBCUtil.notSupported();
}
//#endif
public static JDBCDriver driverInstance;
static {
try {
driverInstance = new JDBCDriver();
DriverManager.registerDriver(driverInstance);
} catch (Exception e) {
}
}
/**
* As a separate instance of this class is registered with DriverManager
* for each class loader, the threadConnection is not declared as static.
* The registered instance is kept to allow access to the its
* threadConnection.
*
*/
//#ifdef JAVA6
public final ThreadLocal threadConnection =
new ThreadLocal();
//#else
/*
public final ThreadLocal threadConnection = new ThreadLocal();
*/
//#endif JAVA6
}