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

io.vertx.ext.mongo.impl.MongoClientImpl Maven / Gradle / Ivy

/*
 * Copyright 2014 Red Hat, Inc.
 *
 *  All rights reserved. This program and the accompanying materials
 *  are made available under the terms of the Eclipse Public License v1.0
 *  and Apache License v2.0 which accompanies this distribution.
 *
 *  The Eclipse Public License is available at
 *  http://www.eclipse.org/legal/epl-v10.html
 *
 *  The Apache License v2.0 is available at
 *  http://www.opensource.org/licenses/apache2.0.php
 *
 *  You may elect to redistribute this code under either of these licenses.
 */

package io.vertx.ext.mongo.impl;

import com.mongodb.MongoClientSettings;
import com.mongodb.WriteConcern;
import com.mongodb.bulk.BulkWriteResult;
import com.mongodb.client.model.*;
import com.mongodb.client.model.changestream.ChangeStreamDocument;
import com.mongodb.client.model.changestream.FullDocument;
import com.mongodb.client.result.DeleteResult;
import com.mongodb.client.result.UpdateResult;
import com.mongodb.reactivestreams.client.*;
import com.mongodb.reactivestreams.client.gridfs.GridFSBucket;
import com.mongodb.reactivestreams.client.gridfs.GridFSBuckets;
import io.vertx.codegen.annotations.GenIgnore;
import io.vertx.codegen.annotations.Nullable;
import io.vertx.core.*;
import io.vertx.core.impl.ContextInternal;
import io.vertx.core.impl.VertxInternal;
import io.vertx.core.impl.future.PromiseInternal;
import io.vertx.core.json.JsonArray;
import io.vertx.core.json.JsonObject;
import io.vertx.core.shareddata.LocalMap;
import io.vertx.core.shareddata.Shareable;
import io.vertx.core.streams.ReadStream;
import io.vertx.ext.mongo.BulkWriteOptions;
import io.vertx.ext.mongo.CountOptions;
import io.vertx.ext.mongo.CreateCollectionOptions;
import io.vertx.ext.mongo.IndexModel;
import io.vertx.ext.mongo.IndexOptions;
import io.vertx.ext.mongo.MongoClient;
import io.vertx.ext.mongo.UpdateOptions;
import io.vertx.ext.mongo.*;
import io.vertx.ext.mongo.impl.codec.json.JsonObjectCodec;
import io.vertx.ext.mongo.impl.config.MongoClientOptionsParser;
import org.bson.BsonDocument;
import org.bson.conversions.Bson;
import org.bson.types.ObjectId;
import org.reactivestreams.Publisher;

import java.io.IOException;
import java.util.*;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;

import static io.vertx.ext.mongo.impl.Utils.ID_FIELD;
import static io.vertx.ext.mongo.impl.Utils.setHandler;
import static java.util.Objects.requireNonNull;

/**
 * The implementation of the {@link io.vertx.ext.mongo.MongoClient}. This implementation is based on the async driver
 * provided by Mongo.
 *
 * @author Tim Fox
 */
public class MongoClientImpl implements io.vertx.ext.mongo.MongoClient, Closeable {
  private static final UpdateOptions DEFAULT_UPDATE_OPTIONS = new UpdateOptions();
  private static final FindOptions DEFAULT_FIND_OPTIONS = new FindOptions();
  private static final AggregateOptions DEFAULT_AGGREGATE_OPTIONS = new AggregateOptions();
  private static final BulkWriteOptions DEFAULT_BULK_WRITE_OPTIONS = new BulkWriteOptions();

  private static final String DS_LOCAL_MAP_NAME = "__vertx.MongoClient.datasources";
  public static final String COLLECTION_CANNOT_BE_NULL = "collection cannot be null";
  public static final String QUERY_CANNOT_BE_NULL = "query cannot be null";
  public static final String FIELD_NAME_CANNOT_BE_NULL = "fieldName cannot be null";
  public static final String UPDATE_CANNOT_BE_NULL = "update cannot be null";
  public static final String OPTIONS_CANNOT_BE_NULL = "options cannot be null";
  public static final String PIPELINE_CANNOT_BE_NULL = "pipeline cannot be null";
  public static final String FIND_OPTIONS_CANNOT_BE_NULL = "find options cannot be null";

  private final VertxInternal vertx;
  private final ContextInternal creatingContext;
  protected com.mongodb.reactivestreams.client.MongoClient mongo;

  private final MongoHolder holder;
  private final boolean useObjectId;

  public MongoClientImpl(Vertx vertx, JsonObject config, String dataSourceName) {
    Objects.requireNonNull(vertx);
    Objects.requireNonNull(config);
    Objects.requireNonNull(dataSourceName);
    this.vertx = (VertxInternal) vertx;
    this.creatingContext = this.vertx.getOrCreateContext();
    this.holder = lookupHolder(dataSourceName, config);
    this.mongo = holder.mongo(vertx);
    this.useObjectId = config.getBoolean("useObjectId", false);

    creatingContext.addCloseHook(this);
  }

  public MongoClientImpl(Vertx vertx, JsonObject config, String dataSourceName, MongoClientSettings settings) {
    Objects.requireNonNull(vertx);
    Objects.requireNonNull(config);
    Objects.requireNonNull(dataSourceName);
    Objects.requireNonNull(settings);
    this.vertx = (VertxInternal) vertx;
    this.creatingContext = this.vertx.getOrCreateContext();
    this.holder = lookupHolder(dataSourceName, config);
    this.mongo = holder.mongo(vertx, settings);
    this.useObjectId = config.getBoolean("useObjectId", false);

    creatingContext.addCloseHook(this);
  }

  @GenIgnore
  public static  DistinctPublisher setDistinctOptions(DistinctPublisher distinctPublisher, DistinctOptions distinctOptions) {
    if (distinctOptions != null && distinctOptions.getCollation() != null) {
      distinctPublisher.collation(distinctOptions.getCollation().toMongoDriverObject());
    }
    return distinctPublisher;
  }

  @Override
  public void close(Promise completionHandler) {
    holder.close();
    completionHandler.complete();
  }

  @Override
  public Future close() {
    holder.close();
    creatingContext.removeCloseHook(this);
    return vertx.getOrCreateContext().succeededFuture();
  }

  @Override
  public MongoClient save(String collection, JsonObject document, Handler> resultHandler) {
    Future future = save(collection, document);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable String> save(String collection, JsonObject document) {
    return saveWithOptions(collection, document, null);
  }

  @Override
  public MongoClient saveWithOptions(String collection, JsonObject document, @Nullable WriteOption writeOption, Handler> resultHandler) {
    Future future = saveWithOptions(collection, document, writeOption);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable String> saveWithOptions(String collection, JsonObject document, @Nullable WriteOption writeOption) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(document, "document cannot be null");

    MongoCollection coll = getCollection(collection, writeOption);
    Object id = document.getValue(ID_FIELD);

    if (id == null) {
      Promise promise = vertx.promise();
      coll.insertOne(document).subscribe(new CompletionSubscriber<>(promise));
      return promise.future().map(v -> useObjectId ? document.getJsonObject(ID_FIELD).getString(JsonObjectCodec.OID_FIELD) : document.getString(ID_FIELD));
    }

    JsonObject filter = new JsonObject();
    JsonObject encodedDocument = encodeKeyWhenUseObjectId(document);
    filter.put(ID_FIELD, encodedDocument.getValue(ID_FIELD));

    ReplaceOptions replaceOptions = new ReplaceOptions().upsert(true);

    Promise promise = vertx.promise();
    coll.replaceOne(wrap(filter), encodedDocument, replaceOptions).subscribe(new CompletionSubscriber<>(promise));
    return promise.future().mapEmpty();
  }

  @Override
  public MongoClient insert(String collection, JsonObject document, Handler> resultHandler) {
    Future future = insert(collection, document);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable String> insert(String collection, JsonObject document) {
    return insertWithOptions(collection, document, null);
  }

  @Override
  public MongoClient insertWithOptions(String collection, JsonObject document, @Nullable WriteOption writeOption, Handler> resultHandler) {
    Future future = insertWithOptions(collection, document, writeOption);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable String> insertWithOptions(String collection, JsonObject document, @Nullable WriteOption writeOption) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(document, "document cannot be null");

    JsonObject encodedDocument = encodeKeyWhenUseObjectId(document);
    boolean hasCustomId = document.containsKey(ID_FIELD);

    MongoCollection coll = getCollection(collection, writeOption);

    Promise promise = vertx.promise();
    coll.insertOne(encodedDocument).subscribe(new CompletionSubscriber<>(promise));
    return promise.future().map(v -> hasCustomId ? null : decodeKeyWhenUseObjectId(encodedDocument).getString(ID_FIELD));
  }

  @Override
  public MongoClient updateCollection(String collection, JsonObject query, JsonObject update, Handler> resultHandler) {
    Future future = updateCollection(collection, query, update);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientUpdateResult> updateCollection(String collection, JsonObject query, JsonObject update) {
    return updateCollectionWithOptions(collection, query, update, DEFAULT_UPDATE_OPTIONS);
  }

  @Override
  public MongoClient updateCollection(String collection, JsonObject query, JsonArray update, Handler> resultHandler) {
    Future future = updateCollection(collection, query, update);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientUpdateResult> updateCollection(String collection, JsonObject query, JsonArray update) {
    return updateCollectionWithOptions(collection, query, update, DEFAULT_UPDATE_OPTIONS);
  }

  @Override
  public MongoClient updateCollectionWithOptions(String collection, JsonObject query, JsonObject update, UpdateOptions options, Handler> resultHandler) {
    Future future = updateCollectionWithOptions(collection, query, update, options);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientUpdateResult> updateCollectionWithOptions(String collection, JsonObject query, JsonObject update, UpdateOptions options) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);
    requireNonNull(update, UPDATE_CANNOT_BE_NULL);
    requireNonNull(options, OPTIONS_CANNOT_BE_NULL);

    MongoCollection coll = getCollection(collection, options.getWriteOption());
    Bson bquery = wrap(deepEncodeKeyWhenUseObjectId(query));
    Bson bupdate = wrap(encodeKeyWhenUseObjectId(generateIdIfNeeded(query, update, options)));

    com.mongodb.client.model.UpdateOptions updateOptions = new com.mongodb.client.model.UpdateOptions().upsert(options.isUpsert());
    JsonArray arrayFilters = options.getArrayFilters();
    if (arrayFilters != null && !arrayFilters.isEmpty()) {
      List bArrayFilters = new ArrayList<>(arrayFilters.size());
      for (int i = 0; i < arrayFilters.size(); i++) {
        bArrayFilters.add(wrap(arrayFilters.getJsonObject(i)));
      }
      updateOptions.arrayFilters(bArrayFilters);
    }
    if (options.getHint() != null) {
      updateOptions.hint(wrap(options.getHint()));
    }
    if (options.getHintString() != null && !options.getHintString().isEmpty()) {
      updateOptions.hintString(options.getHintString());
    }
    if (options.getCollation() != null) {
      updateOptions.collation(options.getCollation().toMongoDriverObject());
    }

    Publisher publisher;
    if (options.isMulti()) {
      publisher = coll.updateMany(bquery, bupdate, updateOptions);
    } else {
      publisher = coll.updateOne(bquery, bupdate, updateOptions);
    }

    Promise promise = vertx.promise();
    publisher.subscribe(new SingleResultSubscriber<>(promise));
    return promise.future().map(Utils::toMongoClientUpdateResult);
  }

  @Override
  public MongoClient updateCollectionWithOptions(String collection, JsonObject query, JsonArray update, UpdateOptions options, Handler> resultHandler) {
    Future future = updateCollectionWithOptions(collection, query, update, options);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientUpdateResult> updateCollectionWithOptions(String collection, JsonObject query, JsonArray pipeline, UpdateOptions options) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);
    requireNonNull(pipeline, PIPELINE_CANNOT_BE_NULL);
    requireNonNull(options, OPTIONS_CANNOT_BE_NULL);

    MongoCollection coll = getCollection(collection, options.getWriteOption());
    Bson bquery = wrap(deepEncodeKeyWhenUseObjectId(query));
    List bpipeline = new ArrayList<>(pipeline.size());
    for (int i=0 ; i bArrayFilters = new ArrayList<>(arrayFilters.size());
      for (int i = 0; i < arrayFilters.size(); i++) {
        bArrayFilters.add(wrap(arrayFilters.getJsonObject(i)));
      }
      updateOptions.arrayFilters(bArrayFilters);
    }
    if (options.getHint() != null) {
      updateOptions.hint(wrap(options.getHint()));
    }
    if (options.getHintString() != null && !options.getHintString().isEmpty()) {
      updateOptions.hintString(options.getHintString());
    }
    if (options.getCollation() != null) {
      updateOptions.collation(options.getCollation().toMongoDriverObject());
    }

    Publisher publisher = coll.updateMany(bquery, bpipeline, updateOptions);

    Promise promise = vertx.promise();
    publisher.subscribe(new SingleResultSubscriber<>(promise));
    return promise.future().map(Utils::toMongoClientUpdateResult);
  }

  private JsonObject generateIdIfNeeded(JsonObject query, JsonObject update, UpdateOptions options) {
    if (options.isUpsert() && !update.containsKey(ID_FIELD) && !useObjectId) {
      JsonObject setId = update.getJsonObject("$setOnInsert", new JsonObject());
      String id;

      //This seems odd, but if you filter based on _id, mongo expects the generated _id to match
      if (query.containsKey(ID_FIELD)) {
        id = query.getString(ID_FIELD);
      } else {
        id = JsonObjectCodec.generateHexObjectId();
      }
      setId.put(ID_FIELD, id);
      update.put("$setOnInsert", setId);
    }
    return update;
  }

  @Override
  public MongoClient replaceDocuments(String collection, JsonObject query, JsonObject replace, Handler> resultHandler) {
    Future future = replaceDocumentsWithOptions(collection, query, replace, DEFAULT_UPDATE_OPTIONS);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientUpdateResult> replaceDocuments(String collection, JsonObject query, JsonObject replace) {
    return replaceDocumentsWithOptions(collection, query, replace, DEFAULT_UPDATE_OPTIONS);
  }

  @Override
  public MongoClient replaceDocumentsWithOptions(String collection, JsonObject query, JsonObject replace, UpdateOptions options, Handler> resultHandler) {
    Future future = replaceDocumentsWithOptions(collection, query, replace, options);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientUpdateResult> replaceDocumentsWithOptions(String collection, JsonObject query, JsonObject replace, UpdateOptions options) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);
    requireNonNull(replace, UPDATE_CANNOT_BE_NULL);
    requireNonNull(options, OPTIONS_CANNOT_BE_NULL);

    MongoCollection coll = getCollection(collection, options.getWriteOption());
    Bson bquery = wrap(deepEncodeKeyWhenUseObjectId(query));
    com.mongodb.client.model.ReplaceOptions replaceOptions = new com.mongodb.client.model.ReplaceOptions().upsert(options.isUpsert());
    if (options.getHint() != null) {
      replaceOptions.hint(wrap(options.getHint()));
    }
    if (options.getHintString() != null && !options.getHintString().isEmpty()) {
      replaceOptions.hintString(options.getHintString());
    }
    if (options.getCollation() != null) {
      replaceOptions.collation(options.getCollation().toMongoDriverObject());
    }
    Promise promise = vertx.promise();
    coll.replaceOne(bquery, encodeKeyWhenUseObjectId(replace), replaceOptions).subscribe(new SingleResultSubscriber<>(promise));
    return promise.future().map(Utils::toMongoClientUpdateResult);
  }

  @Override
  public MongoClient find(String collection, JsonObject query, Handler>> resultHandler) {
    Future> future = find(collection, query);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future> find(String collection, JsonObject query) {
    return findWithOptions(collection, query, DEFAULT_FIND_OPTIONS);
  }

  @Override
  public MongoClient findWithOptions(String collection, JsonObject query, FindOptions options, Handler>> resultHandler) {
    Future> future = findWithOptions(collection, query, options);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future> findWithOptions(String collection, JsonObject query, FindOptions options) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);

    Promise> promise = vertx.promise();
    doFind(collection, deepEncodeKeyWhenUseObjectId(query), options)
      .subscribe(new MappingAndBufferingSubscriber<>(this::decodeKeyWhenUseObjectId, promise));
    return promise.future();
  }

  @Override
  public ReadStream findBatch(String collection, JsonObject query) {
    return findBatchWithOptions(collection, query, DEFAULT_FIND_OPTIONS);
  }

  @Override
  public ReadStream findBatchWithOptions(String collection, JsonObject query, FindOptions options) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);
    FindPublisher view = doFind(collection, query, options);
    return new PublisherAdapter<>(vertx.getOrCreateContext(), view, options.getBatchSize());
  }

