package com.mongodb.client.internal;
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.client.AggregateIterable;
import com.mongodb.client.ChangeStreamIterable;
import com.mongodb.client.ClientSession;
import com.mongodb.client.ListCollectionsIterable;
import com.mongodb.client.MongoCollection;
import com.mongodb.client.MongoDatabase;
import com.mongodb.client.MongoIterable;
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.client.model.AggregationLevel;
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.UuidRepresentation;
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;
import static org.bson.internal.CodecRegistryHelper.createRegistry;
public 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;
private UuidRepresentation uuidRepresentation;
public MongoDatabaseImpl(final String name, final CodecRegistry codecRegistry, final ReadPreference readPreference,
final WriteConcern writeConcern, final boolean retryWrites, final boolean retryReads,
final ReadConcern readConcern, final UuidRepresentation uuidRepresentation, 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.uuidRepresentation = notNull("uuidRepresentation", uuidRepresentation);
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, createRegistry(codecRegistry, uuidRepresentation), readPreference, writeConcern, retryWrites,
retryReads, readConcern, uuidRepresentation, executor);
}
@Override
public MongoDatabase withReadPreference(final ReadPreference readPreference) {
return new MongoDatabaseImpl(name, codecRegistry, readPreference, writeConcern, retryWrites, retryReads, readConcern,
uuidRepresentation, executor);
}
@Override
public MongoDatabase withWriteConcern(final WriteConcern writeConcern) {
return new MongoDatabaseImpl(name, codecRegistry, readPreference, writeConcern, retryWrites, retryReads, readConcern,
uuidRepresentation, executor);
}
@Override
public MongoDatabase withReadConcern(final ReadConcern readConcern) {
return new MongoDatabaseImpl(name, codecRegistry, readPreference, writeConcern, retryWrites, retryReads, readConcern,
uuidRepresentation, executor);
}
@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, uuidRepresentation, executor);
}
@Override
public Document runCommand(final Bson command) {
return runCommand(command, Document.class);
}
@Override
public Document runCommand(final Bson command, final ReadPreference readPreference) {
return runCommand(command, readPreference, Document.class);
}
@Override
public <TResult> TResult runCommand(final Bson command, final Class<TResult> resultClass) {
return runCommand(command, ReadPreference.primary(), resultClass);
}
@Override
public <TResult> TResult runCommand(final Bson command, final ReadPreference readPreference, final Class<TResult> resultClass) {
return executeCommand(null, command, readPreference, resultClass);
}
@Override
public Document runCommand(final ClientSession clientSession, final Bson command) {
return runCommand(clientSession, command, ReadPreference.primary(), Document.class);
}
@Override
public Document runCommand(final ClientSession clientSession, final Bson command, final ReadPreference readPreference) {
return runCommand(clientSession, command, readPreference, Document.class);
}
@Override
public <TResult> TResult runCommand(final ClientSession clientSession, final Bson command, final Class<TResult> resultClass) {
return runCommand(clientSession, command, ReadPreference.primary(), resultClass);
}
@Override
public <TResult> TResult runCommand(final ClientSession clientSession, final Bson command, final ReadPreference readPreference,
final Class<TResult> resultClass) {
notNull("clientSession", clientSession);
return executeCommand(clientSession, command, readPreference, resultClass);
}
private <TResult> TResult executeCommand(@Nullable final ClientSession clientSession, final Bson command,
final ReadPreference readPreference, final Class<TResult> resultClass) {
notNull("readPreference", readPreference);
if (clientSession != null && clientSession.hasActiveTransaction() && !readPreference.equals(ReadPreference.primary())) {
throw new MongoClientException("Read preference in a transaction must be primary");
}
return executor.execute(new CommandReadOperation<TResult>(getName(), toBsonDocument(command), codecRegistry.get(resultClass)),
readPreference, readConcern, clientSession);
}
@Override
public void drop() {
executeDrop(null);
}
@Override
public void drop(final ClientSession clientSession) {
notNull("clientSession", clientSession);
executeDrop(clientSession);
}
private void executeDrop(@Nullable final ClientSession clientSession) {
executor.execute(new DropDatabaseOperation(name, getWriteConcern()), readConcern, clientSession);
}
@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 MongoIterables.listCollectionsOf(clientSession, name, collectionNamesOnly, resultClass, codecRegistry,
ReadPreference.primary(), executor, retryReads);
}
@Override
public void createCollection(final String collectionName) {
createCollection(collectionName, new CreateCollectionOptions());
}
@Override
public void createCollection(final String collectionName, final CreateCollectionOptions createCollectionOptions) {
executeCreateCollection(null, collectionName, createCollectionOptions);
}
@Override
public void createCollection(final ClientSession clientSession, final String collectionName) {
createCollection(clientSession, collectionName, new CreateCollectionOptions());
}
@Override
public void createCollection(final ClientSession clientSession, final String collectionName,
final CreateCollectionOptions createCollectionOptions) {
notNull("clientSession", clientSession);
executeCreateCollection(clientSession, collectionName, createCollectionOptions);
}
@SuppressWarnings("deprecation")
private void executeCreateCollection(@Nullable final ClientSession clientSession, final String collectionName,
final CreateCollectionOptions createCollectionOptions) {
CreateCollectionOperation operation = new CreateCollectionOperation(name, collectionName, writeConcern)
.collation(createCollectionOptions.getCollation())
.capped(createCollectionOptions.isCapped())
.sizeInBytes(createCollectionOptions.getSizeInBytes())
.autoIndex(createCollectionOptions.isAutoIndex())
.maxDocuments(createCollectionOptions.getMaxDocuments())
.usePowerOf2Sizes(createCollectionOptions.isUsePowerOf2Sizes())
.storageEngineOptions(toBsonDocument(createCollectionOptions.getStorageEngineOptions()));
IndexOptionDefaults indexOptionDefaults = createCollectionOptions.getIndexOptionDefaults();
Bson storageEngine = indexOptionDefaults.getStorageEngine();
if (storageEngine != null) {
operation.indexOptionDefaults(new BsonDocument("storageEngine", toBsonDocument(storageEngine)));
}
ValidationOptions validationOptions = createCollectionOptions.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);
}
@Override
public void createView(final String viewName, final String viewOn, final List<? extends Bson> pipeline) {
createView(viewName, viewOn, pipeline, new CreateViewOptions());
}
@Override
public void createView(final String viewName, final String viewOn, final List<? extends Bson> pipeline,
final CreateViewOptions createViewOptions) {
executeCreateView(null, viewName, viewOn, pipeline, createViewOptions);
}
@Override
public void createView(final ClientSession clientSession, final String viewName, final String viewOn,
final List<? extends Bson> pipeline) {
createView(clientSession, viewName, viewOn, pipeline, new CreateViewOptions());
}
@Override
public void createView(final ClientSession clientSession, final String viewName, final String viewOn,
final List<? extends Bson> pipeline, final CreateViewOptions createViewOptions) {
notNull("clientSession", clientSession);
executeCreateView(clientSession, viewName, viewOn, pipeline, createViewOptions);
}
@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) {
notNull("clientSession", clientSession);
return createAggregateIterable(clientSession, pipeline, resultClass);
}
private <TResult> AggregateIterable<TResult> createAggregateIterable(@Nullable final ClientSession clientSession,
final List<? extends Bson> pipeline,
final Class<TResult> resultClass) {
return MongoIterables.aggregateOf(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 MongoIterables.changeStreamOf(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) {
notNull("createViewOptions", createViewOptions);
executor.execute(new CreateViewOperation(name, viewName, viewOn, createBsonDocumentList(pipeline), writeConcern)
.collation(createViewOptions.getCollation()),
readConcern, clientSession);
}
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);
}
}