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

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

package net.pincette.mongo.streams;

import static java.util.Optional.ofNullable;
import static java.util.logging.Level.SEVERE;
import static net.pincette.json.Jslt.transformerObject;
import static net.pincette.json.Jslt.tryReader;
import static net.pincette.json.JsonUtil.asString;
import static net.pincette.json.JsonUtil.isString;
import static net.pincette.json.JsonUtil.string;
import static net.pincette.rs.Mapper.map;
import static net.pincette.rs.streams.Message.message;
import static net.pincette.util.Util.must;
import static net.pincette.util.Util.rethrow;
import static net.pincette.util.Util.tryToGet;

import java.util.Optional;
import java.util.concurrent.Flow.Processor;
import java.util.function.UnaryOperator;
import javax.json.JsonObject;
import javax.json.JsonValue;
import net.pincette.function.SideEffect;
import net.pincette.rs.streams.Message;

/**
 * The $jslt operator.
 *
 * @author Werner Donné
 */
class Jslt {
  private static final String ID = "_id";

  private Jslt() {}

  private static JsonObject logCall(
      final UnaryOperator transformer,
      final JsonObject json,
      final JsonValue expression,
      final Context context) {
    return tryToGet(
            () -> transformer.apply(json),
            e ->
                SideEffect.run(
                        () -> {
                          if (context.logger != null) {
                            context.logger.log(
                                SEVERE,
                                e,
                                () ->
                                    "Expression:\n"
                                        + asString(expression)
                                        + "\n\nWith:\n"
                                        + string(json));
                          }

                          rethrow(e);
                        })
                    .andThenGet(() -> null))
        .orElse(null);
  }

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

    final UnaryOperator transformer = transformer(expression, context);

    return map(
        m ->
            Optional.of(logCall(transformer, m.value, expression, context))
                .map(
                    result -> message(ofNullable(result.getString(ID, null)).orElse(m.key), result))
                .orElse(m));
  }

  private static UnaryOperator transformer(
      final JsonValue expression, final Context context) {
    final String expr = asString(expression).getString();
    final net.pincette.json.Jslt.Context transformerContext =
        new net.pincette.json.Jslt.Context(tryReader(expr));

    return tryToGet(
            () ->
                transformerObject(
                    context.features != null
                        ? transformerContext
                            .withFunctions(context.features.customJsltFunctions)
                            .withResolver(context.features.jsltResolver)
                        : transformerContext),
            e ->
                SideEffect.>run(
                        () -> {
                          if (context.logger != null) {
                            context.logger.log(SEVERE, e, () -> "Expression:\n" + expr);
                          }

                          rethrow(e);
                        })
                    .andThenGet(() -> null))
        .orElse(null);
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy