com.mongodb.operation.CommandOperationHelper Maven / Gradle / Ivy
/*
* Copyright (c) 2008-2014 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.operation;
import com.mongodb.MongoCommandException;
import com.mongodb.ReadPreference;
import com.mongodb.ServerAddress;
import com.mongodb.async.SingleResultCallback;
import com.mongodb.binding.AsyncConnectionSource;
import com.mongodb.binding.AsyncReadBinding;
import com.mongodb.binding.AsyncWriteBinding;
import com.mongodb.binding.ConnectionSource;
import com.mongodb.binding.ReadBinding;
import com.mongodb.binding.WriteBinding;
import com.mongodb.connection.AsyncConnection;
import com.mongodb.connection.Connection;
import com.mongodb.connection.ConnectionDescription;
import com.mongodb.internal.validator.NoOpFieldNameValidator;
import org.bson.BsonDocument;
import org.bson.FieldNameValidator;
import org.bson.codecs.BsonDocumentCodec;
import org.bson.codecs.Decoder;
import static com.mongodb.ReadPreference.primary;
import static com.mongodb.assertions.Assertions.notNull;
import static com.mongodb.connection.ServerType.SHARD_ROUTER;
import static com.mongodb.internal.async.ErrorHandlingResultCallback.errorHandlingCallback;
import static com.mongodb.operation.OperationHelper.releasingCallback;
final class CommandOperationHelper {
interface CommandTransformer {
/**
* Yield an appropriate result object for the input object.
*
* @param t the input object
* @return the function result
*/
R apply(T t, ServerAddress serverAddress);
}
static class IdentityTransformer implements CommandTransformer {
@Override
public T apply(final T t, final ServerAddress serverAddress) {
return t;
}
}
static class VoidTransformer implements CommandTransformer {
@Override
public Void apply(final T t, final ServerAddress serverAddress) {
return null;
}
}
/* Read Binding Helpers */
static BsonDocument executeWrappedCommandProtocol(final ReadBinding binding, final String database, final BsonDocument command) {
return executeWrappedCommandProtocol(binding, database, command, new BsonDocumentCodec());
}
static T executeWrappedCommandProtocol(final ReadBinding binding, final String database, final BsonDocument command,
final CommandTransformer transformer) {
return executeWrappedCommandProtocol(binding, database, command, new BsonDocumentCodec(), transformer);
}
static T executeWrappedCommandProtocol(final ReadBinding binding, final String database, final BsonDocument command,
final Decoder decoder) {
return executeWrappedCommandProtocol(binding, database, command, decoder, new IdentityTransformer());
}
static T executeWrappedCommandProtocol(final ReadBinding binding, final String database, final BsonDocument command,
final Decoder decoder, final CommandTransformer transformer) {
ConnectionSource source = binding.getReadConnectionSource();
try {
return transformer.apply(executeWrappedCommandProtocol(database, command, decoder, source,
binding.getReadPreference()),
source.getServerDescription().getAddress());
} finally {
source.release();
}
}
static BsonDocument executeWrappedCommandProtocol(final ReadBinding binding, final String database, final BsonDocument command,
final Connection connection) {
return executeWrappedCommandProtocol(binding, database, command, connection, new IdentityTransformer());
}
static T executeWrappedCommandProtocol(final ReadBinding binding, final String database, final BsonDocument command,
final Connection connection, final CommandTransformer transformer) {
return executeWrappedCommandProtocol(binding, database, command, new BsonDocumentCodec(), connection, transformer);
}
static T executeWrappedCommandProtocol(final ReadBinding binding, final String database, final BsonDocument command,
final Decoder decoder, final Connection connection,
final CommandTransformer transformer) {
return executeWrappedCommandProtocol(database, command, decoder, connection, binding.getReadPreference(), transformer);
}
/* Write Binding Helpers */
static BsonDocument executeWrappedCommandProtocol(final WriteBinding binding, final String database, final BsonDocument command) {
return executeWrappedCommandProtocol(binding, database, command, new IdentityTransformer());
}
static T executeWrappedCommandProtocol(final WriteBinding binding, final String database, final BsonDocument command,
final Decoder decoder) {
return executeWrappedCommandProtocol(binding, database, command, decoder, new IdentityTransformer());
}
static T executeWrappedCommandProtocol(final WriteBinding binding, final String database, final BsonDocument command,
final CommandTransformer transformer) {
return executeWrappedCommandProtocol(binding, database, command, new BsonDocumentCodec(), transformer);
}
static T executeWrappedCommandProtocol(final WriteBinding binding, final String database, final BsonDocument command,
final Decoder decoder, final CommandTransformer transformer) {
return executeWrappedCommandProtocol(binding, database, command, new NoOpFieldNameValidator(), decoder, transformer);
}
static T executeWrappedCommandProtocol(final WriteBinding binding, final String database, final BsonDocument command,
final Connection connection, final CommandTransformer transformer) {
return executeWrappedCommandProtocol(binding, database, command, new BsonDocumentCodec(), connection, transformer);
}
static T executeWrappedCommandProtocol(final WriteBinding binding, final String database, final BsonDocument command,
final Decoder decoder, final Connection connection,
final CommandTransformer transformer) {
notNull("binding", binding);
return executeWrappedCommandProtocol(database, command, decoder, connection, primary(), transformer);
}
static T executeWrappedCommandProtocol(final WriteBinding binding, final String database, final BsonDocument command,
final FieldNameValidator fieldNameValidator, final Decoder decoder,
final Connection connection, final CommandTransformer transformer) {
notNull("binding", binding);
return executeWrappedCommandProtocol(database, command, fieldNameValidator, decoder, connection, primary(), transformer);
}
static T executeWrappedCommandProtocol(final WriteBinding binding, final String database, final BsonDocument command,
final FieldNameValidator fieldNameValidator, final Decoder decoder,
final CommandTransformer transformer) {
ConnectionSource source = binding.getWriteConnectionSource();
try {
return transformer.apply(executeWrappedCommandProtocol(database, command, fieldNameValidator, decoder,
source, primary()), source.getServerDescription().getAddress());
} finally {
source.release();
}
}
static BsonDocument executeWrappedCommandProtocol(final WriteBinding binding, final String database, final BsonDocument command,
final Connection connection) {
notNull("binding", binding);
return executeWrappedCommandProtocol(database, command, new BsonDocumentCodec(), connection, primary());
}
/* Private Connection Source Helpers */
private static T executeWrappedCommandProtocol(final String database, final BsonDocument command,
final Decoder decoder, final ConnectionSource source,
final ReadPreference readPreference) {
return executeWrappedCommandProtocol(database, command, new NoOpFieldNameValidator(), decoder, source, readPreference);
}
private static T executeWrappedCommandProtocol(final String database, final BsonDocument command,
final FieldNameValidator fieldNameValidator, final Decoder decoder,
final ConnectionSource source, final ReadPreference readPreference) {
Connection connection = source.getConnection();
try {
return executeWrappedCommandProtocol(database, command, fieldNameValidator, decoder, connection,
readPreference, new IdentityTransformer());
} finally {
connection.release();
}
}
/* Private Connection Helpers */
private static T executeWrappedCommandProtocol(final String database, final BsonDocument command,
final Decoder decoder, final Connection connection,
final ReadPreference readPreference) {
return executeWrappedCommandProtocol(database, command, new NoOpFieldNameValidator(), decoder, connection,
readPreference, new IdentityTransformer());
}
private static T executeWrappedCommandProtocol(final String database, final BsonDocument command,
final Decoder decoder, final Connection connection,
final ReadPreference readPreference,
final CommandTransformer transformer) {
return executeWrappedCommandProtocol(database, command, new NoOpFieldNameValidator(), decoder, connection,
readPreference, transformer);
}
private static T executeWrappedCommandProtocol(final String database, final BsonDocument command,
final FieldNameValidator fieldNameValidator, final Decoder decoder,
final Connection connection, final ReadPreference readPreference,
final CommandTransformer transformer) {
return transformer.apply(connection.command(database, wrapCommand(command, readPreference, connection.getDescription()),
readPreference.isSlaveOk(), fieldNameValidator, decoder), connection.getDescription().getServerAddress());
}
/* Async Read Binding Helpers */
static void executeWrappedCommandProtocolAsync(final AsyncReadBinding binding,
final String database,
final BsonDocument command,
final SingleResultCallback callback) {
executeWrappedCommandProtocolAsync(binding, database, command, new BsonDocumentCodec(), callback);
}
static void executeWrappedCommandProtocolAsync(final AsyncReadBinding binding,
final String database,
final BsonDocument command,
final Decoder decoder,
final SingleResultCallback callback) {
executeWrappedCommandProtocolAsync(binding, database, command, decoder, new IdentityTransformer(), callback);
}
static void executeWrappedCommandProtocolAsync(final AsyncReadBinding binding,
final String database,
final BsonDocument command,
final CommandTransformer transformer,
final SingleResultCallback callback) {
executeWrappedCommandProtocolAsync(binding, database, command, new BsonDocumentCodec(), transformer, callback);
}
static void executeWrappedCommandProtocolAsync(final AsyncReadBinding binding,
final String database,
final BsonDocument command,
final Decoder decoder,
final CommandTransformer transformer,
final SingleResultCallback callback) {
binding.getReadConnectionSource(new CommandProtocolExecutingCallback(database, command, new NoOpFieldNameValidator(),
decoder, binding.getReadPreference(), transformer,
errorHandlingCallback(callback)));
}
static void executeWrappedCommandProtocolAsync(final AsyncReadBinding binding,
final String database,
final BsonDocument command,
final Decoder decoder,
final AsyncConnection connection,
final CommandTransformer transformer,
final SingleResultCallback callback) {
notNull("binding", binding);
executeWrappedCommandProtocolAsync(database, command, decoder, connection, binding.getReadPreference(), transformer,
callback);
}
/* Async Write Binding Helpers */
static void executeWrappedCommandProtocolAsync(final AsyncWriteBinding binding,
final String database,
final BsonDocument command,
final SingleResultCallback callback) {
executeWrappedCommandProtocolAsync(binding, database, command, new BsonDocumentCodec(), callback);
}
static void executeWrappedCommandProtocolAsync(final AsyncWriteBinding binding,
final String database,
final BsonDocument command,
final Decoder decoder,
final SingleResultCallback callback) {
executeWrappedCommandProtocolAsync(binding, database, command, decoder, new IdentityTransformer(), callback);
}
static void executeWrappedCommandProtocolAsync(final AsyncWriteBinding binding,
final String database,
final BsonDocument command,
final CommandTransformer transformer,
final SingleResultCallback callback) {
executeWrappedCommandProtocolAsync(binding, database, command, new BsonDocumentCodec(), transformer, callback);
}
static void executeWrappedCommandProtocolAsync(final AsyncWriteBinding binding,
final String database, final BsonDocument command,
final Decoder decoder,
final CommandTransformer transformer,
final SingleResultCallback callback) {
executeWrappedCommandProtocolAsync(binding, database, command, new NoOpFieldNameValidator(), decoder, transformer, callback);
}
static void executeWrappedCommandProtocolAsync(final AsyncWriteBinding binding,
final String database,
final BsonDocument command,
final Decoder decoder,
final AsyncConnection connection,
final CommandTransformer transformer,
final SingleResultCallback callback) {
notNull("binding", binding);
executeWrappedCommandProtocolAsync(database, command, decoder, connection, primary(), transformer, callback);
}
static void executeWrappedCommandProtocolAsync(final AsyncWriteBinding binding,
final String database,
final BsonDocument command,
final FieldNameValidator fieldNameValidator,
final Decoder decoder,
final AsyncConnection connection,
final CommandTransformer transformer,
final SingleResultCallback callback) {
notNull("binding", binding);
executeWrappedCommandProtocolAsync(database, command, fieldNameValidator, decoder, connection, primary(), transformer, callback);
}
static void executeWrappedCommandProtocolAsync(final AsyncWriteBinding binding,
final String database, final BsonDocument command,
final FieldNameValidator fieldNameValidator,
final Decoder decoder,
final CommandTransformer transformer,
final SingleResultCallback callback) {
binding.getWriteConnectionSource(new CommandProtocolExecutingCallback(database, command, fieldNameValidator, decoder,
primary(), transformer,
errorHandlingCallback(callback)));
}
static void executeWrappedCommandProtocolAsync(final AsyncWriteBinding binding,
final String database,
final BsonDocument command,
final AsyncConnection connection,
final SingleResultCallback callback) {
executeWrappedCommandProtocolAsync(binding, database, command, connection, new IdentityTransformer(), callback);
}
static void executeWrappedCommandProtocolAsync(final AsyncWriteBinding binding,
final String database,
final BsonDocument command,
final AsyncConnection connection,
final CommandTransformer transformer,
final SingleResultCallback callback) {
notNull("binding", binding);
executeWrappedCommandProtocolAsync(database, command, new BsonDocumentCodec(), connection, primary(), transformer, callback);
}
/* Async Connection Helpers */
private static void executeWrappedCommandProtocolAsync(final String database, final BsonDocument command,
final Decoder decoder, final AsyncConnection connection,
final ReadPreference readPreference,
final CommandTransformer transformer,
final SingleResultCallback callback) {
executeWrappedCommandProtocolAsync(database, command, new NoOpFieldNameValidator(), decoder, connection, readPreference,
transformer, callback);
}
private static void executeWrappedCommandProtocolAsync(final String database, final BsonDocument command,
final FieldNameValidator fieldNameValidator,
final Decoder decoder, final AsyncConnection connection,
final ReadPreference readPreference,
final CommandTransformer transformer,
final SingleResultCallback callback) {
connection.commandAsync(database, wrapCommand(command, readPreference, connection.getDescription()),
readPreference.isSlaveOk(), fieldNameValidator, decoder, new SingleResultCallback() {
@Override
public void onResult(final D result, final Throwable t) {
if (t != null) {
callback.onResult(null, t);
} else {
try {
T transformedResult = transformer.apply(result, connection.getDescription().getServerAddress());
callback.onResult(transformedResult, null);
} catch (Exception e) {
callback.onResult(null, e);
}
}
}
});
}
/* Misc operation helpers */
static void rethrowIfNotNamespaceError(final MongoCommandException e) {
rethrowIfNotNamespaceError(e, null);
}
static T rethrowIfNotNamespaceError(final MongoCommandException e, final T defaultValue) {
if (!isNamespaceError(e)) {
throw e;
}
return defaultValue;
}
static boolean isNamespaceError(final Throwable t) {
if (t instanceof MongoCommandException) {
MongoCommandException e = (MongoCommandException) t;
return (e.getErrorMessage().contains("ns not found") || e.getErrorCode() == 26);
} else {
return false;
}
}
static BsonDocument wrapCommand(final BsonDocument command, final ReadPreference readPreference,
final ConnectionDescription connectionDescription) {
if (connectionDescription.getServerType() == SHARD_ROUTER && !readPreference.equals(primary())) {
return new BsonDocument("$query", command).append("$readPreference", readPreference.toDocument());
} else {
return command;
}
}
private static class CommandProtocolExecutingCallback implements SingleResultCallback {
private final String database;
private final BsonDocument command;
private final Decoder decoder;
private final ReadPreference readPreference;
private final FieldNameValidator fieldNameValidator;
private final CommandTransformer transformer;
private final SingleResultCallback callback;
public CommandProtocolExecutingCallback(final String database, final BsonDocument command,
final FieldNameValidator fieldNameValidator,
final Decoder decoder,
final ReadPreference readPreference,
final CommandTransformer transformer,
final SingleResultCallback callback) {
this.database = database;
this.command = command;
this.fieldNameValidator = fieldNameValidator;
this.decoder = decoder;
this.readPreference = readPreference;
this.transformer = transformer;
this.callback = callback;
}
@Override
public void onResult(final AsyncConnectionSource source, final Throwable t) {
if (t != null) {
callback.onResult(null, t);
} else {
source.getConnection(new SingleResultCallback() {
@Override
public void onResult(final AsyncConnection connection, final Throwable t) {
if (t != null) {
callback.onResult(null, t);
} else {
final SingleResultCallback wrappedCallback = releasingCallback(callback, source, connection);
connection.commandAsync(database, wrapCommand(command, readPreference, connection.getDescription()),
readPreference.isSlaveOk(), fieldNameValidator, decoder, new SingleResultCallback() {
@Override
public void onResult(final D response, final Throwable t) {
if (t != null) {
wrappedCallback.onResult(null, t);
} else {
wrappedCallback.onResult(transformer.apply(response,
connection.getDescription().getServerAddress()),
null);
}
}
});
}
}
});
}
}
}
private CommandOperationHelper() {
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy