PooledDataSource.java
/*
* Copyright (c) 2001-2017, Zoltan Farkas All Rights Reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* GNU General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* Additionally licensed with:
*
* 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.
*/
package org.spf4j.pool.jdbc;
import com.google.common.annotations.Beta;
import java.io.PrintWriter;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.sql.Connection;
import java.sql.SQLException;
import java.sql.SQLFeatureNotSupportedException;
import java.sql.SQLTimeoutException;
import java.util.Properties;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.logging.Logger;
import javax.sql.DataSource;
import org.spf4j.jdbc.DataSourceEx;
import org.spf4j.recyclable.ObjectBorrowException;
import org.spf4j.recyclable.ObjectCreationException;
import org.spf4j.recyclable.ObjectDisposeException;
import org.spf4j.recyclable.RecyclingSupplier;
import org.spf4j.recyclable.impl.RecyclingSupplierBuilder;
/**
*
* @author zoly
*/
@Beta
public final class PooledDataSource implements DataSourceEx, AutoCloseable {
private final RecyclingSupplier<Connection> pool;
public PooledDataSource(final int initialSize, final int maxSize,
final String driverName, final String url, final String user, final String password)
throws ObjectCreationException {
this(initialSize, maxSize, new JdbcConnectionFactory(driverName, url, user, password));
}
public PooledDataSource(final int initialSize, final int maxSize,
final String driverName, final String url, final Properties properties, final int loginTimeoutSeconds)
throws ObjectCreationException {
this(initialSize, maxSize, new JdbcConnectionFactory(driverName, url, properties, loginTimeoutSeconds));
}
public PooledDataSource(final int initialSize, final int maxSize,
final RecyclingSupplier.Factory<Connection> jdbcConnectionFactory) throws ObjectCreationException {
RecyclingSupplierBuilder<Connection> builder
= new RecyclingSupplierBuilder<>(maxSize, jdbcConnectionFactory);
builder.withInitialSize(initialSize);
pool = builder.build();
}
@Override
public Connection getConnection() throws SQLException {
Connection raw;
try {
raw = pool.get();
} catch (InterruptedException | ObjectBorrowException | ObjectCreationException ex) {
throw new SQLException(ex);
} catch (TimeoutException ex) {
throw new SQLTimeoutException(ex);
}
return (Connection) Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),
new Class<?>[]{Connection.class}, new PooledConnectionInvocationHandler(raw, pool));
}
@Override
public Connection getConnection(final long timeout, final TimeUnit unit) throws SQLException {
Connection raw;
try {
raw = pool.tryGet(timeout, unit);
if (raw == null) {
throw new SQLTimeoutException("Unable to obtain connection in " + timeout + " " + unit);
}
} catch (InterruptedException | ObjectBorrowException | ObjectCreationException ex) {
throw new SQLException(ex);
}
return (Connection) Proxy.newProxyInstance(Thread.currentThread().getContextClassLoader(),
new Class<?>[]{Connection.class}, new PooledConnectionInvocationHandler(raw, pool));
}
@Override
public Connection getConnection(final String username, final String password)
throws SQLFeatureNotSupportedException {
throw new SQLFeatureNotSupportedException();
}
@Override
public PrintWriter getLogWriter() throws SQLFeatureNotSupportedException {
throw new SQLFeatureNotSupportedException();
}
@Override
public void setLogWriter(final PrintWriter out) throws SQLFeatureNotSupportedException {
throw new SQLFeatureNotSupportedException();
}
@Override
public void setLoginTimeout(final int seconds) throws SQLFeatureNotSupportedException {
throw new SQLFeatureNotSupportedException();
}
@Override
public int getLoginTimeout() throws SQLFeatureNotSupportedException {
throw new SQLFeatureNotSupportedException();
}
@Override
public <T> T unwrap(final Class<T> iface) throws SQLException {
if (iface.equals(DataSource.class) || iface.equals(PooledDataSource.class)) {
return (T) this;
} else {
throw new SQLException("Not a wrapper for " + iface);
}
}
@Override
public boolean isWrapperFor(final Class<?> iface) {
return iface.equals(DataSource.class) || iface.equals(PooledDataSource.class);
}
@Override
public Logger getParentLogger() throws SQLFeatureNotSupportedException {
throw new SQLFeatureNotSupportedException();
}
@Override
public String toString() {
return "PooledDataSource{" + "pool=" + pool + '}';
}
@Override
public void close() throws ObjectDisposeException, InterruptedException {
pool.dispose();
}
private static final class PooledConnectionInvocationHandler implements InvocationHandler {
private final Connection raw;
private final RecyclingSupplier<Connection> pool;
PooledConnectionInvocationHandler(final Connection raw, final RecyclingSupplier<Connection> pool) {
this.raw = raw;
this.pool = pool;
}
private Exception ex;
private boolean closed = false;
@Override
public Object invoke(final Object proxy, final Method method, final Object[] args) throws Exception {
String mName = method.getName();
if ("close".equals(mName)) {
if (!closed) {
pool.recycle(raw, ex);
ex = null;
closed = true;
}
return null;
} else {
if (closed) {
throw new IllegalStateException("not aowner of this connection,"
+ " it has been returned already to " + pool);
}
try {
return method.invoke(raw, args);
} catch (IllegalAccessException | InvocationTargetException | RuntimeException e) {
ex = e;
throw e;
}
}
}
}
}