com.mongodb.internal.connection.DefaultConnectionPool Maven / Gradle / Ivy
Show all versions of mongodb-driver-core Show documentation
/*
* Copyright 2008-present MongoDB, Inc.
*
* 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 com.mongodb.internal.connection;
import com.mongodb.MongoConnectionPoolClearedException;
import com.mongodb.MongoException;
import com.mongodb.MongoInterruptedException;
import com.mongodb.MongoServerUnavailableException;
import com.mongodb.MongoTimeoutException;
import com.mongodb.RequestContext;
import com.mongodb.annotations.NotThreadSafe;
import com.mongodb.annotations.ThreadSafe;
import com.mongodb.connection.ClusterId;
import com.mongodb.connection.ConnectionDescription;
import com.mongodb.connection.ConnectionId;
import com.mongodb.connection.ConnectionPoolSettings;
import com.mongodb.connection.ServerDescription;
import com.mongodb.connection.ServerId;
import com.mongodb.event.ConnectionCheckOutFailedEvent;
import com.mongodb.event.ConnectionCheckOutFailedEvent.Reason;
import com.mongodb.event.ConnectionCheckOutStartedEvent;
import com.mongodb.event.ConnectionCheckedInEvent;
import com.mongodb.event.ConnectionCheckedOutEvent;
import com.mongodb.event.ConnectionClosedEvent;
import com.mongodb.event.ConnectionCreatedEvent;
import com.mongodb.event.ConnectionPoolClearedEvent;
import com.mongodb.event.ConnectionPoolClosedEvent;
import com.mongodb.event.ConnectionPoolCreatedEvent;
import com.mongodb.event.ConnectionPoolListener;
import com.mongodb.event.ConnectionPoolReadyEvent;
import com.mongodb.event.ConnectionReadyEvent;
import com.mongodb.internal.VisibleForTesting;
import com.mongodb.internal.async.SingleResultCallback;
import com.mongodb.internal.connection.SdamServerDescriptionManager.SdamIssue;
import com.mongodb.internal.diagnostics.logging.Logger;
import com.mongodb.internal.diagnostics.logging.Loggers;
import com.mongodb.internal.event.EventReasonMessageResolver;
import com.mongodb.internal.inject.OptionalProvider;
import com.mongodb.internal.logging.LogMessage;
import com.mongodb.internal.logging.StructuredLogger;
import com.mongodb.internal.session.SessionContext;
import com.mongodb.internal.thread.DaemonThreadFactory;
import com.mongodb.internal.time.TimePoint;
import com.mongodb.internal.time.Timeout;
import com.mongodb.lang.NonNull;
import com.mongodb.lang.Nullable;
import org.bson.ByteBuf;
import org.bson.codecs.Decoder;
import org.bson.types.ObjectId;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Deque;
import java.util.LinkedList;
import java.util.List;
import java.util.Optional;
import java.util.Queue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.RejectedExecutionException;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.LongAdder;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantLock;
import java.util.concurrent.locks.StampedLock;
import java.util.function.Consumer;
import java.util.function.Predicate;
import java.util.function.Supplier;
import static com.mongodb.assertions.Assertions.assertFalse;
import static com.mongodb.assertions.Assertions.assertNotNull;
import static com.mongodb.assertions.Assertions.assertNull;
import static com.mongodb.assertions.Assertions.assertTrue;
import static com.mongodb.assertions.Assertions.fail;
import static com.mongodb.assertions.Assertions.isTrue;
import static com.mongodb.assertions.Assertions.notNull;
import static com.mongodb.event.ConnectionClosedEvent.Reason.ERROR;
import static com.mongodb.internal.Locks.lockInterruptibly;
import static com.mongodb.internal.Locks.withLock;
import static com.mongodb.internal.Locks.withUnfairLock;
import static com.mongodb.internal.VisibleForTesting.AccessModifier.PRIVATE;
import static com.mongodb.internal.async.ErrorHandlingResultCallback.errorHandlingCallback;
import static com.mongodb.internal.connection.ConcurrentPool.INFINITE_SIZE;
import static com.mongodb.internal.connection.ConcurrentPool.sizeToString;
import static com.mongodb.internal.event.EventListenerHelper.getConnectionPoolListener;
import static com.mongodb.internal.logging.LogMessage.Component.CONNECTION;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.DRIVER_CONNECTION_ID;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.DURATION_MS;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.ERROR_DESCRIPTION;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.MAX_CONNECTING;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.MAX_IDLE_TIME_MS;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.MAX_POOL_SIZE;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.MIN_POOL_SIZE;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.REASON_DESCRIPTION;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.SERVER_HOST;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.SERVER_PORT;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.SERVICE_ID;
import static com.mongodb.internal.logging.LogMessage.Entry.Name.WAIT_QUEUE_TIMEOUT_MS;
import static com.mongodb.internal.logging.LogMessage.Level.DEBUG;
import static com.mongodb.internal.thread.InterruptionUtil.interruptAndCreateMongoInterruptedException;
import static java.lang.String.format;
import static java.util.concurrent.TimeUnit.MILLISECONDS;
import static java.util.concurrent.TimeUnit.NANOSECONDS;
@SuppressWarnings("deprecation")
@ThreadSafe
final class DefaultConnectionPool implements ConnectionPool {
private static final Logger LOGGER = Loggers.getLogger("connection");
private static final StructuredLogger STRUCTURED_LOGGER = new StructuredLogger("connection");
private final ConcurrentPool pool;
private final ConnectionPoolSettings settings;
private final BackgroundMaintenanceManager backgroundMaintenance;
private final AsyncWorkManager asyncWorkManager;
private final ConnectionPoolListener connectionPoolListener;
private final ServerId serverId;
private final PinnedStatsManager pinnedStatsManager = new PinnedStatsManager();
private final ServiceStateManager serviceStateManager = new ServiceStateManager();
private final ConnectionGenerationSupplier connectionGenerationSupplier;
private final OpenConcurrencyLimiter openConcurrencyLimiter;
private final StateAndGeneration stateAndGeneration;
private final OptionalProvider sdamProvider;
@VisibleForTesting(otherwise = PRIVATE)
DefaultConnectionPool(final ServerId serverId, final InternalConnectionFactory internalConnectionFactory,
final ConnectionPoolSettings settings, final OptionalProvider sdamProvider) {
this(serverId, internalConnectionFactory, settings, InternalConnectionPoolSettings.builder().build(), sdamProvider);
}
/**
* @param sdamProvider For handling exceptions via the
*
* SDAM machinery as specified
*
* here.
* Must provide an {@linkplain Optional#isPresent() empty} {@link Optional} if created in load-balanced mode,
* otherwise must provide a non-empty {@link Optional}.
*/
DefaultConnectionPool(final ServerId serverId, final InternalConnectionFactory internalConnectionFactory,
final ConnectionPoolSettings settings, final InternalConnectionPoolSettings internalSettings,
final OptionalProvider sdamProvider) {
this.serverId = notNull("serverId", serverId);
this.settings = notNull("settings", settings);
UsageTrackingInternalConnectionItemFactory connectionItemFactory =
new UsageTrackingInternalConnectionItemFactory(internalConnectionFactory);
pool = new ConcurrentPool<>(maxSize(settings), connectionItemFactory, format("The server at %s is no longer available",
serverId.getAddress()));
this.sdamProvider = assertNotNull(sdamProvider);
this.connectionPoolListener = getConnectionPoolListener(settings);
backgroundMaintenance = new BackgroundMaintenanceManager();
connectionPoolCreated(connectionPoolListener, serverId, settings);
openConcurrencyLimiter = new OpenConcurrencyLimiter(settings.getMaxConnecting());
asyncWorkManager = new AsyncWorkManager(internalSettings.isPrestartAsyncWorkManager());
stateAndGeneration = new StateAndGeneration();
connectionGenerationSupplier = new ConnectionGenerationSupplier() {
@Override
public int getGeneration() {
return stateAndGeneration.generation();
}
@Override
public int getGeneration(@NonNull final ObjectId serviceId) {
return serviceStateManager.getGeneration(serviceId);
}
};
}
@Override
public InternalConnection get(final OperationContext operationContext) {
return get(operationContext, settings.getMaxWaitTime(MILLISECONDS), MILLISECONDS);
}
@Override
public InternalConnection get(final OperationContext operationContext, final long timeoutValue, final TimeUnit timeUnit) {
TimePoint checkoutStart = connectionCheckoutStarted(operationContext);
Timeout timeout = Timeout.started(timeoutValue, timeUnit, checkoutStart);
try {
stateAndGeneration.throwIfClosedOrPaused();
PooledConnection connection = getPooledConnection(timeout);
if (!connection.opened()) {
connection = openConcurrencyLimiter.openOrGetAvailable(connection, timeout);
}
connection.checkedOutForOperation(operationContext);
connectionCheckedOut(operationContext, connection, checkoutStart);
return connection;
} catch (Exception e) {
throw (RuntimeException) checkOutFailed(e, operationContext, checkoutStart);
}
}
@Override
public void getAsync(final OperationContext operationContext, final SingleResultCallback callback) {
TimePoint checkoutStart = connectionCheckoutStarted(operationContext);
Timeout timeout = Timeout.started(settings.getMaxWaitTime(NANOSECONDS), checkoutStart);
SingleResultCallback eventSendingCallback = (connection, failure) -> {
SingleResultCallback errHandlingCallback = errorHandlingCallback(callback, LOGGER);
if (failure == null) {
connection.checkedOutForOperation(operationContext);
connectionCheckedOut(operationContext, connection, checkoutStart);
errHandlingCallback.onResult(connection, null);
} else {
errHandlingCallback.onResult(null, checkOutFailed(failure, operationContext, checkoutStart));
}
};
try {
stateAndGeneration.throwIfClosedOrPaused();
} catch (Exception e) {
eventSendingCallback.onResult(null, e);
return;
}
asyncWorkManager.enqueue(new Task(timeout, t -> {
if (t != null) {
eventSendingCallback.onResult(null, t);
} else {
PooledConnection connection;
try {
connection = getPooledConnection(timeout);
} catch (Exception e) {
eventSendingCallback.onResult(null, e);
return;
}
if (connection.opened()) {
eventSendingCallback.onResult(connection, null);
} else {
openConcurrencyLimiter.openAsyncWithConcurrencyLimit(connection, timeout, eventSendingCallback);
}
}
}));
}
/**
* Sends {@link ConnectionCheckOutFailedEvent}
* and returns {@code t} if it is not {@link MongoOpenConnectionInternalException},
* or returns {@code t.}{@linkplain MongoOpenConnectionInternalException#getCause() getCause()} otherwise.
*/
private Throwable checkOutFailed(final Throwable t, final OperationContext operationContext, final TimePoint checkoutStart) {
Throwable result = t;
Reason reason;
if (t instanceof MongoTimeoutException) {
reason = Reason.TIMEOUT;
} else if (t instanceof MongoOpenConnectionInternalException) {
reason = Reason.CONNECTION_ERROR;
result = t.getCause();
} else if (t instanceof MongoConnectionPoolClearedException) {
reason = Reason.CONNECTION_ERROR;
} else if (ConcurrentPool.isPoolClosedException(t)) {
reason = Reason.POOL_CLOSED;
} else {
reason = Reason.UNKNOWN;
}
Duration checkoutDuration = checkoutStart.elapsed();
ClusterId clusterId = serverId.getClusterId();
if (requiresLogging(clusterId)) {
String message = "Checkout failed for connection to {}:{}. Reason: {}.[ Error: {}.] Duration: {} ms";
List entries = createBasicEntries();
entries.add(new LogMessage.Entry(REASON_DESCRIPTION, EventReasonMessageResolver.getMessage(reason)));
entries.add(new LogMessage.Entry(ERROR_DESCRIPTION, reason == Reason.CONNECTION_ERROR ? result.toString() : null));
entries.add(new LogMessage.Entry(DURATION_MS, checkoutDuration.toMillis()));
logMessage("Connection checkout failed", clusterId, message, entries);
}
connectionPoolListener.connectionCheckOutFailed(
new ConnectionCheckOutFailedEvent(serverId, operationContext.getId(), reason, checkoutDuration.toNanos()));
return result;
}
@Override
public void invalidate(@Nullable final Throwable cause) {
assertFalse(isLoadBalanced());
if (stateAndGeneration.pauseAndIncrementGeneration(cause)) {
openConcurrencyLimiter.signalClosedOrPaused();
}
}
@Override
public void ready() {
stateAndGeneration.ready();
}
public void invalidate(final ObjectId serviceId, final int generation) {
assertTrue(isLoadBalanced());
if (generation == InternalConnection.NOT_INITIALIZED_GENERATION) {
return;
}
if (serviceStateManager.incrementGeneration(serviceId, generation)) {
ClusterId clusterId = serverId.getClusterId();
if (requiresLogging(clusterId)) {
String message = "Connection pool for {}:{} cleared for serviceId {}";
List entries = createBasicEntries();
entries.add(new LogMessage.Entry(SERVICE_ID, serviceId.toHexString()));
logMessage("Connection pool cleared", clusterId, message, entries);
}
connectionPoolListener.connectionPoolCleared(new ConnectionPoolClearedEvent(this.serverId, serviceId));
}
}
@Override
public void close() {
if (stateAndGeneration.close()) {
pool.close();
backgroundMaintenance.close();
asyncWorkManager.close();
openConcurrencyLimiter.signalClosedOrPaused();
logEventMessage("Connection pool closed", "Connection pool closed for {}:{}");
connectionPoolListener.connectionPoolClosed(new ConnectionPoolClosedEvent(serverId));
}
}
@Override
public int getGeneration() {
return stateAndGeneration.generation();
}
private PooledConnection getPooledConnection(final Timeout timeout) throws MongoTimeoutException {
try {
UsageTrackingInternalConnection internalConnection = pool.get(timeout.remainingOrInfinite(NANOSECONDS), NANOSECONDS);
while (shouldPrune(internalConnection)) {
pool.release(internalConnection, true);
internalConnection = pool.get(timeout.remainingOrInfinite(NANOSECONDS), NANOSECONDS);
}
return new PooledConnection(internalConnection);
} catch (MongoTimeoutException e) {
throw createTimeoutException(timeout);
}
}
@Nullable
private PooledConnection getPooledConnectionImmediateUnfair() {
UsageTrackingInternalConnection internalConnection = pool.getImmediateUnfair();
while (internalConnection != null && shouldPrune(internalConnection)) {
pool.release(internalConnection, true);
internalConnection = pool.getImmediateUnfair();
}
return internalConnection == null ? null : new PooledConnection(internalConnection);
}
private MongoTimeoutException createTimeoutException(final Timeout timeout) {
int numPinnedToCursor = pinnedStatsManager.getNumPinnedToCursor();
int numPinnedToTransaction = pinnedStatsManager.getNumPinnedToTransaction();
if (numPinnedToCursor == 0 && numPinnedToTransaction == 0) {
return new MongoTimeoutException(format("Timed out after %s while waiting for a connection to server %s.",
timeout.toUserString(), serverId.getAddress()));
} else {
int maxSize = pool.getMaxSize();
int numInUse = pool.getInUseCount();
/* At this point in an execution we consider at least one of `numPinnedToCursor`, `numPinnedToTransaction` to be positive.
* `numPinnedToCursor`, `numPinnedToTransaction` and `numInUse` are not a snapshot view,
* but we still must maintain the following invariants:
* - numInUse > 0
* we consider at least one of `numPinnedToCursor`, `numPinnedToTransaction` to be positive,
* so if we observe `numInUse` to be 0, we have to estimate it based on `numPinnedToCursor` and `numPinnedToTransaction`;
* - numInUse < maxSize
* `numInUse` must not exceed the limit in situations when we estimate `numInUse`;
* - numPinnedToCursor + numPinnedToTransaction <= numInUse
* otherwise the numbers do not make sense.
*/
if (numInUse == 0) {
numInUse = Math.min(
numPinnedToCursor + numPinnedToTransaction, // must be at least a big as this sum but not bigger than `maxSize`
maxSize);
}
numPinnedToCursor = Math.min(
numPinnedToCursor, // prefer the observed value, but it must not be bigger than `numInUse`
numInUse);
numPinnedToTransaction = Math.min(
numPinnedToTransaction, // prefer the observed value, but it must not be bigger than `numInUse` - `numPinnedToCursor`
numInUse - numPinnedToCursor);
int numOtherInUse = numInUse - numPinnedToCursor - numPinnedToTransaction;
assertTrue(numOtherInUse >= 0);
assertTrue(numPinnedToCursor + numPinnedToTransaction + numOtherInUse <= maxSize);
return new MongoTimeoutException(format("Timed out after %s while waiting for a connection to server %s. Details: "
+ "maxPoolSize: %s, connections in use by cursors: %d, connections in use by transactions: %d, "
+ "connections in use by other operations: %d",
timeout.toUserString(), serverId.getAddress(),
sizeToString(maxSize), numPinnedToCursor, numPinnedToTransaction,
numOtherInUse));
}
}
@VisibleForTesting(otherwise = PRIVATE)
ConcurrentPool getPool() {
return pool;
}
/**
* Synchronously prune idle connections and ensure the minimum pool size.
*/
@VisibleForTesting(otherwise = PRIVATE)
void doMaintenance() {
Predicate silentlyComplete = e ->
e instanceof MongoInterruptedException || e instanceof MongoTimeoutException
|| e instanceof MongoConnectionPoolClearedException || ConcurrentPool.isPoolClosedException(e);
try {
pool.prune();
if (shouldEnsureMinSize()) {
pool.ensureMinSize(settings.getMinSize(), newConnection -> {
try {
openConcurrencyLimiter.openImmediatelyAndTryHandOverOrRelease(new PooledConnection(newConnection));
} catch (MongoException | MongoOpenConnectionInternalException e) {
RuntimeException actualException = e instanceof MongoOpenConnectionInternalException
? (RuntimeException) e.getCause()
: e;
try {
sdamProvider.optional().ifPresent(sdam -> {
if (!silentlyComplete.test(actualException)) {
sdam.handleExceptionBeforeHandshake(SdamIssue.specific(actualException, sdam.context(newConnection)));
}
});
} catch (Exception suppressed) {
actualException.addSuppressed(suppressed);
}
throw actualException;
}
});
}
} catch (Exception e) {
if (!silentlyComplete.test(e)) {
LOGGER.warn("Exception thrown during connection pool background maintenance task", e);
throw e;
}
}
}
private boolean shouldEnsureMinSize() {
return settings.getMinSize() > 0;
}
private boolean shouldPrune(final UsageTrackingInternalConnection connection) {
return fromPreviousGeneration(connection) || pastMaxLifeTime(connection) || pastMaxIdleTime(connection);
}
private boolean pastMaxIdleTime(final UsageTrackingInternalConnection connection) {
return expired(connection.getLastUsedAt(), System.currentTimeMillis(), settings.getMaxConnectionIdleTime(MILLISECONDS));
}
private boolean pastMaxLifeTime(final UsageTrackingInternalConnection connection) {
return expired(connection.getOpenedAt(), System.currentTimeMillis(), settings.getMaxConnectionLifeTime(MILLISECONDS));
}
private boolean fromPreviousGeneration(final UsageTrackingInternalConnection connection) {
int generation = connection.getGeneration();
if (generation == InternalConnection.NOT_INITIALIZED_GENERATION) {
return false;
}
ObjectId serviceId = connection.getDescription().getServiceId();
if (serviceId != null) {
return serviceStateManager.getGeneration(serviceId) > generation;
} else {
return stateAndGeneration.generation() > generation;
}
}
private boolean expired(final long startTime, final long curTime, final long maxTime) {
return maxTime != 0 && curTime - startTime > maxTime;
}
/**
* Send both current and deprecated events in order to preserve backwards compatibility.
* Must not throw {@link Exception}s.
*/
private void connectionPoolCreated(final ConnectionPoolListener connectionPoolListener, final ServerId serverId,
final ConnectionPoolSettings settings) {
ClusterId clusterId = serverId.getClusterId();
if (requiresLogging(clusterId)) {
String message = "Connection pool created for {}:{} using options maxIdleTimeMS={}, minPoolSize={}, "
+ "maxPoolSize={}, maxConnecting={}, waitQueueTimeoutMS={}";
List entries = createBasicEntries();
entries.add(new LogMessage.Entry(MAX_IDLE_TIME_MS, settings.getMaxConnectionIdleTime(MILLISECONDS)));
entries.add(new LogMessage.Entry(MIN_POOL_SIZE, settings.getMinSize()));
entries.add(new LogMessage.Entry(MAX_POOL_SIZE, settings.getMaxSize()));
entries.add(new LogMessage.Entry(MAX_CONNECTING, settings.getMaxConnecting()));
entries.add(new LogMessage.Entry(WAIT_QUEUE_TIMEOUT_MS, settings.getMaxWaitTime(MILLISECONDS)));
logMessage("Connection pool created", clusterId, message, entries);
}
connectionPoolListener.connectionPoolCreated(new ConnectionPoolCreatedEvent(serverId, settings));
}
/**
* Send both current and deprecated events in order to preserve backwards compatibility.
* Must not throw {@link Exception}s.
*
* @return A {@link TimePoint} before executing {@link ConnectionPoolListener#connectionCreated(ConnectionCreatedEvent)}
* and logging the event. This order is required by
* CMAP
* and {@link ConnectionReadyEvent#getElapsedTime(TimeUnit)}.
*/
private TimePoint connectionCreated(final ConnectionPoolListener connectionPoolListener, final ConnectionId connectionId) {
TimePoint openStart = TimePoint.now();
logEventMessage("Connection created",
"Connection created: address={}:{}, driver-generated ID={}",
connectionId.getLocalValue());
connectionPoolListener.connectionCreated(new ConnectionCreatedEvent(connectionId));
return openStart;
}
/**
* Send both current and deprecated events in order to preserve backwards compatibility.
* Must not throw {@link Exception}s.
*/
private void connectionClosed(final ConnectionPoolListener connectionPoolListener, final ConnectionId connectionId,
final ConnectionClosedEvent.Reason reason) {
ClusterId clusterId = serverId.getClusterId();
if (requiresLogging(clusterId)) {
String errorReason = "There was a socket exception raised by this connection";
List entries = createBasicEntries();
entries.add(new LogMessage.Entry(DRIVER_CONNECTION_ID, connectionId.getLocalValue()));
entries.add(new LogMessage.Entry(REASON_DESCRIPTION, EventReasonMessageResolver.getMessage(reason)));
entries.add(new LogMessage.Entry(ERROR_DESCRIPTION, reason == ERROR ? errorReason : null));
logMessage("Connection closed",
clusterId,
"Connection closed: address={}:{}, driver-generated ID={}. Reason: {}.[ Error: {}]",
entries);
}
connectionPoolListener.connectionClosed(new ConnectionClosedEvent(connectionId, reason));
}
private void connectionCheckedOut(
final OperationContext operationContext,
final PooledConnection connection,
final TimePoint checkoutStart) {
Duration checkoutDuration = checkoutStart.elapsed();
ConnectionId connectionId = getId(connection);
ClusterId clusterId = serverId.getClusterId();
if (requiresLogging(clusterId)) {
List entries = createBasicEntries();
entries.add(new LogMessage.Entry(DRIVER_CONNECTION_ID, connectionId.getLocalValue()));
entries.add(new LogMessage.Entry(DURATION_MS, checkoutDuration.toMillis()));
logMessage("Connection checked out", clusterId,
"Connection checked out: address={}:{}, driver-generated ID={}, duration={} ms", entries);
}
connectionPoolListener.connectionCheckedOut(
new ConnectionCheckedOutEvent(connectionId, operationContext.getId(), checkoutDuration.toNanos()));
}
/**
* @return A {@link TimePoint} before executing
* {@link ConnectionPoolListener#connectionCheckOutStarted(ConnectionCheckOutStartedEvent)} and logging the event.
* This order is required by
* CMAP
* and {@link ConnectionCheckedOutEvent#getElapsedTime(TimeUnit)}, {@link ConnectionCheckOutFailedEvent#getElapsedTime(TimeUnit)}.
*/
private TimePoint connectionCheckoutStarted(final OperationContext operationContext) {
TimePoint checkoutStart = TimePoint.now();
logEventMessage("Connection checkout started", "Checkout started for connection to {}:{}");
connectionPoolListener.connectionCheckOutStarted(new ConnectionCheckOutStartedEvent(serverId, operationContext.getId()));
return checkoutStart;
}
/**
* Must not throw {@link Exception}s.
*/
private ConnectionId getId(final InternalConnection internalConnection) {
return internalConnection.getDescription().getConnectionId();
}
private boolean isLoadBalanced() {
return !sdamProvider.optional().isPresent();
}
/**
* @return {@link ConnectionPoolSettings#getMaxSize()} if it is not 0, otherwise returns {@link ConcurrentPool#INFINITE_SIZE}.
*/
private static int maxSize(final ConnectionPoolSettings settings) {
return settings.getMaxSize() == 0 ? INFINITE_SIZE : settings.getMaxSize();
}
private class PooledConnection implements InternalConnection {
private final UsageTrackingInternalConnection wrapped;
private final AtomicBoolean isClosed = new AtomicBoolean();
private Connection.PinningMode pinningMode;
private OperationContext operationContext;
PooledConnection(final UsageTrackingInternalConnection wrapped) {
this.wrapped = notNull("wrapped", wrapped);
}
@Override
public int getGeneration() {
return wrapped.getGeneration();
}
/**
* Associates this with the operation context and establishes the checked out start time
*/
public void checkedOutForOperation(final OperationContext operationContext) {
this.operationContext = operationContext;
}
@Override
public void open() {
assertFalse(isClosed.get());
TimePoint openStart;
try {
openStart = connectionCreated(connectionPoolListener, wrapped.getDescription().getConnectionId());
wrapped.open();
} catch (Exception e) {
closeAndHandleOpenFailure();
throw new MongoOpenConnectionInternalException(e);
}
handleOpenSuccess(openStart);
}
@Override
public void openAsync(final SingleResultCallback callback) {
assertFalse(isClosed.get());
TimePoint openStart = connectionCreated(connectionPoolListener, wrapped.getDescription().getConnectionId());
wrapped.openAsync((nullResult, failure) -> {
if (failure != null) {
closeAndHandleOpenFailure();
callback.onResult(null, new MongoOpenConnectionInternalException(failure));
} else {
handleOpenSuccess(openStart);
callback.onResult(nullResult, null);
}
});
}
@Override
public void close() {
// All but the first call is a no-op
if (!isClosed.getAndSet(true)) {
unmarkAsPinned();
connectionCheckedIn();
if (wrapped.isClosed() || shouldPrune(wrapped)) {
pool.release(wrapped, true);
} else {
openConcurrencyLimiter.tryHandOverOrRelease(wrapped);
}
}
}
private void connectionCheckedIn() {
ConnectionId connectionId = getId(wrapped);
logEventMessage("Connection checked in",
"Connection checked in: address={}:{}, driver-generated ID={}",
connectionId.getLocalValue());
connectionPoolListener.connectionCheckedIn(new ConnectionCheckedInEvent(connectionId, operationContext.getId()));
}
void release() {
if (!isClosed.getAndSet(true)) {
pool.release(wrapped);
}
}
/**
* {@linkplain ConcurrentPool#release(Object, boolean) Prune} this connection without sending a {@link ConnectionClosedEvent}.
* This method must be used if and only if {@link ConnectionCreatedEvent} was not sent for the connection.
* Must not throw {@link Exception}s.
*/
void closeSilently() {
if (!isClosed.getAndSet(true)) {
wrapped.setCloseSilently();
pool.release(wrapped, true);
}
}
/**
* Must not throw {@link Exception}s.
*/
private void closeAndHandleOpenFailure() {
if (!isClosed.getAndSet(true)) {
if (wrapped.getDescription().getServiceId() != null) {
invalidate(assertNotNull(wrapped.getDescription().getServiceId()), wrapped.getGeneration());
}
pool.release(wrapped, true);
}
}
/**
* Must not throw {@link Exception}s.
*/
private void handleOpenSuccess(final TimePoint openStart) {
Duration openDuration = openStart.elapsed();
ConnectionId connectionId = getId(this);
ClusterId clusterId = serverId.getClusterId();
if (requiresLogging(clusterId)) {
List entries = createBasicEntries();
entries.add(new LogMessage.Entry(DRIVER_CONNECTION_ID, connectionId.getLocalValue()));
entries.add(new LogMessage.Entry(DURATION_MS, openDuration.toMillis()));
logMessage("Connection ready", clusterId, "Connection ready: address={}:{}, driver-generated ID={}, established in={} ms", entries);
}
connectionPoolListener.connectionReady(new ConnectionReadyEvent(connectionId, openDuration.toNanos()));
}
@Override
public boolean opened() {
isTrue("open", !isClosed.get());
return wrapped.opened();
}
@Override
public boolean isClosed() {
return isClosed.get() || wrapped.isClosed();
}
@Override
public ByteBuf getBuffer(final int capacity) {
return wrapped.getBuffer(capacity);
}
@Override
public void sendMessage(final List byteBuffers, final int lastRequestId) {
isTrue("open", !isClosed.get());
wrapped.sendMessage(byteBuffers, lastRequestId);
}
@Override
public T sendAndReceive(final CommandMessage message, final Decoder decoder, final SessionContext sessionContext,
final RequestContext requestContext, final OperationContext operationContext) {
isTrue("open", !isClosed.get());
return wrapped.sendAndReceive(message, decoder, sessionContext, requestContext, operationContext);
}
@Override
public void send(final CommandMessage message, final Decoder decoder, final SessionContext sessionContext) {
isTrue("open", !isClosed.get());
wrapped.send(message, decoder, sessionContext);
}
@Override
public T receive(final Decoder decoder, final SessionContext sessionContext) {
isTrue("open", !isClosed.get());
return wrapped.receive(decoder, sessionContext);
}
@Override
public T receive(final Decoder decoder, final SessionContext sessionContext, final int additionalTimeout) {
isTrue("open", !isClosed.get());
return wrapped.receive(decoder, sessionContext, additionalTimeout);
}
@Override
public boolean hasMoreToCome() {
isTrue("open", !isClosed.get());
return wrapped.hasMoreToCome();
}
@Override
public void sendAndReceiveAsync(final CommandMessage message, final Decoder decoder, final SessionContext sessionContext,
final RequestContext requestContext, final OperationContext operationContext, final SingleResultCallback callback) {
isTrue("open", !isClosed.get());
wrapped.sendAndReceiveAsync(message, decoder, sessionContext, requestContext, operationContext, (result, t) -> callback.onResult(result, t));
}
@Override
public ResponseBuffers receiveMessage(final int responseTo) {
isTrue("open", !isClosed.get());
return wrapped.receiveMessage(responseTo);
}
@Override
public void sendMessageAsync(final List byteBuffers, final int lastRequestId, final SingleResultCallback callback) {
isTrue("open", !isClosed.get());
wrapped.sendMessageAsync(byteBuffers, lastRequestId, (result, t) -> callback.onResult(null, t));
}
@Override
public void receiveMessageAsync(final int responseTo, final SingleResultCallback callback) {
isTrue("open", !isClosed.get());
wrapped.receiveMessageAsync(responseTo, (result, t) -> callback.onResult(result, t));
}
@Override
public void markAsPinned(final Connection.PinningMode pinningMode) {
assertNotNull(pinningMode);
// if the connection is already pinned for some other mode, the additional mode can be ignored.
// The typical case is the connection is first pinned for a transaction, then pinned for a cursor withing that transaction
// In this case, the cursor pinning is subsumed by the transaction pinning.
if (this.pinningMode == null) {
this.pinningMode = pinningMode;
pinnedStatsManager.increment(pinningMode);
}
}
void unmarkAsPinned() {
if (pinningMode != null) {
pinnedStatsManager.decrement(pinningMode);
}
}
@Override
public ConnectionDescription getDescription() {
return wrapped.getDescription();
}
@Override
public ServerDescription getInitialServerDescription() {
isTrue("open", !isClosed.get());
return wrapped.getInitialServerDescription();
}
}
/**
* This internal exception is used to express an exceptional situation encountered when opening a connection.
* It exists because it allows consolidating the code that sends events for exceptional situations in a
* {@linkplain #checkOutFailed(Throwable, OperationContext, TimePoint) single place}, it must not be observable by an external code.
*/
private static final class MongoOpenConnectionInternalException extends RuntimeException {
private static final long serialVersionUID = 1;
MongoOpenConnectionInternalException(@NonNull final Throwable cause) {
super(cause);
}
@Override
@NonNull
public Throwable getCause() {
return assertNotNull(super.getCause());
}
}
private class UsageTrackingInternalConnectionItemFactory implements ConcurrentPool.ItemFactory {
private final InternalConnectionFactory internalConnectionFactory;
UsageTrackingInternalConnectionItemFactory(final InternalConnectionFactory internalConnectionFactory) {
this.internalConnectionFactory = internalConnectionFactory;
}
@Override
public UsageTrackingInternalConnection create() {
return new UsageTrackingInternalConnection(internalConnectionFactory.create(serverId, connectionGenerationSupplier),
serviceStateManager);
}
@Override
public void close(final UsageTrackingInternalConnection connection) {
if (!connection.isCloseSilently()) {
connectionClosed(connectionPoolListener, getId(connection), getReasonForClosing(connection));
}
connection.close();
}
private ConnectionClosedEvent.Reason getReasonForClosing(final UsageTrackingInternalConnection connection) {
ConnectionClosedEvent.Reason reason;
if (connection.isClosed()) {
reason = ConnectionClosedEvent.Reason.ERROR;
} else if (fromPreviousGeneration(connection)) {
reason = ConnectionClosedEvent.Reason.STALE;
} else if (pastMaxIdleTime(connection)) {
reason = ConnectionClosedEvent.Reason.IDLE;
} else {
reason = ConnectionClosedEvent.Reason.POOL_CLOSED;
}
return reason;
}
@Override
public boolean shouldPrune(final UsageTrackingInternalConnection usageTrackingConnection) {
return DefaultConnectionPool.this.shouldPrune(usageTrackingConnection);
}
}
/**
* Package-access methods are thread-safe,
* and only they should be called outside of the {@link OpenConcurrencyLimiter}'s code.
*/
@ThreadSafe
private final class OpenConcurrencyLimiter {
private final ReentrantLock lock;
private final Condition permitAvailableOrHandedOverOrClosedOrPausedCondition;
private final int maxPermits;
private int permits;
private final Deque> desiredConnectionSlots;
OpenConcurrencyLimiter(final int maxConnecting) {
lock = new ReentrantLock(true);
permitAvailableOrHandedOverOrClosedOrPausedCondition = lock.newCondition();
maxPermits = maxConnecting;
permits = maxPermits;
desiredConnectionSlots = new LinkedList<>();
}
PooledConnection openOrGetAvailable(final PooledConnection connection, final Timeout timeout) throws MongoTimeoutException {
PooledConnection result = openWithConcurrencyLimit(connection, OpenWithConcurrencyLimitMode.TRY_GET_AVAILABLE, timeout);
return assertNotNull(result);
}
void openImmediatelyAndTryHandOverOrRelease(final PooledConnection connection) throws MongoTimeoutException {
assertNull(openWithConcurrencyLimit(connection, OpenWithConcurrencyLimitMode.TRY_HAND_OVER_OR_RELEASE, Timeout.immediate()));
}
/**
* This method can be thought of as operating in two phases.
* In the first phase it tries to synchronously acquire a permit to open the {@code connection}
* or get a different {@linkplain PooledConnection#opened() opened} connection if {@code mode} is
* {@link OpenWithConcurrencyLimitMode#TRY_GET_AVAILABLE} and one becomes available while waiting for a permit.
* The first phase has one of the following outcomes:
*
* - A {@link MongoTimeoutException} or a different {@link Exception} is thrown,
* and the specified {@code connection} is {@linkplain PooledConnection#closeSilently() silently closed}.
* - An opened connection different from the specified one is returned,
* and the specified {@code connection} is {@linkplain PooledConnection#closeSilently() silently closed}.
* This outcome is possible only if {@code mode} is {@link OpenWithConcurrencyLimitMode#TRY_GET_AVAILABLE}.
* - A permit is acquired, {@link #connectionCreated(ConnectionPoolListener, ConnectionId)} is reported
* and an attempt to open the specified {@code connection} is made. This is the second phase in which
* the {@code connection} is {@linkplain PooledConnection#open() opened synchronously}.
* The attempt to open the {@code connection} has one of the following outcomes
* combined with releasing the acquired permit:
*
* - An {@link Exception} is thrown
* and the {@code connection} is {@linkplain PooledConnection#closeAndHandleOpenFailure() closed}.
* - Else if the specified {@code connection} is opened successfully and
* {@code mode} is {@link OpenWithConcurrencyLimitMode#TRY_HAND_OVER_OR_RELEASE},
* then {@link #tryHandOverOrRelease(UsageTrackingInternalConnection)} is called and {@code null} is returned.
* - Else the specified {@code connection}, which is now opened, is returned.
*
*
*
*
* @param timeout Applies only to the first phase.
* @return An {@linkplain PooledConnection#opened() opened} connection which is
* either the specified {@code connection},
* or potentially a different one if {@code mode} is {@link OpenWithConcurrencyLimitMode#TRY_GET_AVAILABLE},
* or {@code null} if {@code mode} is {@link OpenWithConcurrencyLimitMode#TRY_HAND_OVER_OR_RELEASE}.
* @throws MongoTimeoutException If the first phase timed out.
*/
@Nullable
private PooledConnection openWithConcurrencyLimit(final PooledConnection connection, final OpenWithConcurrencyLimitMode mode,
final Timeout timeout) throws MongoTimeoutException {
PooledConnection availableConnection;
try {//phase one
availableConnection = acquirePermitOrGetAvailableOpenedConnection(
mode == OpenWithConcurrencyLimitMode.TRY_GET_AVAILABLE, timeout);
} catch (Exception e) {
connection.closeSilently();
throw e;
}
if (availableConnection != null) {
connection.closeSilently();
return availableConnection;
} else {//acquired a permit, phase two
try {
connection.open();
if (mode == OpenWithConcurrencyLimitMode.TRY_HAND_OVER_OR_RELEASE) {
tryHandOverOrRelease(connection.wrapped);
return null;
} else {
return connection;
}
} finally {
releasePermit();
}
}
}
/**
* This method is similar to {@link #openWithConcurrencyLimit(PooledConnection, OpenWithConcurrencyLimitMode, Timeout)}
* with the following differences:
*
* - It does not have the {@code mode} parameter and acts as if this parameter were
* {@link OpenWithConcurrencyLimitMode#TRY_GET_AVAILABLE}.
* - While the first phase is still synchronous, the {@code connection} is
* {@linkplain PooledConnection#openAsync(SingleResultCallback) opened asynchronously} in the second phase.
* - Instead of returning a result or throwing an exception via Java {@code return}/{@code throw} statements,
* it calls {@code callback.}{@link SingleResultCallback#onResult(Object, Throwable) onResult(result, failure)}
* and passes either a {@link PooledConnection} or an {@link Exception}.
*
*/
void openAsyncWithConcurrencyLimit(
final PooledConnection connection, final Timeout timeout, final SingleResultCallback callback) {
PooledConnection availableConnection;
try {//phase one
availableConnection = acquirePermitOrGetAvailableOpenedConnection(true, timeout);
} catch (Exception e) {
connection.closeSilently();
callback.onResult(null, e);
return;
}
if (availableConnection != null) {
connection.closeSilently();
callback.onResult(availableConnection, null);
} else {//acquired a permit, phase two
connection.openAsync((nullResult, failure) -> {
releasePermit();
if (failure != null) {
callback.onResult(null, failure);
} else {
callback.onResult(connection, null);
}
});
}
}
/**
* @return Either {@code null} if a permit has been acquired, or a {@link PooledConnection}
* if {@code tryGetAvailable} is {@code true} and an {@linkplain PooledConnection#opened() opened} one becomes available while
* waiting for a permit.
* @throws MongoTimeoutException If timed out.
* @throws MongoInterruptedException If the current thread has its {@linkplain Thread#interrupted() interrupted status}
* set on entry to this method or is interrupted while waiting to get an available opened connection.
*/
@Nullable
private PooledConnection acquirePermitOrGetAvailableOpenedConnection(final boolean tryGetAvailable, final Timeout timeout)
throws MongoTimeoutException, MongoInterruptedException {
PooledConnection availableConnection = null;
boolean expressedDesireToGetAvailableConnection = false;
lockInterruptibly(lock);
try {
if (tryGetAvailable) {
/* An attempt to get an available opened connection from the pool (must be done while holding the lock)
* happens here at most once to prevent the race condition in the following execution
* (actions are specified in the execution total order,
* which by definition exists if an execution is either sequentially consistent or linearizable):
* 1. Thread#1 starts checking out and gets a non-opened connection.
* 2. Thread#2 checks in a connection. Tries to hand it over, but there are no threads desiring to get one.
* 3. Thread#1 executes the current code. Expresses the desire to get a connection via the hand-over mechanism,
* but thread#2 has already tried handing over and released its connection to the pool.
* As a result, thread#1 is waiting for a permit to open a connection despite one being available in the pool.
*
* This attempt should be unfair because the current thread (Thread#1) has already waited for its turn fairly.
* Waiting fairly again puts the current thread behind other threads, which is unfair to the current thread. */
availableConnection = getPooledConnectionImmediateUnfair();
if (availableConnection != null) {
return availableConnection;
}
expressDesireToGetAvailableConnection();
expressedDesireToGetAvailableConnection = true;
}
long remainingNanos = timeout.remainingOrInfinite(NANOSECONDS);
while (permits == 0
// the absence of short-circuiting is of importance
& !stateAndGeneration.throwIfClosedOrPaused()
& (availableConnection = tryGetAvailable ? tryGetAvailableConnection() : null) == null) {
if (Timeout.expired(remainingNanos)) {
throw createTimeoutException(timeout);
}
remainingNanos = awaitNanos(permitAvailableOrHandedOverOrClosedOrPausedCondition, remainingNanos);
}
if (availableConnection == null) {
assertTrue(permits > 0);
permits--;
}
return availableConnection;
} finally {
try {
if (expressedDesireToGetAvailableConnection && availableConnection == null) {
giveUpOnTryingToGetAvailableConnection();
}
} finally {
lock.unlock();
}
}
}
private void releasePermit() {
withUnfairLock(lock, () -> {
assertTrue(permits < maxPermits);
permits++;
permitAvailableOrHandedOverOrClosedOrPausedCondition.signal();
});
}
private void expressDesireToGetAvailableConnection() {
desiredConnectionSlots.addLast(new MutableReference<>());
}
@Nullable
private PooledConnection tryGetAvailableConnection() {
assertFalse(desiredConnectionSlots.isEmpty());
PooledConnection result = desiredConnectionSlots.peekFirst().reference;
if (result != null) {
desiredConnectionSlots.removeFirst();
assertTrue(result.opened());
}
return result;
}
private void giveUpOnTryingToGetAvailableConnection() {
assertFalse(desiredConnectionSlots.isEmpty());
PooledConnection connection = desiredConnectionSlots.removeLast().reference;
if (connection != null) {
connection.release();
}
}
/**
* The hand-over mechanism is needed to prevent other threads doing checkout from stealing newly released connections
* from threads that are waiting for a permit to open a connection.
*/
void tryHandOverOrRelease(final UsageTrackingInternalConnection openConnection) {
boolean handedOver = withUnfairLock(lock, () -> {
for (//iterate from first (head) to last (tail)
MutableReference desiredConnectionSlot : desiredConnectionSlots) {
if (desiredConnectionSlot.reference == null) {
desiredConnectionSlot.reference = new PooledConnection(openConnection);
permitAvailableOrHandedOverOrClosedOrPausedCondition.signal();
return true;
}
}
return false;
});
if (!handedOver) {
pool.release(openConnection);
}
}
void signalClosedOrPaused() {
withUnfairLock(lock, permitAvailableOrHandedOverOrClosedOrPausedCondition::signalAll);
}
/**
* @param timeoutNanos See {@link Timeout#started(long, TimePoint)}.
* @return The remaining duration as per {@link Timeout#remainingOrInfinite(TimeUnit)} if waiting ended early either
* spuriously or because of receiving a signal.
*/
private long awaitNanos(final Condition condition, final long timeoutNanos) throws MongoInterruptedException {
try {
if (timeoutNanos < 0 || timeoutNanos == Long.MAX_VALUE) {
condition.await();
return -1;
} else {
return Math.max(0, condition.awaitNanos(timeoutNanos));
}
} catch (InterruptedException e) {
throw interruptAndCreateMongoInterruptedException(null, e);
}
}
}
/**
* @see OpenConcurrencyLimiter#openWithConcurrencyLimit(PooledConnection, OpenWithConcurrencyLimitMode, Timeout)
*/
private enum OpenWithConcurrencyLimitMode {
TRY_GET_AVAILABLE,
TRY_HAND_OVER_OR_RELEASE
}
@NotThreadSafe
private static final class MutableReference {
@Nullable
private T reference;
private MutableReference() {
}
}
@ThreadSafe
static final class ServiceStateManager {
private final ConcurrentHashMap stateByServiceId = new ConcurrentHashMap<>();
void addConnection(final ObjectId serviceId) {
stateByServiceId.compute(serviceId, (k, v) -> {
if (v == null) {
v = new ServiceState();
}
v.incrementConnectionCount();
return v;
});
}
/**
* Removes the mapping from {@code serviceId} to a {@link ServiceState} if its connection count reaches 0.
* This is done to prevent memory leaks.
*
* This method must be called once for any connection for which {@link #addConnection(ObjectId)} was called.
*/
void removeConnection(final ObjectId serviceId) {
stateByServiceId.compute(serviceId, (k, v) -> {
assertNotNull(v);
return v.decrementAndGetConnectionCount() == 0 ? null : v;
});
}
/**
* In some cases we may increment the generation even for an unregistered serviceId, as when open fails on the only connection to
* a given serviceId. In this case this method does not track the generation increment but does return true.
*
* @return true if the generation was incremented
*/
boolean incrementGeneration(final ObjectId serviceId, final int expectedGeneration) {
ServiceState state = stateByServiceId.get(serviceId);
return state == null || state.incrementGeneration(expectedGeneration);
}
int getGeneration(final ObjectId serviceId) {
ServiceState state = stateByServiceId.get(serviceId);
return state == null ? 0 : state.getGeneration();
}
private static final class ServiceState {
private final AtomicInteger generation = new AtomicInteger();
private final AtomicInteger connectionCount = new AtomicInteger();
void incrementConnectionCount() {
connectionCount.incrementAndGet();
}
int decrementAndGetConnectionCount() {
return connectionCount.decrementAndGet();
}
boolean incrementGeneration(final int expectedGeneration) {
return generation.compareAndSet(expectedGeneration, expectedGeneration + 1);
}
public int getGeneration() {
return generation.get();
}
}
}
private static final class PinnedStatsManager {
private final LongAdder numPinnedToCursor = new LongAdder();
private final LongAdder numPinnedToTransaction = new LongAdder();
void increment(final Connection.PinningMode pinningMode) {
switch (pinningMode) {
case CURSOR:
numPinnedToCursor.increment();
break;
case TRANSACTION:
numPinnedToTransaction.increment();
break;
default:
fail();
}
}
void decrement(final Connection.PinningMode pinningMode) {
switch (pinningMode) {
case CURSOR:
numPinnedToCursor.decrement();
break;
case TRANSACTION:
numPinnedToTransaction.decrement();
break;
default:
fail();
}
}
int getNumPinnedToCursor() {
return numPinnedToCursor.intValue();
}
int getNumPinnedToTransaction() {
return numPinnedToTransaction.intValue();
}
}
/**
* This class maintains threads needed to perform {@link ConnectionPool#getAsync(OperationContext, SingleResultCallback)}.
*/
@ThreadSafe
private static class AsyncWorkManager implements AutoCloseable {
private volatile State state;
private volatile BlockingQueue tasks;
private final Lock lock;
@Nullable
private ExecutorService worker;
AsyncWorkManager(final boolean prestart) {
state = State.NEW;
tasks = new LinkedBlockingQueue<>();
lock = new StampedLock().asWriteLock();
if (prestart) {
assertTrue(initUnlessClosed());
}
}
void enqueue(final Task task) {
boolean closed = withLock(lock, () -> {
if (initUnlessClosed()) {
tasks.add(task);
return false;
}
return true;
});
if (closed) {
task.failAsClosed();
}
}
/**
* Invocations of this method must be guarded by {@link #lock}, unless done from the constructor.
*
* @return {@code false} iff the {@link #state} is {@link State#CLOSED}.
*/
private boolean initUnlessClosed() {
boolean result = true;
if (state == State.NEW) {
worker = Executors.newSingleThreadExecutor(new DaemonThreadFactory("AsyncGetter"));
worker.submit(() -> runAndLogUncaught(this::workerRun));
state = State.INITIALIZED;
} else if (state == State.CLOSED) {
result = false;
}
return result;
}
/**
* {@linkplain Thread#interrupt() Interrupts} all workers and causes queued tasks to
* {@linkplain Task#failAsClosed() fail} asynchronously.
*/
@Override
@SuppressWarnings("try")
public void close() {
withLock(lock, () -> {
if (state != State.CLOSED) {
state = State.CLOSED;
if (worker != null) {
worker.shutdownNow(); // at this point we interrupt `worker`s thread
}
}
});
}
private void workerRun() {
while (state != State.CLOSED) {
try {
Task task = tasks.take();
if (task.timeout().expired()) {
task.failAsTimedOut();
} else {
task.execute();
}
} catch (InterruptedException closed) {
// fail the rest of the tasks and stop
} catch (Exception e) {
LOGGER.error(null, e);
}
}
failAllTasksAfterClosing();
}
private void failAllTasksAfterClosing() {
Queue localGets = withLock(lock, () -> {
assertTrue(state == State.CLOSED);
// at this point it is guaranteed that no thread enqueues a task
Queue result = tasks;
if (!tasks.isEmpty()) {
tasks = new LinkedBlockingQueue<>();
}
return result;
});
localGets.forEach(Task::failAsClosed);
localGets.clear();
}
private void runAndLogUncaught(final Runnable runnable) {
try {
runnable.run();
} catch (Throwable t) {
LOGGER.error("The pool is not going to work correctly from now on. You may want to recreate the MongoClient", t);
throw t;
}
}
private enum State {
NEW,
INITIALIZED,
CLOSED
}
}
/**
* An action that is allowed to be completed (failed or executed) at most once, and a timeout associated with it.
*/
@NotThreadSafe
final class Task {
private final Timeout timeout;
private final Consumer action;
private boolean completed;
Task(final Timeout timeout, final Consumer action) {
this.timeout = timeout;
this.action = action;
}
void execute() {
doComplete(() -> null);
}
void failAsClosed() {
doComplete(pool::poolClosedException);
}
void failAsTimedOut() {
doComplete(() -> createTimeoutException(timeout));
}
private void doComplete(final Supplier failureSupplier) {
assertFalse(completed);
completed = true;
action.accept(failureSupplier.get());
}
Timeout timeout() {
return timeout;
}
}
/**
* Methods {@link #start()} and {@link #runOnceAndStop()} must be called sequentially. Each {@link #start()} must be followed by
* {@link #runOnceAndStop()} unless {@link BackgroundMaintenanceManager} is {@linkplain #close() closed}.
*
* This class implements
*
* CMAP background thread.
*/
@NotThreadSafe
private final class BackgroundMaintenanceManager implements AutoCloseable {
@Nullable
private final ScheduledExecutorService maintainer;
@Nullable
private Future> cancellationHandle;
private boolean initialStart;
private BackgroundMaintenanceManager() {
maintainer = settings.getMaintenanceInitialDelay(NANOSECONDS) < Long.MAX_VALUE
? Executors.newSingleThreadScheduledExecutor(new DaemonThreadFactory("MaintenanceTimer"))
: null;
cancellationHandle = null;
initialStart = true;
}
void start() {
if (maintainer != null) {
assertTrue(cancellationHandle == null);
cancellationHandle = ignoreRejectedExectution(() -> maintainer.scheduleAtFixedRate(
DefaultConnectionPool.this::doMaintenance,
initialStart ? settings.getMaintenanceInitialDelay(MILLISECONDS) : 0,
settings.getMaintenanceFrequency(MILLISECONDS), MILLISECONDS));
initialStart = false;
}
}
void runOnceAndStop() {
if (maintainer != null) {
if (cancellationHandle != null) {
cancellationHandle.cancel(false);
cancellationHandle = null;
}
ignoreRejectedExectution(() -> maintainer.execute(DefaultConnectionPool.this::doMaintenance));
}
}
@Override
public void close() {
if (maintainer != null) {
maintainer.shutdownNow();
}
}
private void ignoreRejectedExectution(final Runnable action) {
ignoreRejectedExectution(() -> {
action.run();
return null;
});
}
@Nullable
private T ignoreRejectedExectution(final Supplier action) {
try {
return action.get();
} catch (RejectedExecutionException ignored) {
// `close` either completed or is in progress
return null;
}
}
}
@ThreadSafe
private final class StateAndGeneration {
private final ReadWriteLock lock;
private volatile boolean paused;
private final AtomicBoolean closed;
private volatile int generation;
@Nullable
private Throwable cause;
StateAndGeneration() {
lock = new StampedLock().asReadWriteLock();
paused = true;
closed = new AtomicBoolean();
generation = 0;
cause = null;
}
int generation() {
return generation;
}
/**
* @return {@code true} if and only if the state changed from ready to paused as a result of the operation.
* The generation is incremented regardless of the returned value.
*/
boolean pauseAndIncrementGeneration(@Nullable final Throwable cause) {
return withLock(lock.writeLock(), () -> {
boolean result = false;
if (!paused) {
paused = true;
pool.pause(() -> new MongoConnectionPoolClearedException(serverId, cause));
result = true;
}
this.cause = cause;
//noinspection NonAtomicOperationOnVolatileField
generation++;
if (result) {
logEventMessage("Connection pool cleared", "Connection pool for {}:{} cleared");
connectionPoolListener.connectionPoolCleared(new ConnectionPoolClearedEvent(serverId));
// one additional run is required to guarantee that a paused pool releases resources
backgroundMaintenance.runOnceAndStop();
}
return result;
});
}
boolean ready() {
boolean result = false;
if (paused) {
result = withLock(lock.writeLock(), () -> {
if (paused) {
paused = false;
cause = null;
pool.ready();
logEventMessage("Connection pool ready", "Connection pool ready for {}:{}");
connectionPoolListener.connectionPoolReady(new ConnectionPoolReadyEvent(serverId));
backgroundMaintenance.start();
return true;
}
return false;
});
}
return result;
}
/**
* @return {@code true} if and only if the state changed as a result of the operation.
*/
boolean close() {
return closed.compareAndSet(false, true);
}
/**
* @return {@code false} which means that the method did not throw.
* The method returns to allow using it conveniently as part of a condition check when waiting on a {@link Condition}.
* Short-circuiting operators {@code &&} and {@code ||} must not be used with this method to ensure that it is called.
* @throws MongoServerUnavailableException If and only if {@linkplain #close() closed}.
* @throws MongoConnectionPoolClearedException If and only if {@linkplain #pauseAndIncrementGeneration(Throwable) paused}
* and not {@linkplain #close() closed}.
*/
boolean throwIfClosedOrPaused() {
if (closed.get()) {
throw pool.poolClosedException();
}
if (paused) {
withLock(lock.readLock(), () -> {
if (paused) {
throw new MongoConnectionPoolClearedException(serverId, cause);
}
});
}
return false;
}
}
private void logEventMessage(final String messageId, final String format, final long driverConnectionId) {
ClusterId clusterId = serverId.getClusterId();
if (requiresLogging(clusterId)) {
List entries = createBasicEntries();
entries.add(new LogMessage.Entry(DRIVER_CONNECTION_ID, driverConnectionId));
logMessage(messageId, clusterId, format, entries);
}
}
private void logEventMessage(final String messageId, final String format) {
ClusterId clusterId = serverId.getClusterId();
if (requiresLogging(clusterId)) {
List entries = createBasicEntries();
logMessage(messageId, clusterId, format, entries);
}
}
private List createBasicEntries() {
List entries = new ArrayList<>();
entries.add(new LogMessage.Entry(SERVER_HOST, serverId.getAddress().getHost()));
entries.add(new LogMessage.Entry(SERVER_PORT, serverId.getAddress().getPort()));
return entries;
}
private static void logMessage(final String messageId, final ClusterId clusterId, final String format, final List entries) {
STRUCTURED_LOGGER.log(new LogMessage(CONNECTION, DEBUG, messageId, clusterId, entries, format));
}
private static boolean requiresLogging(final ClusterId clusterId) {
return STRUCTURED_LOGGER.isRequired(DEBUG, clusterId);
}
}