All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.springframework.data.mongodb.MongoTransactionManager Maven / Gradle / Ivy

There is a newer version: 4.4.1
Show newest version
/*
 * Copyright 2018-2024 the original author or authors.
 *
 * 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
 *
 *      https://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.springframework.data.mongodb;

import org.springframework.beans.factory.InitializingBean;
import org.springframework.lang.Nullable;
import org.springframework.transaction.TransactionDefinition;
import org.springframework.transaction.TransactionException;
import org.springframework.transaction.TransactionSystemException;
import org.springframework.transaction.support.AbstractPlatformTransactionManager;
import org.springframework.transaction.support.DefaultTransactionStatus;
import org.springframework.transaction.support.ResourceTransactionManager;
import org.springframework.transaction.support.SmartTransactionObject;
import org.springframework.transaction.support.TransactionSynchronizationManager;
import org.springframework.transaction.support.TransactionSynchronizationUtils;
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils;

import com.mongodb.ClientSessionOptions;
import com.mongodb.MongoException;
import com.mongodb.TransactionOptions;
import com.mongodb.client.ClientSession;

/**
 * A {@link org.springframework.transaction.PlatformTransactionManager} implementation that manages
 * {@link ClientSession} based transactions for a single {@link MongoDatabaseFactory}.
 * 
* Binds a {@link ClientSession} from the specified {@link MongoDatabaseFactory} to the thread. *
* {@link TransactionDefinition#isReadOnly() Readonly} transactions operate on a {@link ClientSession} and enable causal * consistency, and also {@link ClientSession#startTransaction() start}, {@link ClientSession#commitTransaction() * commit} or {@link ClientSession#abortTransaction() abort} a transaction. *
* Application code is required to retrieve the {@link com.mongodb.client.MongoDatabase} via * {@link MongoDatabaseUtils#getDatabase(MongoDatabaseFactory)} instead of a standard * {@link MongoDatabaseFactory#getMongoDatabase()} call. Spring classes such as * {@link org.springframework.data.mongodb.core.MongoTemplate} use this strategy implicitly. *
* By default failure of a {@literal commit} operation raises a {@link TransactionSystemException}. One may override * {@link #doCommit(MongoTransactionObject)} to implement the * Retry Commit Operation * behavior as outlined in the MongoDB reference manual. * * @author Christoph Strobl * @author Mark Paluch * @currentRead Shadow's Edge - Brent Weeks * @since 2.1 * @see MongoDB Transaction Documentation * @see MongoDatabaseUtils#getDatabase(MongoDatabaseFactory, SessionSynchronization) */ public class MongoTransactionManager extends AbstractPlatformTransactionManager implements ResourceTransactionManager, InitializingBean { private @Nullable MongoDatabaseFactory databaseFactory; private MongoTransactionOptions options; private final MongoTransactionOptionsResolver transactionOptionsResolver; /** * Create a new {@link MongoTransactionManager} for bean-style usage.
* Note:The {@link MongoDatabaseFactory db factory} has to be * {@link #setDatabaseFactory(MongoDatabaseFactory) set} before using the instance. Use this constructor to prepare a * {@link MongoTransactionManager} via a {@link org.springframework.beans.factory.BeanFactory}.
* Optionally it is possible to set default {@link TransactionOptions transaction options} defining * {@link com.mongodb.ReadConcern} and {@link com.mongodb.WriteConcern}. * * @see #setDatabaseFactory(MongoDatabaseFactory) * @see #setTransactionSynchronization(int) */ public MongoTransactionManager() { this.transactionOptionsResolver = MongoTransactionOptionsResolver.defaultResolver(); } /** * Create a new {@link MongoTransactionManager} obtaining sessions from the given {@link MongoDatabaseFactory}. * * @param databaseFactory must not be {@literal null}. */ public MongoTransactionManager(MongoDatabaseFactory databaseFactory) { this(databaseFactory, null); } /** * Create a new {@link MongoTransactionManager} obtaining sessions from the given {@link MongoDatabaseFactory} * applying the given {@link TransactionOptions options}, if present, when starting a new transaction. * * @param databaseFactory must not be {@literal null}. * @param options can be {@literal null}. */ public MongoTransactionManager(MongoDatabaseFactory databaseFactory, @Nullable TransactionOptions options) { this(databaseFactory, MongoTransactionOptionsResolver.defaultResolver(), MongoTransactionOptions.of(options)); } /** * Create a new {@link MongoTransactionManager} obtaining sessions from the given {@link MongoDatabaseFactory} * applying the given {@link TransactionOptions options}, if present, when starting a new transaction. * * @param databaseFactory must not be {@literal null}. * @param transactionOptionsResolver must not be {@literal null}. * @param defaultTransactionOptions can be {@literal null}. * @since 4.3 */ public MongoTransactionManager(MongoDatabaseFactory databaseFactory, MongoTransactionOptionsResolver transactionOptionsResolver, MongoTransactionOptions defaultTransactionOptions) { Assert.notNull(databaseFactory, "MongoDatabaseFactory must not be null"); Assert.notNull(transactionOptionsResolver, "MongoTransactionOptionsResolver must not be null"); this.databaseFactory = databaseFactory; this.transactionOptionsResolver = transactionOptionsResolver; this.options = defaultTransactionOptions; } @Override protected Object doGetTransaction() throws TransactionException { MongoResourceHolder resourceHolder = (MongoResourceHolder) TransactionSynchronizationManager .getResource(getRequiredDbFactory()); return new MongoTransactionObject(resourceHolder); } @Override protected boolean isExistingTransaction(Object transaction) throws TransactionException { return extractMongoTransaction(transaction).hasResourceHolder(); } @Override protected void doBegin(Object transaction, TransactionDefinition definition) throws TransactionException { MongoTransactionObject mongoTransactionObject = extractMongoTransaction(transaction); MongoResourceHolder resourceHolder = newResourceHolder(definition, ClientSessionOptions.builder().causallyConsistent(true).build()); mongoTransactionObject.setResourceHolder(resourceHolder); if (logger.isDebugEnabled()) { logger .debug(String.format("About to start transaction for session %s.", debugString(resourceHolder.getSession()))); } try { MongoTransactionOptions mongoTransactionOptions = transactionOptionsResolver.resolve(definition).mergeWith(options); mongoTransactionObject.startTransaction(mongoTransactionOptions.toDriverOptions()); } catch (MongoException ex) { throw new TransactionSystemException(String.format("Could not start Mongo transaction for session %s.", debugString(mongoTransactionObject.getSession())), ex); } if (logger.isDebugEnabled()) { logger.debug(String.format("Started transaction for session %s.", debugString(resourceHolder.getSession()))); } resourceHolder.setSynchronizedWithTransaction(true); TransactionSynchronizationManager.bindResource(getRequiredDbFactory(), resourceHolder); } @Override protected Object doSuspend(Object transaction) throws TransactionException { MongoTransactionObject mongoTransactionObject = extractMongoTransaction(transaction); mongoTransactionObject.setResourceHolder(null); return TransactionSynchronizationManager.unbindResource(getRequiredDbFactory()); } @Override protected void doResume(@Nullable Object transaction, Object suspendedResources) { TransactionSynchronizationManager.bindResource(getRequiredDbFactory(), suspendedResources); } @Override protected final void doCommit(DefaultTransactionStatus status) throws TransactionException { MongoTransactionObject mongoTransactionObject = extractMongoTransaction(status); if (logger.isDebugEnabled()) { logger.debug(String.format("About to commit transaction for session %s.", debugString(mongoTransactionObject.getSession()))); } try { doCommit(mongoTransactionObject); } catch (Exception ex) { throw new TransactionSystemException(String.format("Could not commit Mongo transaction for session %s.", debugString(mongoTransactionObject.getSession())), ex); } } /** * Customization hook to perform an actual commit of the given transaction.
* If a commit operation encounters an error, the MongoDB driver throws a {@link MongoException} holding * {@literal error labels}.
* By default those labels are ignored, nevertheless one might check for * {@link MongoException#UNKNOWN_TRANSACTION_COMMIT_RESULT_LABEL transient commit errors labels} and retry the the * commit.
*
	 * 
	 * int retries = 3;
	 * do {
	 *     try {
	 *         transactionObject.commitTransaction();
	 *         break;
	 *     } catch (MongoException ex) {
	 *         if (!ex.hasErrorLabel(MongoException.UNKNOWN_TRANSACTION_COMMIT_RESULT_LABEL)) {
	 *             throw ex;
	 *         }
	 *     }
	 *     Thread.sleep(500);
	 * } while (--retries > 0);
	 * 
	 * 
* * @param transactionObject never {@literal null}. * @throws Exception in case of transaction errors. */ protected void doCommit(MongoTransactionObject transactionObject) throws Exception { transactionObject.commitTransaction(); } @Override protected void doRollback(DefaultTransactionStatus status) throws TransactionException { MongoTransactionObject mongoTransactionObject = extractMongoTransaction(status); if (logger.isDebugEnabled()) { logger.debug(String.format("About to abort transaction for session %s.", debugString(mongoTransactionObject.getSession()))); } try { mongoTransactionObject.abortTransaction(); } catch (MongoException ex) { throw new TransactionSystemException(String.format("Could not abort Mongo transaction for session %s.", debugString(mongoTransactionObject.getSession())), ex); } } @Override protected void doSetRollbackOnly(DefaultTransactionStatus status) throws TransactionException { MongoTransactionObject transactionObject = extractMongoTransaction(status); transactionObject.getRequiredResourceHolder().setRollbackOnly(); } @Override protected void doCleanupAfterCompletion(Object transaction) { Assert.isInstanceOf(MongoTransactionObject.class, transaction, () -> String.format("Expected to find a %s but it turned out to be %s.", MongoTransactionObject.class, transaction.getClass())); MongoTransactionObject mongoTransactionObject = (MongoTransactionObject) transaction; // Remove the connection holder from the thread. TransactionSynchronizationManager.unbindResource(getRequiredDbFactory()); mongoTransactionObject.getRequiredResourceHolder().clear(); if (logger.isDebugEnabled()) { logger.debug(String.format("About to release Session %s after transaction.", debugString(mongoTransactionObject.getSession()))); } mongoTransactionObject.closeSession(); } /** * Set the {@link MongoDatabaseFactory} that this instance should manage transactions for. * * @param databaseFactory must not be {@literal null}. */ public void setDatabaseFactory(MongoDatabaseFactory databaseFactory) { Assert.notNull(databaseFactory, "DbFactory must not be null"); this.databaseFactory = databaseFactory; } /** * Set the {@link TransactionOptions} to be applied when starting transactions. * * @param options can be {@literal null}. */ public void setOptions(@Nullable TransactionOptions options) { this.options = MongoTransactionOptions.of(options); } /** * Get the {@link MongoDatabaseFactory} that this instance manages transactions for. * * @return can be {@literal null}. */ @Nullable public MongoDatabaseFactory getDatabaseFactory() { return databaseFactory; } @Override public MongoDatabaseFactory getResourceFactory() { return getRequiredDbFactory(); } @Override public void afterPropertiesSet() { getRequiredDbFactory(); } private MongoResourceHolder newResourceHolder(TransactionDefinition definition, ClientSessionOptions options) { MongoDatabaseFactory dbFactory = getResourceFactory(); MongoResourceHolder resourceHolder = new MongoResourceHolder(dbFactory.getSession(options), dbFactory); resourceHolder.setTimeoutIfNotDefaulted(determineTimeout(definition)); return resourceHolder; } /** * @throws IllegalStateException if {@link #databaseFactory} is {@literal null}. */ private MongoDatabaseFactory getRequiredDbFactory() { Assert.state(databaseFactory != null, "MongoTransactionManager operates upon a MongoDbFactory; Did you forget to provide one; It's required"); return databaseFactory; } private static MongoTransactionObject extractMongoTransaction(Object transaction) { Assert.isInstanceOf(MongoTransactionObject.class, transaction, () -> String.format("Expected to find a %s but it turned out to be %s.", MongoTransactionObject.class, transaction.getClass())); return (MongoTransactionObject) transaction; } private static MongoTransactionObject extractMongoTransaction(DefaultTransactionStatus status) { Assert.isInstanceOf(MongoTransactionObject.class, status.getTransaction(), () -> String.format("Expected to find a %s but it turned out to be %s.", MongoTransactionObject.class, status.getTransaction().getClass())); return (MongoTransactionObject) status.getTransaction(); } private static String debugString(@Nullable ClientSession session) { if (session == null) { return "null"; } String debugString = String.format("[%s@%s ", ClassUtils.getShortName(session.getClass()), Integer.toHexString(session.hashCode())); try { if (session.getServerSession() != null) { debugString += String.format("id = %s, ", session.getServerSession().getIdentifier()); debugString += String.format("causallyConsistent = %s, ", session.isCausallyConsistent()); debugString += String.format("txActive = %s, ", session.hasActiveTransaction()); debugString += String.format("txNumber = %d, ", session.getServerSession().getTransactionNumber()); debugString += String.format("closed = %b, ", session.getServerSession().isClosed()); debugString += String.format("clusterTime = %s", session.getClusterTime()); } else { debugString += "id = n/a"; debugString += String.format("causallyConsistent = %s, ", session.isCausallyConsistent()); debugString += String.format("txActive = %s, ", session.hasActiveTransaction()); debugString += String.format("clusterTime = %s", session.getClusterTime()); } } catch (RuntimeException e) { debugString += String.format("error = %s", e.getMessage()); } debugString += "]"; return debugString; } /** * MongoDB specific transaction object, representing a {@link MongoResourceHolder}. Used as transaction object by * {@link MongoTransactionManager}. * * @author Christoph Strobl * @author Mark Paluch * @since 2.1 * @see MongoResourceHolder */ protected static class MongoTransactionObject implements SmartTransactionObject { private @Nullable MongoResourceHolder resourceHolder; MongoTransactionObject(@Nullable MongoResourceHolder resourceHolder) { this.resourceHolder = resourceHolder; } /** * Set the {@link MongoResourceHolder}. * * @param resourceHolder can be {@literal null}. */ void setResourceHolder(@Nullable MongoResourceHolder resourceHolder) { this.resourceHolder = resourceHolder; } /** * @return {@literal true} if a {@link MongoResourceHolder} is set. */ final boolean hasResourceHolder() { return resourceHolder != null; } /** * Start a MongoDB transaction optionally given {@link TransactionOptions}. * * @param options can be {@literal null} */ void startTransaction(@Nullable TransactionOptions options) { ClientSession session = getRequiredSession(); if (options != null) { session.startTransaction(options); } else { session.startTransaction(); } } /** * Commit the transaction. */ public void commitTransaction() { getRequiredSession().commitTransaction(); } /** * Rollback (abort) the transaction. */ public void abortTransaction() { getRequiredSession().abortTransaction(); } /** * Close a {@link ClientSession} without regard to its transactional state. */ void closeSession() { ClientSession session = getRequiredSession(); if (session.getServerSession() != null && !session.getServerSession().isClosed()) { session.close(); } } @Nullable public ClientSession getSession() { return resourceHolder != null ? resourceHolder.getSession() : null; } private MongoResourceHolder getRequiredResourceHolder() { Assert.state(resourceHolder != null, "MongoResourceHolder is required but not present; o_O"); return resourceHolder; } private ClientSession getRequiredSession() { ClientSession session = getSession(); Assert.state(session != null, "A Session is required but it turned out to be null"); return session; } @Override public boolean isRollbackOnly() { return this.resourceHolder != null && this.resourceHolder.isRollbackOnly(); } @Override public void flush() { TransactionSynchronizationUtils.triggerFlush(); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy