buckelieg.jdbc.DefaultConnectionManager Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jdbc-fn Show documentation
Show all versions of jdbc-fn Show documentation
Functional style programming with plain JDBC
The newest version!
/*
* Copyright 2024- Anatoly Kutyakov
*
* 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 buckelieg.jdbc;
import buckelieg.jdbc.fn.TrySupplier;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
final class DefaultConnectionManager implements ConnectionManager {
private final TrySupplier connectionSupplier;
private final int maxConnections;
private final BlockingQueue pool;
private final List obtainedConnections;
private final AtomicInteger size = new AtomicInteger(0);
private final AtomicBoolean isShuttingDown = new AtomicBoolean(false);
DefaultConnectionManager(TrySupplier connectionSupplier, int maxConnections) {
this.connectionSupplier = connectionSupplier;
this.maxConnections = maxConnections;
this.pool = new ArrayBlockingQueue<>(maxConnections);
this.obtainedConnections = new ArrayList<>(maxConnections);
}
@Nonnull
@Override
public Connection getConnection() throws SQLException {
if (isShuttingDown.get()) throw new SQLException("Connection pool is shutting down");
Connection connection;
try {
if (size.getAndIncrement() < maxConnections) {
connection = connectionSupplier.get();
if (null == connection) throw new NullPointerException("Provided connection is null");
if (obtainedConnections.contains(connection)) {
connection = pool.take();
} else obtainedConnections.add(connection);
} else {
connection = pool.take();
}
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new SQLException(e);
}
connection.setAutoCommit(false);
return connection;
}
@Override
public void close(@Nullable Connection connection) throws SQLException {
if (null == connection) return;
connection.setAutoCommit(true);
if (!pool.offer(connection)) {
throw new SQLException("Connection pool is full");
}
}
@Override
public void close() throws SQLException {
isShuttingDown.set(true);
pool.clear();
SQLException exception = null;
for (Connection connection : obtainedConnections) {
try {
connection.close();
} catch (SQLException e) {
if (null == exception) exception = new SQLException(e);
else exception.setNextException(e);
}
}
obtainedConnections.clear();
if (null != exception) {
throw exception;
}
}
}