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

com.mongodb.internal.operation.CommitTransactionOperation Maven / Gradle / Ivy

Go to download

The Java operations layer for the MongoDB Java Driver. Third parties can wrap this layer to provide custom higher-level APIs

There is a newer version: 5.3.0-beta0
Show newest version
/*
 * 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.operation;

import com.mongodb.Function;
import com.mongodb.MongoException;
import com.mongodb.MongoExecutionTimeoutException;
import com.mongodb.MongoNodeIsRecoveringException;
import com.mongodb.MongoNotPrimaryException;
import com.mongodb.MongoSocketException;
import com.mongodb.MongoTimeoutException;
import com.mongodb.MongoWriteConcernException;
import com.mongodb.WriteConcern;
import com.mongodb.internal.async.SingleResultCallback;
import com.mongodb.internal.binding.AsyncWriteBinding;
import com.mongodb.internal.binding.WriteBinding;
import com.mongodb.lang.Nullable;
import org.bson.BsonDocument;
import org.bson.BsonInt32;
import org.bson.BsonInt64;

import java.util.List;
import java.util.concurrent.TimeUnit;

import static com.mongodb.MongoException.UNKNOWN_TRANSACTION_COMMIT_RESULT_LABEL;
import static com.mongodb.assertions.Assertions.isTrueArgument;
import static com.mongodb.assertions.Assertions.notNull;
import static com.mongodb.internal.operation.CommandOperationHelper.CommandCreator;
import static com.mongodb.internal.operation.CommandOperationHelper.RETRYABLE_WRITE_ERROR_LABEL;
import static java.util.Arrays.asList;
import static java.util.concurrent.TimeUnit.MILLISECONDS;

/**
 * An operation that commits a transaction.
 *
 * 

This class is not part of the public API and may be removed or changed at any time

*/ public class CommitTransactionOperation extends TransactionOperation { private final boolean alreadyCommitted; private BsonDocument recoveryToken; private Long maxCommitTimeMS; public CommitTransactionOperation(final WriteConcern writeConcern) { this(writeConcern, false); } public CommitTransactionOperation(final WriteConcern writeConcern, final boolean alreadyCommitted) { super(writeConcern); this.alreadyCommitted = alreadyCommitted; } public CommitTransactionOperation recoveryToken(@Nullable final BsonDocument recoveryToken) { this.recoveryToken = recoveryToken; return this; } public CommitTransactionOperation maxCommitTime(@Nullable final Long maxCommitTime, final TimeUnit timeUnit) { if (maxCommitTime == null) { this.maxCommitTimeMS = null; } else { notNull("timeUnit", timeUnit); isTrueArgument("maxCommitTime > 0", maxCommitTime > 0); this.maxCommitTimeMS = MILLISECONDS.convert(maxCommitTime, timeUnit); } return this; } @Nullable public Long getMaxCommitTime(final TimeUnit timeUnit) { notNull("timeUnit", timeUnit); if (maxCommitTimeMS == null) { return null; } return timeUnit.convert(maxCommitTimeMS, MILLISECONDS); } @Override public Void execute(final WriteBinding binding) { try { return super.execute(binding); } catch (MongoException e) { addErrorLabels(e); throw e; } } @Override public void executeAsync(final AsyncWriteBinding binding, final SingleResultCallback callback) { super.executeAsync(binding, (result, t) -> { if (t instanceof MongoException) { addErrorLabels((MongoException) t); } callback.onResult(result, t); }); } private void addErrorLabels(final MongoException e) { if (shouldAddUnknownTransactionCommitResultLabel(e)) { e.addLabel(UNKNOWN_TRANSACTION_COMMIT_RESULT_LABEL); } } private static final List NON_RETRYABLE_WRITE_CONCERN_ERROR_CODES = asList(79, 100); private static boolean shouldAddUnknownTransactionCommitResultLabel(final MongoException e) { if (e instanceof MongoSocketException || e instanceof MongoTimeoutException || e instanceof MongoNotPrimaryException || e instanceof MongoNodeIsRecoveringException || e instanceof MongoExecutionTimeoutException) { return true; } if (e.hasErrorLabel(RETRYABLE_WRITE_ERROR_LABEL)) { return true; } if (e instanceof MongoWriteConcernException) { return !NON_RETRYABLE_WRITE_CONCERN_ERROR_CODES.contains(e.getCode()); } return false; } @Override protected String getCommandName() { return "commitTransaction"; } @Override CommandCreator getCommandCreator() { CommandCreator creator = (serverDescription, connectionDescription) -> { BsonDocument command = CommitTransactionOperation.super.getCommandCreator().create(serverDescription, connectionDescription); if (maxCommitTimeMS != null) { command.append("maxTimeMS", maxCommitTimeMS > Integer.MAX_VALUE ? new BsonInt64(maxCommitTimeMS) : new BsonInt32(maxCommitTimeMS.intValue())); } return command; }; if (alreadyCommitted) { return (serverDescription, connectionDescription) -> getRetryCommandModifier().apply(creator.create(serverDescription, connectionDescription)); } else if (recoveryToken != null) { return (serverDescription, connectionDescription) -> creator.create(serverDescription, connectionDescription).append("recoveryToken", recoveryToken); } return creator; } @Override protected Function getRetryCommandModifier() { return command -> { WriteConcern retryWriteConcern = getWriteConcern().withW("majority"); if (retryWriteConcern.getWTimeout(MILLISECONDS) == null) { retryWriteConcern = retryWriteConcern.withWTimeout(10000, MILLISECONDS); } command.put("writeConcern", retryWriteConcern.asDocument()); if (recoveryToken != null) { command.put("recoveryToken", recoveryToken); } return command; }; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy