/*
* (C) Copyright 2012-2016 Nuxeo SA (http://nuxeo.com/) and others.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* Contributors:
* Florent Guillaume
*/
package org.nuxeo.runtime.datasource;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.sql.Connection;
import java.sql.SQLException;
import javax.naming.NamingException;
import javax.sql.DataSource;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.nuxeo.common.utils.JDBCUtils;
import org.nuxeo.runtime.api.Framework;
import org.nuxeo.runtime.datasource.PooledDataSourceRegistry.PooledDataSource;
/**
* Helper to acquire a JDBC {@link Connection} from a datasource name.
*
* @since 5.7
*/
public class ConnectionHelper {
private static final Log log = LogFactory.getLog(ConnectionHelper.class);
/**
* Tries to unwrap the connection to get the real physical one (returned by the original datasource).
* <p>
* This should only be used by code that needs to cast the connection to a driver-specific class to use
* driver-specific features.
*
* @throws SQLException if no actual physical connection was allocated yet
*/
public static Connection unwrap(Connection connection) throws SQLException {
if (connection instanceof org.tranql.connector.jdbc.ConnectionHandle) {
return ((org.tranql.connector.jdbc.ConnectionHandle) connection).getAssociation().getPhysicalConnection();
}
// now try Apache DBCP unwrap (standard or Tomcat), to skip datasource wrapping layers
// this needs accessToUnderlyingConnectionAllowed=true in the pool config
try {
Method m = connection.getClass().getMethod("getInnermostDelegate");
m.setAccessible(true); // needed, method of inner private class
Connection delegate = (Connection) m.invoke(connection);
if (delegate == null) {
log.error("Cannot access underlying connection, you must use "
+ "accessToUnderlyingConnectionAllowed=true in the pool configuration");
} else {
connection = delegate;
}
} catch (NoSuchMethodException | SecurityException | IllegalAccessException | InvocationTargetException e) {
// ignore missing method, connection not coming from Apache pool
}
return connection;
}
/**
* Gets a new connection for the given dataSource. The connection <strong>MUST</strong> be closed in a finally block
* when code is done using it.
*
* @param dataSourceName the datasource for which the connection is requested
* @return a new connection
*/
public static Connection getConnection(String dataSourceName) throws SQLException {
return getConnection(dataSourceName, false);
}
/**
* Gets a new connection for the given dataSource. The connection <strong>MUST</strong> be closed in a finally block
* when code is done using it.
*
* @param dataSourceName the datasource for which the connection is requested
* @param noSharing {@code true} if this connection must not be shared with others
* @return a new connection
*/
public static Connection getConnection(String dataSourceName, boolean noSharing) throws SQLException {
DataSource dataSource = getDataSource(dataSourceName);
if (dataSource instanceof PooledDataSource) {
return ((PooledDataSource) dataSource).getConnection(noSharing);
} else {
return JDBCUtils.getConnection(dataSource);
}
}
/**
* Gets a datasource from a datasource name, or in test mode use test connection parameters.
*
* @param dataSourceName the datasource name
* @return the datasource
*/
private static DataSource getDataSource(String dataSourceName) throws SQLException {
try {
return DataSourceHelper.getDataSource(dataSourceName);
} catch (NamingException e) {
if (Framework.isTestModeSet()) {
String url = Framework.getProperty("nuxeo.test.vcs.url");
String user = Framework.getProperty("nuxeo.test.vcs.user");
String password = Framework.getProperty("nuxeo.test.vcs.password");
if (url != null && user != null) {
return new DataSourceFromUrl(url, user, password); // driver?
}
}
throw new SQLException("Cannot find datasource: " + dataSourceName, e);
}
}
}