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

net.pincette.mongo.streams.Merge Maven / Gradle / Ivy

The newest version!
package net.pincette.mongo.streams;

import static java.util.Optional.ofNullable;
import static java.util.UUID.randomUUID;
import static java.util.concurrent.CompletableFuture.completedFuture;
import static net.pincette.json.JsonUtil.copy;
import static net.pincette.json.JsonUtil.createObjectBuilder;
import static net.pincette.json.JsonUtil.createValue;
import static net.pincette.json.JsonUtil.emptyObject;
import static net.pincette.json.JsonUtil.getValue;
import static net.pincette.json.JsonUtil.isObject;
import static net.pincette.json.JsonUtil.string;
import static net.pincette.mongo.Expression.function;
import static net.pincette.mongo.JsonClient.findOne;
import static net.pincette.mongo.JsonClient.insert;
import static net.pincette.mongo.streams.Util.ID;
import static net.pincette.mongo.streams.Util.RETRY;
import static net.pincette.mongo.streams.Util.exceptionLogger;
import static net.pincette.mongo.streams.Util.matchFields;
import static net.pincette.mongo.streams.Util.matchQuery;
import static net.pincette.rs.Async.mapAsyncSequential;
import static net.pincette.rs.Filter.filter;
import static net.pincette.rs.Mapper.map;
import static net.pincette.rs.Pipe.pipe;
import static net.pincette.util.Pair.pair;
import static net.pincette.util.ScheduledCompletionStage.composeAsyncAfter;
import static net.pincette.util.Util.must;
import static net.pincette.util.Util.rethrow;

import com.mongodb.reactivestreams.client.MongoCollection;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.CompletionStage;
import java.util.concurrent.Flow.Processor;
import java.util.function.Function;
import javax.json.JsonObject;
import javax.json.JsonValue;
import net.pincette.mongo.JsonClient;
import net.pincette.rs.streams.Message;
import org.bson.Document;

/**
 * The $merge operator.
 *
 * @author Werner Donné
 */
class Merge {
  private static final String FAIL = "fail";
  private static final String INSERT = "insert";
  private static final String INTO = "into";
  private static final String KEEP_EXISTING = "keepExisting";
  private static final String KEY = "key";
  private static final String MERGE_FIELD = "merge";
  private static final String REPLACE = "replace";
  private static final String WHEN_MATCHED = "whenMatched";
  private static final String WHEN_NOT_MATCHED = "whenNotMatched";

  private Merge() {}

  private static JsonObject addId(final JsonObject json) {
    return json.containsKey(ID)
        ? json
        : createObjectBuilder(json).add(ID, randomUUID().toString()).build();
  }

  private static JsonObject addId(final JsonObject json, final JsonValue value) {
    return createObjectBuilder(json).add(ID, value).build();
  }

  private static FailException exception(final JsonObject expression) {
    return new FailException("$merge " + string(expression) + " failed");
  }

  private static String getWhenMatched(final JsonObject expression) {
    return expression.getString(WHEN_MATCHED, MERGE_FIELD);
  }

  private static CompletionStage process(
      final JsonObject fromStream,
      final JsonValue key,
      final JsonObject query,
      final JsonObject expression,
      final MongoCollection collection,
      final Context context) {
    return findOne(collection, query)
        .thenComposeAsync(
            found ->
                found
                    .map(f -> processExisting(fromStream, f, expression, collection))
                    .orElseGet(() -> processNew(addId(fromStream, key), expression, collection)))
        .exceptionally(
            t -> {
              exceptionLogger(t.getCause(), "$merge", context);

              if (t.getCause() instanceof FailException) {
                rethrow(t.getCause());
              }

              return null;
            })
        .thenComposeAsync(
            value ->
                value == null
                    ? composeAsyncAfter(
                        () -> process(fromStream, key, query, expression, collection, context),
                        RETRY)
                    : completedFuture(value));
  }

  private static CompletionStage processExisting(
      final JsonObject fromStream,
      final JsonObject fromCollection,
      final JsonObject expression,
      final MongoCollection collection) {
    return switch (getWhenMatched(expression)) {
      case FAIL -> throw exception(expression);
      case KEEP_EXISTING -> completedFuture(fromCollection);
      case MERGE_FIELD ->
          update(
              collection,
              copy(fromStream, createObjectBuilder(fromCollection)).build(),
              fromCollection);
      case REPLACE -> update(collection, fromStream, fromCollection);
      default -> completedFuture(emptyObject());
    };
  }

  private static CompletionStage processNew(
      final JsonObject fromStream,
      final JsonObject expression,
      final MongoCollection collection) {
    return switch (expression.getString(WHEN_NOT_MATCHED, INSERT)) {
      case FAIL -> throw exception(expression);
      case INSERT ->
          Optional.of(addId(fromStream))
              .map(json -> update(collection, json, null))
              .orElseGet(() -> completedFuture(null));
      default -> completedFuture(emptyObject());
    };
  }

  private static Message setId(
      final Message message, final JsonObject newValue) {
    return message.withValue(
        ofNullable(message.value.get(ID))
            .filter(id -> !newValue.isEmpty())
            .map(id -> addId(newValue, id))
            .orElse(newValue));
  }

  static Processor, Message> stage(
      final JsonValue expression, final Context context) {
    must(isObject(expression));

    final JsonObject expr = expression.asJsonObject();
    final MongoCollection collection =
        context.database.getCollection(expr.getString(INTO));
    final Set fields = matchFields(expr, ID);
    final Function key =
        function(
            getValue(expr, "/" + KEY).orElseGet(() -> createValue("$" + ID)), context.features);

    return pipe(map(
            (Message m) ->
                pair(m, matchQuery(m.value, fields).orElseThrow(() -> exception(expr)))))
        .then(
            mapAsyncSequential(
                pair ->
                    process(
                            pair.first.value,
                            key.apply(pair.first.value),
                            pair.second,
                            expr,
                            collection,
                            context)
                        .thenApply(newValue -> setId(pair.first, newValue))))
        .then(filter(m -> !m.value.isEmpty()))
        .then(
            map(m -> m.withKey(ofNullable(m.value.get(ID)).map(Util::generateKey).orElse(m.key))));
  }

  private static CompletionStage update(
      final MongoCollection collection,
      final JsonObject json,
      final JsonObject existing) {
    return (existing != null
            ? JsonClient.update(collection, existing, addId(json, existing.get(ID)))
            : insert(collection, json))
        .thenApply(result -> must(result, r -> r))
        .thenApply(result -> json);
  }

  private static class FailException extends RuntimeException {
    private FailException(final String message) {
      super(message);
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy