package com.mongodb.async.client;
import com.mongodb.Function;
import com.mongodb.MongoClientException;
import com.mongodb.MongoNamespace;
import com.mongodb.ReadConcern;
import com.mongodb.ReadPreference;
import com.mongodb.WriteConcern;
import com.mongodb.async.SingleResultCallback;
import com.mongodb.client.model.AggregationLevel;
import com.mongodb.client.model.CreateCollectionOptions;
import com.mongodb.client.model.CreateViewOptions;
import com.mongodb.client.model.IndexOptionDefaults;
import com.mongodb.client.model.ValidationOptions;
import com.mongodb.client.model.changestream.ChangeStreamLevel;
import com.mongodb.lang.Nullable;
import com.mongodb.operation.CommandReadOperation;
import com.mongodb.operation.CreateCollectionOperation;
import com.mongodb.operation.CreateViewOperation;
import com.mongodb.operation.DropDatabaseOperation;
import org.bson.BsonDocument;
import org.bson.Document;
import org.bson.codecs.configuration.CodecRegistry;
import org.bson.conversions.Bson;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import static com.mongodb.MongoNamespace.checkDatabaseNameValidity;
import static com.mongodb.assertions.Assertions.notNull;
class MongoDatabaseImpl implements MongoDatabase {
private final String name;
private final ReadPreference readPreference;
private final CodecRegistry codecRegistry;
private final WriteConcern writeConcern;
private final boolean retryWrites;
private final boolean retryReads;
private final ReadConcern readConcern;
private final OperationExecutor executor;
MongoDatabaseImpl(final String name, final CodecRegistry codecRegistry, final ReadPreference readPreference,
final WriteConcern writeConcern, final boolean retryWrites, final boolean retryReads,
final ReadConcern readConcern, final OperationExecutor executor) {
checkDatabaseNameValidity(name);
this.name = notNull("name", name);
this.codecRegistry = notNull("codecRegistry", codecRegistry);
this.readPreference = notNull("readPreference", readPreference);
this.writeConcern = notNull("writeConcern", writeConcern);
this.retryWrites = retryWrites;
this.retryReads = retryReads;
this.readConcern = notNull("readConcern", readConcern);
this.executor = notNull("executor", executor);
}
@Override
public String getName() {
return name;
}
@Override
public CodecRegistry getCodecRegistry() {
return codecRegistry;
}
@Override
public ReadPreference getReadPreference() {
return readPreference;
}
@Override
public WriteConcern getWriteConcern() {
return writeConcern;
}
@Override
public ReadConcern getReadConcern() {
return readConcern;
}
@Override
public MongoDatabase withCodecRegistry(final CodecRegistry codecRegistry) {
return new MongoDatabaseImpl(name, codecRegistry, readPreference, writeConcern, retryWrites, retryReads, readConcern, executor);
}
@Override
public MongoDatabase withReadPreference(final ReadPreference readPreference) {
return new MongoDatabaseImpl(name, codecRegistry, readPreference, writeConcern, retryWrites, retryReads, readConcern, executor);
}
@Override
public MongoDatabase withWriteConcern(final WriteConcern writeConcern) {
return new MongoDatabaseImpl(name, codecRegistry, readPreference, writeConcern, retryWrites, retryReads, readConcern, executor);
}
@Override
public MongoDatabase withReadConcern(final ReadConcern readConcern) {
return new MongoDatabaseImpl(name, codecRegistry, readPreference, writeConcern, retryWrites, retryReads, readConcern, executor);
}
@Override
public MongoIterable<String> listCollectionNames() {
return createListCollectionNamesIterable(null);
}
@Override
public MongoIterable<String> listCollectionNames(final ClientSession clientSession) {
notNull("clientSession", clientSession);
return createListCollectionNamesIterable(clientSession);
}
private MongoIterable<String> createListCollectionNamesIterable(@Nullable final ClientSession clientSession) {
return createListCollectionsIterable(clientSession, BsonDocument.class, true)
.map(new Function<BsonDocument, String>() {
@Override
public String apply(final BsonDocument result) {
return result.getString("name").getValue();
}
});
}
@Override
public ListCollectionsIterable<Document> listCollections() {
return listCollections(Document.class);
}
@Override
public <TResult> ListCollectionsIterable<TResult> listCollections(final Class<TResult> resultClass) {
return createListCollectionsIterable(null, resultClass, false);
}
@Override
public ListCollectionsIterable<Document> listCollections(final ClientSession clientSession) {
return listCollections(clientSession, Document.class);
}
@Override
public <TResult> ListCollectionsIterable<TResult> listCollections(final ClientSession clientSession, final Class<TResult> resultClass) {
notNull("clientSession", clientSession);
return createListCollectionsIterable(clientSession, resultClass, false);
}
private <TResult> ListCollectionsIterable<TResult> createListCollectionsIterable(@Nullable final ClientSession clientSession,
final Class<TResult> resultClass,
final boolean collectionNamesOnly) {
return new ListCollectionsIterableImpl<TResult>(clientSession, name, collectionNamesOnly, resultClass, codecRegistry,
ReadPreference.primary(), executor, retryReads);
}
@Override
public MongoCollection<Document> getCollection(final String collectionName) {
return getCollection(collectionName, Document.class);
}
@Override
public <TDocument> MongoCollection<TDocument> getCollection(final String collectionName, final Class<TDocument> documentClass) {
return new MongoCollectionImpl<TDocument>(new MongoNamespace(name, collectionName), documentClass, codecRegistry, readPreference,
writeConcern, retryWrites, retryReads, readConcern, executor);
}
@Override
public void runCommand(final Bson command, final SingleResultCallback<Document> callback) {
runCommand(command, Document.class, callback);
}
@Override
public void runCommand(final Bson command, final ReadPreference readPreference, final SingleResultCallback<Document> callback) {
runCommand(command, readPreference, Document.class, callback);
}
@Override
public <TResult> void runCommand(final Bson command, final Class<TResult> resultClass,
final SingleResultCallback<TResult> callback) {
runCommand(command, ReadPreference.primary(), resultClass, callback);
}
@Override
public <TResult> void runCommand(final Bson command, final ReadPreference readPreference, final Class<TResult> resultClass,
final SingleResultCallback<TResult> callback) {
executeCommand(null, command, readPreference, resultClass, callback);
}
@Override
public void runCommand(final ClientSession clientSession, final Bson command, final SingleResultCallback<Document> callback) {
runCommand(clientSession, command, Document.class, callback);
}
@Override
public void runCommand(final ClientSession clientSession, final Bson command, final ReadPreference readPreference,
final SingleResultCallback<Document> callback) {
runCommand(clientSession, command, readPreference, Document.class, callback);
}
@Override
public <TResult> void runCommand(final ClientSession clientSession, final Bson command, final Class<TResult> resultClass,
final SingleResultCallback<TResult> callback) {
runCommand(clientSession, command, ReadPreference.primary(), resultClass, callback);
}
@Override
public <TResult> void runCommand(final ClientSession clientSession, final Bson command, final ReadPreference readPreference,
final Class<TResult> resultClass, final SingleResultCallback<TResult> callback) {
notNull("clientSession", clientSession);
executeCommand(clientSession, command, readPreference, resultClass, callback);
}
private <TResult> void executeCommand(@Nullable final ClientSession clientSession, final Bson command,
final ReadPreference readPreference, final Class<TResult> resultClass,
final SingleResultCallback<TResult> callback) {
notNull("command", command);
notNull("readPreference", readPreference);
if (clientSession != null && clientSession.hasActiveTransaction() && !readPreference.equals(ReadPreference.primary())) {
throw new MongoClientException("Read preference in a transaction must be primary");
}
executor.execute(new CommandReadOperation<TResult>(getName(), toBsonDocument(command), codecRegistry.get(resultClass)),
readPreference, readConcern, clientSession, callback);
}
@Override
public void drop(final SingleResultCallback<Void> callback) {
executeDrop(null, callback);
}
@Override
public void drop(final ClientSession clientSession, final SingleResultCallback<Void> callback) {
notNull("clientSession", clientSession);
executeDrop(clientSession, callback);
}
private void executeDrop(@Nullable final ClientSession clientSession, final SingleResultCallback<Void> callback) {
executor.execute(new DropDatabaseOperation(name, writeConcern), readConcern, clientSession, callback);
}
@Override
public void createCollection(final String collectionName, final SingleResultCallback<Void> callback) {
executeCreateCollection(null, collectionName, new CreateCollectionOptions(), callback);
}
@Override
public void createCollection(final String collectionName, final CreateCollectionOptions createCollectionOptions,
final SingleResultCallback<Void> callback) {
executeCreateCollection(null, collectionName, createCollectionOptions, callback);
}
@Override
public void createCollection(final ClientSession clientSession, final String collectionName,
final SingleResultCallback<Void> callback) {
createCollection(clientSession, collectionName, new CreateCollectionOptions(), callback);
}
@Override
public void createCollection(final ClientSession clientSession, final String collectionName, final CreateCollectionOptions options,
final SingleResultCallback<Void> callback) {
notNull("clientSession", clientSession);
executeCreateCollection(clientSession, collectionName, options, callback);
}
@SuppressWarnings("deprecation")
private void executeCreateCollection(@Nullable final ClientSession clientSession, final String collectionName,
final CreateCollectionOptions options, final SingleResultCallback<Void> callback) {
CreateCollectionOperation operation = new CreateCollectionOperation(name, collectionName, writeConcern)
.capped(options.isCapped())
.sizeInBytes(options.getSizeInBytes())
.autoIndex(options.isAutoIndex())
.maxDocuments(options.getMaxDocuments())
.usePowerOf2Sizes(options.isUsePowerOf2Sizes())
.storageEngineOptions(toBsonDocument(options.getStorageEngineOptions()))
.collation(options.getCollation());
IndexOptionDefaults indexOptionDefaults = options.getIndexOptionDefaults();
Bson storageEngine = indexOptionDefaults.getStorageEngine();
if (storageEngine != null) {
operation.indexOptionDefaults(new BsonDocument("storageEngine", toBsonDocument(storageEngine)));
}
ValidationOptions validationOptions = options.getValidationOptions();
Bson validator = validationOptions.getValidator();
if (validator != null) {
operation.validator(toBsonDocument(validator));
}
if (validationOptions.getValidationLevel() != null) {
operation.validationLevel(validationOptions.getValidationLevel());
}
if (validationOptions.getValidationAction() != null) {
operation.validationAction(validationOptions.getValidationAction());
}
executor.execute(operation, readConcern, clientSession, callback);
}
@Override
public void createView(final String viewName, final String viewOn, final List<? extends Bson> pipeline,
final SingleResultCallback<Void> callback) {
createView(viewName, viewOn, pipeline, new CreateViewOptions(), callback);
}
@Override
public void createView(final String viewName, final String viewOn, final List<? extends Bson> pipeline,
final CreateViewOptions createViewOptions, final SingleResultCallback<Void> callback) {
executeCreateView(null, viewName, viewOn, pipeline, createViewOptions, callback);
}
@Override
public void createView(final ClientSession clientSession, final String viewName, final String viewOn,
final List<? extends Bson> pipeline, final SingleResultCallback<Void> callback) {
createView(clientSession, viewName, viewOn, pipeline, new CreateViewOptions(), callback);
}
@Override
public void createView(final ClientSession clientSession, final String viewName, final String viewOn,
final List<? extends Bson> pipeline, final CreateViewOptions createViewOptions,
final SingleResultCallback<Void> callback) {
notNull("clientSession", clientSession);
executeCreateView(clientSession, viewName, viewOn, pipeline, createViewOptions, callback);
}
@Override
public ChangeStreamIterable<Document> watch() {
return watch(Collections.<Bson>emptyList());
}
@Override
public <TResult> ChangeStreamIterable<TResult> watch(final Class<TResult> resultClass) {
return watch(Collections.<Bson>emptyList(), resultClass);
}
@Override
public ChangeStreamIterable<Document> watch(final List<? extends Bson> pipeline) {
return watch(pipeline, Document.class);
}
@Override
public <TResult> ChangeStreamIterable<TResult> watch(final List<? extends Bson> pipeline, final Class<TResult> resultClass) {
return createChangeStreamIterable(null, pipeline, resultClass);
}
@Override
public ChangeStreamIterable<Document> watch(final ClientSession clientSession) {
return watch(clientSession, Collections.<Bson>emptyList(), Document.class);
}
@Override
public <TResult> ChangeStreamIterable<TResult> watch(final ClientSession clientSession, final Class<TResult> resultClass) {
return watch(clientSession, Collections.<Bson>emptyList(), resultClass);
}
@Override
public ChangeStreamIterable<Document> watch(final ClientSession clientSession, final List<? extends Bson> pipeline) {
return watch(clientSession, pipeline, Document.class);
}
@Override
public <TResult> ChangeStreamIterable<TResult> watch(final ClientSession clientSession, final List<? extends Bson> pipeline,
final Class<TResult> resultClass) {
notNull("clientSession", clientSession);
return createChangeStreamIterable(clientSession, pipeline, resultClass);
}
@Override
public AggregateIterable<Document> aggregate(final List<? extends Bson> pipeline) {
return aggregate(pipeline, Document.class);
}
@Override
public <TResult> AggregateIterable<TResult> aggregate(final List<? extends Bson> pipeline, final Class<TResult> resultClass) {
return createAggregateIterable(null, pipeline, resultClass);
}
@Override
public AggregateIterable<Document> aggregate(final ClientSession clientSession, final List<? extends Bson> pipeline) {
return aggregate(clientSession, pipeline, Document.class);
}
@Override
public <TResult> AggregateIterable<TResult> aggregate(final ClientSession clientSession, final List<? extends Bson> pipeline,
final Class<TResult> resultClass) {
return createAggregateIterable(notNull("clientSession", clientSession), pipeline, resultClass);
}
private <TResult> AggregateIterable<TResult> createAggregateIterable(@Nullable final ClientSession clientSession,
final List<? extends Bson> pipeline,
final Class<TResult> resultClass) {
return new AggregateIterableImpl<Document, TResult>(clientSession, name, Document.class, resultClass, codecRegistry,
readPreference, readConcern, writeConcern, executor, pipeline, AggregationLevel.DATABASE, retryReads);
}
private <TResult> ChangeStreamIterable<TResult> createChangeStreamIterable(@Nullable final ClientSession clientSession,
final List<? extends Bson> pipeline,
final Class<TResult> resultClass) {
return new ChangeStreamIterableImpl<TResult>(clientSession, name, codecRegistry, readPreference,
readConcern, executor, pipeline, resultClass, ChangeStreamLevel.DATABASE, retryReads);
}
private void executeCreateView(@Nullable final ClientSession clientSession, final String viewName, final String viewOn,
final List<? extends Bson> pipeline, final CreateViewOptions createViewOptions,
final SingleResultCallback<Void> callback) {
notNull("createViewOptions", createViewOptions);
executor.execute(new CreateViewOperation(name, viewName, viewOn, createBsonDocumentList(pipeline), writeConcern)
.collation(createViewOptions.getCollation()), readConcern, clientSession, callback);
}
private List<BsonDocument> createBsonDocumentList(final List<? extends Bson> pipeline) {
notNull("pipeline", pipeline);
List<BsonDocument> bsonDocumentPipeline = new ArrayList<BsonDocument>(pipeline.size());
for (Bson obj : pipeline) {
if (obj == null) {
throw new IllegalArgumentException("pipeline can not contain a null value");
}
bsonDocumentPipeline.add(obj.toBsonDocument(BsonDocument.class, codecRegistry));
}
return bsonDocumentPipeline;
}
@Nullable
private BsonDocument toBsonDocument(@Nullable final Bson document) {
return document == null ? null : document.toBsonDocument(BsonDocument.class, codecRegistry);
}
}