  @Override
  public MongoClient findOne(String collection, JsonObject query, @Nullable JsonObject fields, Handler> resultHandler) {
    Future future = findOne(collection, query, fields);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable JsonObject> findOne(String collection, JsonObject query, @Nullable JsonObject fields) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);

    JsonObject encodedQuery = deepEncodeKeyWhenUseObjectId(query);

    Bson bquery = wrap(encodedQuery);
    Bson bfields = wrap(fields);
    Promise promise = vertx.promise();
    getCollection(collection).find(bquery).projection(bfields).first().subscribe(new SingleResultSubscriber<>(promise));
    return promise.future().map(object -> object == null ? null : decodeKeyWhenUseObjectId(object));
  }

  @Override
  public MongoClient findOneAndUpdate(String collection, JsonObject query, JsonObject update, Handler> resultHandler) {
    Future future = findOneAndUpdate(collection, query, update);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable JsonObject> findOneAndUpdate(String collection, JsonObject query, JsonObject update) {
    return findOneAndUpdateWithOptions(collection, query, update, DEFAULT_FIND_OPTIONS, DEFAULT_UPDATE_OPTIONS);
  }

  @Override
  public MongoClient findOneAndUpdateWithOptions(String collection, JsonObject query, JsonObject update, FindOptions findOptions, UpdateOptions updateOptions, Handler> resultHandler) {
    Future future = findOneAndUpdateWithOptions(collection, query, update, findOptions, updateOptions);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable JsonObject> findOneAndUpdateWithOptions(String collection, JsonObject query, JsonObject update, FindOptions findOptions, UpdateOptions updateOptions) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);
    requireNonNull(update, UPDATE_CANNOT_BE_NULL);
    requireNonNull(findOptions, FIND_OPTIONS_CANNOT_BE_NULL);
    requireNonNull(updateOptions, "update options cannot be null");

    JsonObject encodedQuery = deepEncodeKeyWhenUseObjectId(query);

    Bson bquery = wrap(encodedQuery);
    Bson bupdate = wrap(update);
    FindOneAndUpdateOptions foauOptions = new FindOneAndUpdateOptions();
    foauOptions.sort(wrap(findOptions.getSort()));
    foauOptions.projection(wrap(findOptions.getFields()));
    foauOptions.upsert(updateOptions.isUpsert());
    foauOptions.returnDocument(updateOptions.isReturningNewDocument() ? ReturnDocument.AFTER : ReturnDocument.BEFORE);
    JsonArray arrayFilters = updateOptions.getArrayFilters();
    if (arrayFilters != null && !arrayFilters.isEmpty()) {
      List bArrayFilters = new ArrayList<>(arrayFilters.size());
      for (int i = 0; i < arrayFilters.size(); i++) {
        bArrayFilters.add(wrap(arrayFilters.getJsonObject(i)));
      }
      foauOptions.arrayFilters(bArrayFilters);
    }
    if (findOptions.getHint() != null) {
      foauOptions.hint(wrap(findOptions.getHint()));
    }
    if (findOptions.getHintString() != null && !findOptions.getHintString().isEmpty()) {
      foauOptions.hintString(findOptions.getHintString());
    }
    if (updateOptions.getHint() != null) {
      foauOptions.hint(wrap(updateOptions.getHint()));
    }
    if (updateOptions.getHintString() != null && !updateOptions.getHintString().isEmpty()) {
      foauOptions.hintString(updateOptions.getHintString());
    }
    if(findOptions.getCollation() != null) {
      foauOptions.collation(findOptions.getCollation().toMongoDriverObject());
    }
    if(updateOptions.getCollation() != null) {
      foauOptions.collation(updateOptions.getCollation().toMongoDriverObject());
    }

    MongoCollection coll = getCollection(collection);
    Promise promise = vertx.promise();
    coll.findOneAndUpdate(bquery, bupdate, foauOptions).subscribe(new SingleResultSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public MongoClient findOneAndReplace(String collection, JsonObject query, JsonObject replace, Handler> resultHandler) {
    Future future = findOneAndReplace(collection, query, replace);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable JsonObject> findOneAndReplace(String collection, JsonObject query, JsonObject replace) {
    return findOneAndReplaceWithOptions(collection, query, replace, DEFAULT_FIND_OPTIONS, DEFAULT_UPDATE_OPTIONS);
  }

  @Override
  public MongoClient findOneAndReplaceWithOptions(String collection, JsonObject query, JsonObject replace, FindOptions findOptions, UpdateOptions updateOptions, Handler> resultHandler) {
    Future future = findOneAndReplaceWithOptions(collection, query, replace, findOptions, updateOptions);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable JsonObject> findOneAndReplaceWithOptions(String collection, JsonObject query, JsonObject replace, FindOptions findOptions, UpdateOptions updateOptions) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);
    requireNonNull(findOptions, FIND_OPTIONS_CANNOT_BE_NULL);
    requireNonNull(updateOptions, "update options cannot be null");

    JsonObject encodedQuery = deepEncodeKeyWhenUseObjectId(query);

    Bson bquery = wrap(encodedQuery);
    FindOneAndReplaceOptions foarOptions = new FindOneAndReplaceOptions();
    foarOptions.sort(wrap(findOptions.getSort()));
    foarOptions.projection(wrap(findOptions.getFields()));
    foarOptions.upsert(updateOptions.isUpsert());
    foarOptions.returnDocument(updateOptions.isReturningNewDocument() ? ReturnDocument.AFTER : ReturnDocument.BEFORE);
    if (findOptions.getHint() != null) {
      foarOptions.hint(wrap(findOptions.getHint()));
    }
    if (findOptions.getHintString() != null && !findOptions.getHintString().isEmpty()) {
      foarOptions.hintString(findOptions.getHintString());
    }
    if (updateOptions.getHint() != null) {
      foarOptions.hint(wrap(updateOptions.getHint()));
    }
    if (updateOptions.getHintString() != null && !updateOptions.getHintString().isEmpty()) {
      foarOptions.hintString(updateOptions.getHintString());
    }

    if(findOptions.getCollation() != null) {
      foarOptions.collation(findOptions.getCollation().toMongoDriverObject());
    }

    MongoCollection coll = getCollection(collection);
    Promise promise = vertx.promise();
    coll.findOneAndReplace(bquery, replace, foarOptions).subscribe(new SingleResultSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public MongoClient findOneAndDelete(String collection, JsonObject query, Handler> resultHandler) {
    Future future = findOneAndDelete(collection, query);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable JsonObject> findOneAndDelete(String collection, JsonObject query) {
    return findOneAndDeleteWithOptions(collection, query, DEFAULT_FIND_OPTIONS);
  }

  @Override
  public MongoClient findOneAndDeleteWithOptions(String collection, JsonObject query, FindOptions findOptions, Handler> resultHandler) {
    Future future = findOneAndDeleteWithOptions(collection, query, findOptions);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable JsonObject> findOneAndDeleteWithOptions(String collection, JsonObject query, FindOptions findOptions) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);
    requireNonNull(findOptions, FIND_OPTIONS_CANNOT_BE_NULL);

    JsonObject encodedQuery = deepEncodeKeyWhenUseObjectId(query);

    Bson bquery = wrap(encodedQuery);
    FindOneAndDeleteOptions foadOptions = new FindOneAndDeleteOptions();
    foadOptions.sort(wrap(findOptions.getSort()));
    foadOptions.projection(wrap(findOptions.getFields()));

    if (findOptions.getHint() != null) {
      foadOptions.hint(wrap(findOptions.getHint()));
    }
    if (findOptions.getHintString() != null && !findOptions.getHintString().isEmpty()) {
      foadOptions.hintString(findOptions.getHintString());
    }

    if(findOptions.getCollation() != null) {
      foadOptions.collation(findOptions.getCollation().toMongoDriverObject());
    }

    MongoCollection coll = getCollection(collection);
    Promise promise = vertx.promise();
    coll.findOneAndDelete(bquery, foadOptions).subscribe(new SingleResultSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public MongoClient count(String collection, JsonObject query, Handler> resultHandler) {
    Future future = countWithOptions(collection, query, null);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public MongoClient countWithOptions(String collection, JsonObject query, CountOptions countOptions, Handler> resultHandler) {
    Future future = countWithOptions(collection, query, countOptions);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future count(String collection, JsonObject query) {
    return countWithOptions(collection, query, null);
  }

  @Override
  public Future countWithOptions(String collection, JsonObject query, CountOptions countOptions) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);

    Bson bquery = wrap(deepEncodeKeyWhenUseObjectId(query));
    MongoCollection coll = getCollection(collection);
    Promise promise = vertx.promise();
    Publisher countPublisher = countOptions != null
      ? coll.countDocuments(bquery, countOptions.toMongoDriverObject())
      : coll.countDocuments(bquery);
    countPublisher.subscribe(new SingleResultSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public MongoClient removeDocuments(String collection, JsonObject query, Handler> resultHandler) {
    Future future = removeDocuments(collection, query);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientDeleteResult> removeDocuments(String collection, JsonObject query) {
    return removeDocumentsWithOptions(collection, query, null);
  }

  @Override
  public MongoClient removeDocumentsWithOptions(String collection, JsonObject query, @Nullable WriteOption writeOption, Handler> resultHandler) {
    Future future = removeDocumentsWithOptions(collection, query, writeOption);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientDeleteResult> removeDocumentsWithOptions(String collection, JsonObject query, @Nullable WriteOption writeOption) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);

    MongoCollection coll = getCollection(collection, writeOption);
    Bson bquery = wrap(deepEncodeKeyWhenUseObjectId(query));
    Promise promise = vertx.promise();
    coll.deleteMany(bquery).subscribe(new SingleResultSubscriber<>(promise));
    return promise.future().map(Utils::toMongoClientDeleteResult);
  }

  @Override
  public MongoClient removeDocument(String collection, JsonObject query, Handler> resultHandler) {
    Future future = removeDocument(collection, query);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientDeleteResult> removeDocument(String collection, JsonObject query) {
    return removeDocumentWithOptions(collection, query, null);
  }

  @Override
  public MongoClient removeDocumentWithOptions(String collection, JsonObject query, @Nullable WriteOption writeOption, Handler> resultHandler) {
    Future future = removeDocumentWithOptions(collection, query, writeOption);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientDeleteResult> removeDocumentWithOptions(String collection, JsonObject query, @Nullable WriteOption writeOption) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);

    MongoCollection coll = getCollection(collection, writeOption);
    Bson bquery = wrap(deepEncodeKeyWhenUseObjectId(query));
    Promise promise = vertx.promise();
    coll.deleteOne(bquery).subscribe(new SingleResultSubscriber<>(promise));
    return promise.future().map(Utils::toMongoClientDeleteResult);
  }

  @Override
  public MongoClient bulkWrite(String collection, List operations, Handler> resultHandler) {
    Future future = bulkWrite(collection, operations);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientBulkWriteResult> bulkWrite(String collection, List operations) {
    return bulkWriteWithOptions(collection, operations, DEFAULT_BULK_WRITE_OPTIONS);
  }

  @Override
  public MongoClient bulkWriteWithOptions(String collection, List operations, BulkWriteOptions bulkWriteOptions, Handler> resultHandler) {
    Future future = bulkWriteWithOptions(collection, operations, bulkWriteOptions);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable MongoClientBulkWriteResult> bulkWriteWithOptions(String collection, List operations, BulkWriteOptions bulkWriteOptions) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(operations, "operations cannot be null");
    requireNonNull(bulkWriteOptions, "bulkWriteOptions cannot be null");
    MongoCollection coll = getCollection(collection, bulkWriteOptions.getWriteOption());
    List> bulkOperations = convertBulkOperations(operations);
    com.mongodb.client.model.BulkWriteOptions options = new com.mongodb.client.model.BulkWriteOptions().ordered(bulkWriteOptions.isOrdered());
    Promise promise = vertx.promise();
    coll.bulkWrite(bulkOperations, options).subscribe(new SingleResultSubscriber<>(promise));
    return promise.future().map(Utils::toMongoClientBulkWriteResult);
  }

  private List> convertBulkOperations(List operations) {
    List> result = new ArrayList<>(operations.size());
    for (BulkOperation bulkOperation : operations) {
      switch (bulkOperation.getType()) {
        case DELETE:
          Bson bsonFilter = toBson(deepEncodeKeyWhenUseObjectId(bulkOperation.getFilter()));
          DeleteOptions deleteOptions = new DeleteOptions();
          if (bulkOperation.getHint() != null) {
            deleteOptions.hint(toBson(bulkOperation.getHint()));
          }
          if (bulkOperation.getHintString() != null && !bulkOperation.getHintString().isEmpty()) {
            deleteOptions.hintString(bulkOperation.getHintString());
          }
          if (bulkOperation.getCollation() != null) {
            deleteOptions.collation(bulkOperation.getCollation().toMongoDriverObject());
          }
          if (bulkOperation.isMulti()) {
            result.add(new DeleteManyModel<>(bsonFilter, deleteOptions));
          } else {
            result.add(new DeleteOneModel<>(bsonFilter, deleteOptions));
          }
          break;
        case INSERT:
          result.add(new InsertOneModel<>(encodeKeyWhenUseObjectId(bulkOperation.getDocument())));
          break;
        case REPLACE:
          ReplaceOptions replaceOptions = new ReplaceOptions();
          if (bulkOperation.getCollation() != null) {
            replaceOptions.collation(bulkOperation.getCollation().toMongoDriverObject());
          }
          if (bulkOperation.getHint() != null) {
            replaceOptions.hint(toBson(bulkOperation.getHint()));
          }
          if (bulkOperation.getHintString() != null && !bulkOperation.getHintString().isEmpty()) {
            replaceOptions.hintString(bulkOperation.getHintString());
          }
          result.add(new ReplaceOneModel<>(toBson(deepEncodeKeyWhenUseObjectId(bulkOperation.getFilter())), bulkOperation.getDocument(),
            replaceOptions.upsert(bulkOperation.isUpsert())));
          break;
        case UPDATE:
          Bson filter = toBson(deepEncodeKeyWhenUseObjectId(bulkOperation.getFilter()));
          Bson document = toBson(encodeKeyWhenUseObjectId(bulkOperation.getDocument()));
          com.mongodb.client.model.UpdateOptions updateOptions = new com.mongodb.client.model.UpdateOptions()
            .upsert(bulkOperation.isUpsert());
          if (bulkOperation.getCollation() != null) {
            updateOptions.collation(bulkOperation.getCollation().toMongoDriverObject());
          }
          if (bulkOperation.getHint() != null) {
            updateOptions.hint(toBson(bulkOperation.getHint()));
          }
          if (bulkOperation.getHintString() != null && !bulkOperation.getHintString().isEmpty()) {
            updateOptions.hintString(bulkOperation.getHintString());
          }
          if (bulkOperation.isMulti()) {
            result.add(new UpdateManyModel<>(filter, document, updateOptions));
          } else {
            result.add(new UpdateOneModel<>(filter, document, updateOptions));
          }
          break;
        default:
          throw new IllegalArgumentException("Unknown bulk operation type: " + bulkOperation.getClass());
      }
    }
    return result;
  }

  @Override
  public MongoClient createCollection(String collectionName, Handler> resultHandler) {
    Future future = createCollection(collectionName);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public MongoClient createCollectionWithOptions(String collectionName, CreateCollectionOptions collectionOptions, Handler> resultHandler) {
    Future future = createCollectionWithOptions(collectionName, collectionOptions);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future createCollection(String collectionName) {
    requireNonNull(collectionName, "collectionName cannot be null");

    Promise promise = vertx.promise();
    holder.db.createCollection(collectionName).subscribe(new CompletionSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public Future createCollectionWithOptions(String collectionName, CreateCollectionOptions collectionOptions) {
    requireNonNull(collectionName, "collectionName cannot be null");

    Promise promise = vertx.promise();
    holder.db.createCollection(collectionName, collectionOptions.toMongoDriverObject())
      .subscribe(new CompletionSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public MongoClient getCollections(Handler>> resultHandler) {
    Future> future = getCollections();
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future> getCollections() {
    Promise> promise = vertx.promise();
    holder.db.listCollectionNames().subscribe(new BufferingSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public MongoClient dropCollection(String collection, Handler> resultHandler) {
    Future future = dropCollection(collection);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future dropCollection(String collection) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);

    MongoCollection coll = getCollection(collection);
    Promise promise = vertx.promise();
    coll.drop().subscribe(new CompletionSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public MongoClient createIndex(String collection, JsonObject key, Handler> resultHandler) {
    Future future = createIndex(collection, key);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future createIndex(String collection, JsonObject key) {
    return createIndexWithOptions(collection, key, new IndexOptions());
  }

  @Override
  public MongoClient createIndexWithOptions(String collection, JsonObject key, IndexOptions options, Handler> resultHandler) {
    Future future = createIndexWithOptions(collection, key, options);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future createIndexWithOptions(String collection, JsonObject key, IndexOptions options) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(key, FIELD_NAME_CANNOT_BE_NULL);

    MongoCollection coll = getCollection(collection);
    com.mongodb.client.model.IndexOptions driverOpts = mongoIndexOptions(options);
    Promise promise = vertx.promise();
    coll.createIndex(wrap(key), driverOpts).subscribe(new CompletionSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public MongoClient createIndexes(String collection, List indexes, Handler> resultHandler) {
    Future future = createIndexes(collection, indexes);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future createIndexes(String collection, List indexes) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);

    final List transformIndexes = indexes.stream().map(it -> {
      if (it.getOptions() != null)
        return new com.mongodb.client.model.IndexModel(wrap(it.getKey()), mongoIndexOptions(it.getOptions()));
      else return new com.mongodb.client.model.IndexModel(wrap(it.getKey()));
    }).collect(Collectors.toList());

    Promise promise = vertx.promise();
    getCollection(collection).createIndexes(transformIndexes).subscribe(new CompletionSubscriber<>(promise));
    return promise.future();
  }

  private static Bson toBson(@Nullable JsonObject json) {
    return json == null ? null : BsonDocument.parse(json.encode());
  }

  @Override
  public MongoClient listIndexes(String collection, Handler> resultHandler) {
    Future future = listIndexes(collection);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future listIndexes(String collection) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    MongoCollection coll = getCollection(collection);
    Promise> promise = vertx.promise();
    coll.listIndexes(JsonObject.class).subscribe(new BufferingSubscriber<>(promise));
    return promise.future().map(JsonArray::new);
  }

  @Override
  public MongoClient dropIndex(String collection, String indexName, Handler> resultHandler) {
    Future future = dropIndex(collection, indexName);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future dropIndex(String collection, String indexName) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(indexName, "indexName cannot be null");
    MongoCollection coll = getCollection(collection);
    Promise promise = vertx.promise();
    coll.dropIndex(indexName).subscribe(new CompletionSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public MongoClient runCommand(String commandName, JsonObject command, Handler> resultHandler) {
    Future future = runCommand(commandName, command);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future<@Nullable JsonObject> runCommand(String commandName, JsonObject command) {
    requireNonNull(commandName, "commandName cannot be null");
    requireNonNull(command, "command cannot be null");
    // The command name must be the first entry in the bson, so to ensure this we must recreate and add the command
    // name as first (JsonObject is internally ordered)
    JsonObject json = new JsonObject();
    Object commandVal = command.getValue(commandName);
    if (commandVal == null) {
      throw new IllegalArgumentException("commandBody does not contain key for " + commandName);
    }
    json.put(commandName, commandVal);
    command.forEach(entry -> {
      if (!entry.getKey().equals(commandName)) {
        json.put(entry.getKey(), entry.getValue());
      }
    });

    Promise promise = vertx.promise();
    holder.db.runCommand(wrap(json), JsonObject.class).subscribe(new SingleResultSubscriber<>(promise));
    return promise.future();
  }

  @Override
  public MongoClient distinct(String collection, String fieldName, String resultClassname, Handler> resultHandler) {
    return distinct(collection, fieldName, resultClassname, null, resultHandler);
  }

  @Override
  public MongoClient distinct(String collection, String fieldName, String resultClassname, DistinctOptions distinctOptions, Handler> resultHandler) {
    Future future = distinct(collection, fieldName, resultClassname, distinctOptions);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future distinct(String collection, String fieldName, String resultClassname) {
    return distinct(collection, fieldName, resultClassname, (DistinctOptions) null);
  }

  @Override
  public Future distinct(String collection, String fieldName, String resultClassname, DistinctOptions distinctOptions) {
    return distinctWithQuery(collection, fieldName, resultClassname, new JsonObject(), distinctOptions);
  }

  @Override
  public MongoClient distinctWithQuery(String collection, String fieldName, String resultClassname, JsonObject query, Handler> resultHandler) {
    return distinctWithQuery(collection, fieldName, resultClassname, query, null, resultHandler);
  }

  @Override
  public MongoClient distinctWithQuery(String collection, String fieldName, String resultClassname, JsonObject query, DistinctOptions distinctOptions, Handler> resultHandler) {
    Future future = distinctWithQuery(collection, fieldName, resultClassname, query, distinctOptions);
    setHandler(future, resultHandler);
    return this;
  }

  @Override
  public Future distinctWithQuery(String collection, String fieldName, String resultClassname, JsonObject query) {
    return distinctWithQuery(collection, fieldName, resultClassname, query, (DistinctOptions) null);
  }

  @Override
  public Future distinctWithQuery(String collection, String fieldName, String resultClassname, JsonObject query, DistinctOptions distinctOptions) {
    try {
      PromiseInternal> promise = vertx.promise();
      findDistinctValuesWithQuery(collection, fieldName, resultClassname, query, distinctOptions).subscribe(new BufferingSubscriber<>(promise));
      return promise.future().map(JsonArray::new);
    } catch (ClassNotFoundException e) {
      return vertx.getOrCreateContext().failedFuture(e);
    }
  }

  @Override
  public ReadStream distinctBatch(String collection, String fieldName, String resultClassname) {
    return distinctBatch(collection, fieldName, resultClassname, null);
  }

  @Override
  public ReadStream distinctBatch(String collection, String fieldName, String resultClassname, DistinctOptions distinctOptions) {
    return distinctBatchWithQuery(collection, fieldName, resultClassname, new JsonObject(), distinctOptions);
  }

  @Override
  public ReadStream distinctBatchWithQuery(String collection, String fieldName, String resultClassname, JsonObject query) {
    return distinctBatchWithQuery(collection, fieldName, resultClassname, query, null);
  }

  @Override
  public ReadStream distinctBatchWithQuery(String collection, String fieldName, String resultClassname, JsonObject query, DistinctOptions distinctOptions) {
    return distinctBatchWithQuery(collection, fieldName, resultClassname, query, FindOptions.DEFAULT_BATCH_SIZE, null);
  }

  @Override
  public ReadStream distinctBatchWithQuery(String collection, String fieldName, String resultClassname, JsonObject query, int batchSize) {
    return distinctBatchWithQuery(collection, fieldName, resultClassname, query, batchSize, null);
  }

  @Override
  public ReadStream distinctBatchWithQuery(String collection, String fieldName, String resultClassname, JsonObject query, int batchSize, DistinctOptions distinctOptions) {
    try {
      DistinctPublisher distinctValuesWithQuery = findDistinctValuesWithQuery(collection, fieldName, resultClassname, query, distinctOptions);
      PublisherAdapter readStream = new PublisherAdapter<>(vertx.getOrCreateContext(), distinctValuesWithQuery, batchSize);
      return new MappingStream<>(readStream, value -> new JsonObject().put(fieldName, value));
    } catch (ClassNotFoundException e) {
      return new FailedStream(e);
    }
  }

  @Override
  public MongoClient createDefaultGridFsBucketService(Handler> resultHandler) {
    return this.createGridFsBucketService("fs", resultHandler);
  }

  @Override
  public Future createDefaultGridFsBucketService() {
    Promise promise = Promise.promise();
    createDefaultGridFsBucketService(promise);
    return promise.future();
  }

  @Override
  public MongoClient createGridFsBucketService(String bucketName, Handler> resultHandler) {
    MongoGridFsClientImpl impl = new MongoGridFsClientImpl(vertx, this, getGridFSBucket(bucketName), holder.db.getCodecRegistry());
    resultHandler.handle(Future.succeededFuture(impl));
    return this;
  }

  @Override
  public Future createGridFsBucketService(String bucketName) {
    Promise promise = Promise.promise();
    createGridFsBucketService(bucketName, promise);
    return promise.future();
  }

  private GridFSBucket getGridFSBucket(String bucketName) {
    return GridFSBuckets.create(holder.db, bucketName);
  }

  @Override
  public ReadStream aggregate(final String collection, final JsonArray pipeline) {
    return aggregateWithOptions(collection, pipeline, DEFAULT_AGGREGATE_OPTIONS);
  }

  @Override
  public ReadStream aggregateWithOptions(final String collection, final JsonArray pipeline, final AggregateOptions options) {
    AggregatePublisher view = doAggregate(collection, pipeline, options);
    return new PublisherAdapter<>(vertx.getOrCreateContext(), view, options.getBatchSize());
  }

  @Override
  public ReadStream> watch(final String collection, final JsonArray pipeline, boolean withUpdatedDoc, int batchSize) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(pipeline, PIPELINE_CANNOT_BE_NULL);
    MongoCollection coll = getCollection(collection);
    final List bpipeline = new ArrayList<>(pipeline.size());
    for (int i = 0; i < pipeline.size(); i++) {
      bpipeline.add(wrap(pipeline.getJsonObject(i)));
    }
    ChangeStreamPublisher changeStreamPublisher = coll.watch(bpipeline, JsonObject.class);
    if (withUpdatedDoc) {
      // By default, only "insert" and "replace" operations return fullDocument
      // Following setting is for "update" operation to return fullDocument
      changeStreamPublisher.fullDocument(FullDocument.UPDATE_LOOKUP);
    }
    if (batchSize < 1) {
      batchSize = 1;
    }
    return new PublisherAdapter<>(vertx.getOrCreateContext(), changeStreamPublisher, batchSize);
  }

  private DistinctPublisher findDistinctValuesWithQuery(String collection, String fieldName, String resultClassname, JsonObject query, DistinctOptions distinctOptions) throws ClassNotFoundException {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(fieldName, FIELD_NAME_CANNOT_BE_NULL);
    requireNonNull(query, QUERY_CANNOT_BE_NULL);

    JsonObject encodedQuery = deepEncodeKeyWhenUseObjectId(query);

    Bson bquery = wrap(encodedQuery);

    MongoCollection mongoCollection = getCollection(collection);
    Class resultClass = this.getClass().getClassLoader().loadClass(resultClassname);
    return setDistinctOptions(mongoCollection.distinct(fieldName, bquery, resultClass), distinctOptions);
  }

  private AggregatePublisher doAggregate(final String collection, final JsonArray pipeline, final AggregateOptions aggregateOptions) {
    requireNonNull(collection, COLLECTION_CANNOT_BE_NULL);
    requireNonNull(pipeline, PIPELINE_CANNOT_BE_NULL);
    requireNonNull(aggregateOptions, "aggregateOptions cannot be null");
    final MongoCollection coll = getCollection(collection);
    final List bpipeline = new ArrayList<>(pipeline.size());
    for (int i = 0; i < pipeline.size(); i++) {
      bpipeline.add(wrap(pipeline.getJsonObject(i)));
    }
    AggregatePublisher aggregate = coll.aggregate(bpipeline, JsonObject.class);

    if(aggregateOptions.getCollation() != null) {
      aggregate.collation(aggregateOptions.getCollation().toMongoDriverObject());
    }
    if (aggregateOptions.getBatchSize() != -1) {
      aggregate.batchSize(aggregateOptions.getBatchSize());
    }
    if (aggregateOptions.getMaxTime() > 0) {
      aggregate.maxTime(aggregateOptions.getMaxTime(), TimeUnit.MILLISECONDS);
    }
    if (aggregateOptions.getAllowDiskUse() != null) {
      aggregate.allowDiskUse(aggregateOptions.getAllowDiskUse());
    }
    return aggregate;
  }


  JsonArray deepEncodeKeyWhenUseObjectId(JsonArray arr) {
    if(!useObjectId) return arr;

    JsonArray newArr = new JsonArray(new ArrayList<>(arr.size()));

    for (Object item : arr) {
      if (item instanceof JsonArray) {
        newArr.add(deepEncodeKeyWhenUseObjectId((JsonArray) item));
      } else if(item instanceof List) {
        newArr.add(deepEncodeKeyWhenUseObjectId(new JsonArray((List) item)));
      } else if (item instanceof JsonObject) {
        newArr.add(deepEncodeKeyWhenUseObjectId((JsonObject) item));
      } else if (item instanceof Map) {
        newArr.add(deepEncodeKeyWhenUseObjectId(new JsonObject((Map) item)));
      } else {
        newArr.add(item);
      }
    }

    return newArr;
  }

  JsonObject deepEncodeKeyWhenUseObjectId(JsonObject json) {
    if(!useObjectId) return json;

    JsonObject newJson = new JsonObject(new LinkedHashMap<>(json.size()));

    for (Map.Entry entry : json) {
      String key = entry.getKey();
      Object value = entry.getValue();
      if (key.equals(ID_FIELD)
          && value instanceof String
          && ObjectId.isValid((String) value)) {
        newJson.put(key, new JsonObject().put(JsonObjectCodec.OID_FIELD, value));
      } else if (value instanceof JsonObject) {
        newJson.put(key, deepEncodeKeyWhenUseObjectId((JsonObject) value));
      } else if (value instanceof Map) {
        newJson.put(key, deepEncodeKeyWhenUseObjectId(new JsonObject((Map) value)));
      } else if (value instanceof JsonArray) {
        newJson.put(key, deepEncodeKeyWhenUseObjectId((JsonArray) value));
      } else if (value instanceof List) {
        newJson.put(key, deepEncodeKeyWhenUseObjectId(new JsonArray((List) value)));
      } else {
        newJson.put(key, value);
      }
    }

    return newJson;
  }

  JsonObject encodeKeyWhenUseObjectId(JsonObject json) {
    if (!useObjectId)
      return json;

    Object idString = json.getValue(ID_FIELD, null);
    if (idString instanceof String && ObjectId.isValid((String) idString)) {
      json.put(ID_FIELD, new JsonObject().put(JsonObjectCodec.OID_FIELD, idString));
    }

    return json;
  }

  private JsonObject decodeKeyWhenUseObjectId(JsonObject json) {
    if (!useObjectId) return json;

    Object idField = json.getValue(ID_FIELD, null);
    if (!(idField instanceof JsonObject)) return json;

    Object idString = ((JsonObject) idField).getValue(JsonObjectCodec.OID_FIELD, null);
    if (!(idString instanceof String)) return json;

    json.put(ID_FIELD, idString);

    return json;
  }

  private FindPublisher doFind(String collection, JsonObject query, FindOptions options) {
    MongoCollection coll = getCollection(collection);
    Bson bquery = wrap(deepEncodeKeyWhenUseObjectId(query));
    FindPublisher find = coll.find(bquery, JsonObject.class);
    if (options.getLimit() != -1) {
      find.limit(options.getLimit());
    }
    if (options.getSkip() > 0) {
      find.skip(options.getSkip());
    }
    if (options.getSort() != null) {
      find.sort(wrap(options.getSort()));
    }
    if (options.getFields() != null) {
      find.projection(wrap(options.getFields()));
    }
    if (options.getHint() != null) {
      find.hint(wrap(options.getHint()));
    }
    if (options.getHintString() != null && !options.getHintString().isEmpty()) {
      find.hintString(options.getHintString());
    }
    if(options.getCollation() != null) {
      find.collation(options.getCollation().toMongoDriverObject());
    }
    return find;
  }

  private MongoCollection getCollection(String name) {
    return getCollection(name, null);
  }

  private MongoCollection getCollection(String name, @Nullable WriteOption writeOption) {
    MongoCollection coll = holder.db.getCollection(name, JsonObject.class);
    if (coll != null && writeOption != null) {
      coll = coll.withWriteConcern(WriteConcern.valueOf(writeOption.name()));
    }
    return coll;
  }

  private com.mongodb.client.model.IndexOptions mongoIndexOptions(IndexOptions options) {
    CollationOptions co = options.getCollation();
    com.mongodb.client.model.IndexOptions o = new com.mongodb.client.model.IndexOptions()
      .background(options.isBackground())
      .unique(options.isUnique())
      .name(options.getName())
      .sparse(options.isSparse())
      .expireAfter(options.getExpireAfter(TimeUnit.SECONDS), TimeUnit.SECONDS)
      .version(options.getVersion())
      .weights(toBson(options.getWeights()))
      .defaultLanguage(options.getDefaultLanguage())
      .languageOverride(options.getLanguageOverride())
      .textVersion(options.getTextVersion())
      .sphereVersion(options.getSphereVersion())
      .bits(options.getBits())
      .min(options.getMin())
      .max(options.getMax())
      .bucketSize(options.getBucketSize())
      .storageEngine(toBson(options.getStorageEngine()))
      .partialFilterExpression(toBson(options.getPartialFilterExpression()));
      if (co != null) {
        o.collation(co.toMongoDriverObject());
      }
      return o;
  }

  @Nullable JsonObjectBsonAdapter wrap(@Nullable JsonObject jsonObject) {
    return jsonObject == null ? null : new JsonObjectBsonAdapter(jsonObject);
  }

  private void removeFromMap(LocalMap map, String dataSourceName) {
    synchronized (vertx) {
      map.remove(dataSourceName);
      if (map.isEmpty()) {
        map.close();
      }
    }
  }

  private MongoHolder lookupHolder(String datasourceName, JsonObject config) {
    synchronized (vertx) {
      LocalMap map = vertx.sharedData().getLocalMap(DS_LOCAL_MAP_NAME);
      MongoHolder theHolder = map.get(datasourceName);
      if (theHolder == null) {
        theHolder = new MongoHolder(config, () -> removeFromMap(map, datasourceName));
        map.put(datasourceName, theHolder);
      } else {
        theHolder.incRefCount();
      }
      return theHolder;
    }
  }

  @Override
  public void close(Handler> handler) {
    ContextInternal ctx = vertx.getOrCreateContext();
    close(ctx.promise(handler));
  }

  private class MongoHolder implements Shareable {
    com.mongodb.reactivestreams.client.MongoClient mongo;
    MongoDatabase db;
    JsonObject config;
    Runnable closeRunner;
    int refCount = 1;

    MongoHolder(JsonObject config, Runnable closeRunner) {
      this.config = config;
      this.closeRunner = closeRunner;
    }

    synchronized com.mongodb.reactivestreams.client.MongoClient mongo(Vertx vertx) {
      if (mongo == null) {
        MongoClientOptionsParser parser = new MongoClientOptionsParser(vertx, config);
        mongo = MongoClients.create(parser.settings());
        db = mongo.getDatabase(parser.database());
      }
      return mongo;
    }

    synchronized com.mongodb.reactivestreams.client.MongoClient mongo(Vertx vertx, MongoClientSettings settings) {
      if (mongo == null) {
        MongoClientOptionsParser parser = new MongoClientOptionsParser(vertx, config);
        mongo = MongoClients.create(settings);
        db = mongo.getDatabase(parser.database());
      }
      return mongo;
    }

    synchronized void incRefCount() {
      refCount++;
    }

    void close() {
      java.io.Closeable client;
      Runnable callback;
      synchronized (this) {
        if (--refCount > 0) {
          return;
        }
        client = mongo;
        mongo = null;
        callback = closeRunner;
        closeRunner = null;
      }
      if (callback != null) {
        callback.run();
      }
      if (client != null) {
        MongoClientImpl.this.vertx.executeBlocking(p -> {
          try {
            client.close();
          } catch (IOException e) {
            p.fail(e);
          }
        });
      }
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy