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

io.hyperfoil.tools.horreum.svc.Util Maven / Gradle / Ivy

package io.hyperfoil.tools.horreum.svc;

import static java.lang.annotation.ElementType.*;
import static java.lang.annotation.RetentionPolicy.RUNTIME;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.lang.annotation.Annotation;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import java.lang.reflect.Method;
import java.net.URLDecoder;
import java.nio.charset.StandardCharsets;
import java.time.Instant;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.time.format.DateTimeParseException;
import java.util.*;
import java.util.function.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import jakarta.enterprise.context.ApplicationScoped;
import jakarta.enterprise.inject.Produces;
import jakarta.enterprise.inject.spi.Bean;
import jakarta.enterprise.inject.spi.BeanManager;
import jakarta.enterprise.inject.spi.CDI;
import jakarta.inject.Qualifier;
import jakarta.persistence.EntityManager;
import jakarta.persistence.NoResultException;
import jakarta.persistence.OptimisticLockException;
import jakarta.persistence.Query;
import jakarta.transaction.*;

import org.eclipse.microprofile.context.ThreadContext;
import org.graalvm.polyglot.*;
import org.graalvm.polyglot.proxy.Proxy;
import org.graalvm.polyglot.proxy.ProxyExecutable;
import org.hibernate.query.NativeQuery;
import org.jboss.logging.Logger;
import org.postgresql.util.PSQLException;

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.*;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import com.jayway.jsonpath.Configuration;
import com.jayway.jsonpath.InvalidPathException;
import com.jayway.jsonpath.JsonPath;
import com.jayway.jsonpath.Option;
import com.jayway.jsonpath.ReadContext;
import com.jayway.jsonpath.spi.json.JacksonJsonNodeJsonProvider;

import io.hyperfoil.tools.horreum.api.SortDirection;
import io.hyperfoil.tools.horreum.server.RolesInterceptor;
import io.quarkus.security.identity.SecurityIdentity;
import io.smallrye.context.SmallRyeContextManagerProvider;
import io.smallrye.mutiny.Uni;
import io.vertx.core.Vertx;
import io.vertx.core.eventbus.EventBus;

public class Util {
    private static final Logger log = Logger.getLogger(Util.class);
    public static final ObjectMapper OBJECT_MAPPER = new ObjectMapper();
    private static final Configuration JSONPATH_CONFIG = Configuration.builder()
            .jsonProvider(new JacksonJsonNodeJsonProvider())
            .options(Option.SUPPRESS_EXCEPTIONS, Option.DEFAULT_PATH_LEAF_TO_NULL).build();
    static final ArrayNode EMPTY_ARRAY = JsonNodeFactory.instance.arrayNode();
    static final ObjectNode EMPTY_OBJECT = JsonNodeFactory.instance.objectNode();

    public static final int MAX_TRANSACTION_RETRIES = 10;
    private static final String RETRY_HINT = "The transaction might succeed if retried";

    private static final Pattern JSONPATH_ROOT_PATTERN = Pattern.compile("^\\$\\.(\"[^\"]+\"|[^\\.\\s]+)");

    static {
        OBJECT_MAPPER.registerModule(new JavaTimeModule());
    }

    @Qualifier
    @Retention(RUNTIME)
    @Target({ METHOD, FIELD, PARAMETER, TYPE })
    public @interface FailUnknownProperties {
    }

    @Produces
    @FailUnknownProperties
    @ApplicationScoped
    public ObjectMapper producerObjectMapper() {
        BeanManager beanManager = CDI.current().getBeanManager();

        Bean bean = (Bean) beanManager.resolve(beanManager.getBeans(ObjectMapper.class));
        ObjectMapper objectMapper = beanManager.getContext(bean.getScope()).get(bean,
                beanManager.createCreationalContext(bean));

        ObjectMapper customMapper = objectMapper.copy();
        customMapper.enable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);

        return customMapper;
    }

    static String destringify(String str) {
        if (str == null || str.isEmpty()) {
            return str;
        }
        if (str.charAt(0) == '"' && str.charAt(str.length() - 1) == '"') {
            return str.substring(1, str.length() - 1);
        } else {
            return str;
        }
    }

    static Double toDoubleOrNull(Value value, Consumer error, Consumer info) {
        if (value.isNumber()) {
            double dValue = value.asDouble();
            if (Double.isFinite(dValue)) {
                return dValue;
            } else {
                error.accept("Not a finite number: " + value);
                return null;
            }
        } else if (value.isString()) {
            try {
                return Double.parseDouble(value.asString());
            } catch (NumberFormatException e) {
                error.accept("Return value " + value + " cannot be parsed into a number.");
                return null;
            }
        } else if (value.isNull()) {
            // returning null is intentional or the data does not exist, don't warn
            info.accept("Result is null, skipping.");
            return null;
        } else if ("undefined".equals(value.toString())) {
            // returning undefined is intentional, don't warn
            info.accept("Result is undefined, skipping.");
            return null;
        } else {
            error.accept("Return value " + value + " is not a number.");
            return null;
        }
    }

    static Double toDoubleOrNull(Object value) {
        if (value instanceof String) {
            String str = (String) value;
            String maybeNumber = str.charAt(0) == '"' && str.charAt(str.length() - 1) == '"'
                    ? str.substring(1, str.length() - 1)
                    : str;
            try {
                return Double.parseDouble(maybeNumber);
            } catch (NumberFormatException e) {
                return null;
            }
        } else if (value instanceof Double) {
            return (Double) value;
        } else if (value instanceof Long) {
            return ((Long) value).doubleValue();
        } else if (value instanceof Integer) {
            return ((Integer) value).doubleValue();
        } else if (value instanceof Float) {
            return ((Float) value).doubleValue();
        } else if (value instanceof Short) {
            return ((Short) value).doubleValue();
        } else {
            return null;
        }
    }

    static void addPaging(StringBuilder sql, Integer limit, Integer page, String sort, SortDirection direction) {
        addOrderBy(sql, sort, direction);
        addLimitOffset(sql, limit, page);
    }

    static void addOrderBy(StringBuilder sql, String sort, SortDirection direction) {
        sort = sort == null || sort.trim().isEmpty() ? "start" : sort;
        direction = direction == null ? SortDirection.Descending : direction;
        sql.append(" ORDER BY ").append(sort);
        addDirection(sql, direction);
    }

    static void addDirection(StringBuilder sql, SortDirection direction) {
        if (direction != null) {
            sql.append("Ascending".equalsIgnoreCase(direction.toString()) ? " ASC" : " DESC");
        }
        sql.append(" NULLS LAST");
    }

    static void addLimitOffset(StringBuilder sql, Integer limit, Integer page) {
        if (limit != null && limit > 0) {
            sql.append(" limit ").append(limit);
            if (page != null && page >= 0) {
                sql.append(" offset ").append(limit * (page - 1));
            }
        }
    }

    public static void doAfterCommitThrowing(TransactionManager tm, Runnable runnable)
            throws SystemException, RollbackException {
        tm.getTransaction().registerSynchronization(new Synchronization() {
            @Override
            public void beforeCompletion() {
            }

            @Override
            public void afterCompletion(int status) {
                try {
                    if (status == Status.STATUS_COMMITTED || status == Status.STATUS_COMMITTING) {
                        runnable.run();
                    }
                } catch (Throwable t) {
                    log.error("Error in TX synchronization", t);
                    throw t;
                }
            }
        });
    }

    public static void doAfterCommit(TransactionManager tm, Runnable runnable) {
        try {
            doAfterCommitThrowing(tm, runnable);
        } catch (RollbackException e) {
            log.debugf("Not performing %s as the transaction has been marked rollback-only", runnable);
        } catch (SystemException e) {
            log.errorf(e, "Failed to perform %s after transaction completion", runnable);
        }
    }

    static void publishLater(TransactionManager tm, final EventBus eventBus, String eventName, Object event) {
        try {
            doAfterCommitThrowing(tm, () -> eventBus.publish(eventName, event));
        } catch (RollbackException e) {
            log.debug("Not publishing the event as the transaction has been marked rollback-only");
        } catch (SystemException e) {
            log.errorf(e, "Failed to publish event %s: %s after transaction completion", eventName, event);
        }
    }

    public static JsonNode toJsonNode(String str) {
        try {
            if (str == null) {
                return null;
            }
            return OBJECT_MAPPER.readTree(str);
        } catch (JsonProcessingException e) {
            log.errorf(e, "Failed to parse into JSON: %s", str);
            return null;
        }
    }

    public static JsonNode toJsonNode(byte[] bytes) {
        try {
            if (bytes == null) {
                return null;
            }
            return OBJECT_MAPPER.readTree(bytes);
        } catch (IOException e) {
            log.errorf(e, "Failed to parse into JSON: %s", new String(bytes, StandardCharsets.UTF_8));
            throw new RuntimeException(e);
        }
    }

    public static LinkedHashMap toMap(JsonNode jsonNode) {
        LinkedHashMap map = new LinkedHashMap<>(jsonNode.size());
        if (jsonNode instanceof ObjectNode) {
            Iterator> it = jsonNode.fields();
            while (it.hasNext()) {
                var entry = it.next();
                map.put(entry.getKey(), entry.getValue());
            }
        } else if (jsonNode instanceof ArrayNode) {
            int index = 0;
            for (JsonNode node : jsonNode) {
                map.put(index++, node);
            }
        }
        return map;
    }

    public static JsonNode convertToJson(Value value) {
        if (value == null || value.isNull()) {
            return JsonNodeFactory.instance.nullNode();
        } else if (value.isProxyObject()) {
            return value.asProxyObject();
        } else if (value.isBoolean()) {
            return JsonNodeFactory.instance.booleanNode(value.asBoolean());
        } else if (value.isNumber()) {
            double v = value.asDouble();
            if (v == Math.rint(v)) {
                return JsonNodeFactory.instance.numberNode((long) v);
            } else {
                return JsonNodeFactory.instance.numberNode(v);
            }
        } else if (value.isString()) {
            return JsonNodeFactory.instance.textNode(value.asString());
        } else if (value.hasArrayElements()) {
            return convertArray(value);
        } else if (value.canExecute()) {
            return JsonNodeFactory.instance.textNode(value.toString());
        } else if (value.hasMembers()) {
            return convertMapping(value);
        } else {
            return JsonNodeFactory.instance.textNode(value.toString());
        }
    }

    public static Object convertFromJson(JsonNode node) {
        switch (node.getNodeType()) {
            case BINARY:
            case STRING:
                return node.asText();
            case BOOLEAN:
                return node.asBoolean();
            case MISSING:
            case NULL:
                return null;
            case NUMBER:
                double value = node.asDouble();
                if (value == Math.rint(value)) {
                    return (long) value;
                } else {
                    return value;
                }
            case OBJECT:
                return (ObjectNode) node;
            case ARRAY:
                return (ArrayNode) node;
            default:
                return node;
        }
    }

    public static Object convert(Value value) {
        if (value == null) {
            return null;
        } else if (value.isNull()) {
            // Value api cannot differentiate null and undefined from javascript
            if (value.toString().contains("undefined")) {
                return ""; //no return is the same as returning a missing key from a ProxyObject?
            } else {
                return null;
            }
        } else if (value.isProxyObject()) {
            Proxy p = value.asProxyObject();
            if (p instanceof ProxyJacksonArray) {
                return ((ProxyJacksonArray) p).getJsonNode();
            } else if (p instanceof ProxyJacksonObject) {
                return ((ProxyJacksonObject) p).getJsonNode();
            } else {
                return p;
            }
        } else if (value.isBoolean()) {
            return value.asBoolean();
        } else if (value.isNumber()) {
            double v = value.asDouble();
            if (v == Math.rint(v)) {
                return (long) v;
            } else {
                return v;
            }
        } else if (value.isString()) {
            return value.asString();
        } else if (value.hasArrayElements()) {
            return convertArray(value);
        } else if (value.canExecute()) {
            return value.toString();
        } else if (value.hasMembers()) {
            return convertMapping(value);
        } else {
            //TODO log error wtf is Value?
            return "";
        }
    }

    public static ArrayNode convertArray(Value value) {
        ArrayNode json = JsonNodeFactory.instance.arrayNode();
        for (int i = 0; i < value.getArraySize(); i++) {
            Value element = value.getArrayElement(i);
            if (element == null || element.isNull()) {
                json.addNull();
            } else if (element.isBoolean()) {
                json.add(element.asBoolean());
            } else if (element.isNumber()) {
                double v = element.asDouble();
                if (v == Math.rint(v)) {
                    json.add(element.asLong());
                } else {
                    json.add(v);
                }
            } else if (element.isString()) {
                json.add(element.asString());
            } else if (element.hasArrayElements()) {
                json.add(convertArray(element));
            } else if (element.hasMembers()) {
                json.add(convertMapping(element));
            } else {
                json.add(element.toString());
            }
        }
        return json;
    }

    public static ObjectNode convertMapping(Value value) {
        ObjectNode json = JsonNodeFactory.instance.objectNode();
        for (String key : value.getMemberKeys()) {
            Value element = value.getMember(key);
            if (element == null || element.isNull()) {
                json.set(key, JsonNodeFactory.instance.nullNode());
            } else if (element.isBoolean()) {
                json.set(key, JsonNodeFactory.instance.booleanNode(element.asBoolean()));
            } else if (element.isNumber()) {
                double v = element.asDouble();
                if (v == Math.rint(v)) {
                    json.set(key, JsonNodeFactory.instance.numberNode(element.asLong()));
                } else {
                    json.set(key, JsonNodeFactory.instance.numberNode(v));
                }
            } else if (element.isString()) {
                json.set(key, JsonNodeFactory.instance.textNode(element.asString()));
            } else if (element.hasArrayElements()) {
                json.set(key, convertArray(element));
            } else if (element.hasMembers()) {
                json.set(key, convertMapping(element));
            } else {
                json.set(key, JsonNodeFactory.instance.textNode(element.toString()));
            }
        }
        return json;
    }

    public static Object findJsonPath(JsonNode input, String jsonPath) {
        ReadContext ctx = JsonPath.parse(input, JSONPATH_CONFIG);
        try {
            JsonPath path = JsonPath.compile(jsonPath);
            Object obj = ctx.read(path);
            if (obj instanceof ArrayNode arr) {
                if (arr.size() == 1) {
                    obj = arr.get(0);
                }
            }
            if (obj instanceof ValueNode) {
                ValueNode node = (ValueNode) obj;
                switch (node.getNodeType()) {
                    case BINARY:
                    case STRING:
                        return node.asText();
                    case BOOLEAN:
                        return node.asBoolean();
                    case MISSING:
                    case NULL:
                        return null;
                    case NUMBER:
                        double value = node.asDouble();
                        if (value == Math.rint(value)) {
                            return (long) value;
                        } else {
                            return value;
                        }
                }
            }
            return obj;
        } catch (InvalidPathException e) {
            return "";
        }
    }

    public static  T withTx(TransactionManager tm, Supplier supplier) {
        for (int retry = 1;; ++retry) {
            try {
                tm.begin();
                try {
                    return supplier.get();
                } catch (Throwable t) {
                    tm.setRollbackOnly();
                    // Similar code is in BaseTransactionRetryInterceptor
                    if (retry > Util.MAX_TRANSACTION_RETRIES) {
                        log.error("Exceeded maximum number of retries.");
                        throw t;
                    }
                    if (!lookupRetryHint(t, new HashSet<>())) {
                        throw t;
                    }
                    yieldAndLog(retry, t);
                } finally {
                    if (tm.getStatus() == Status.STATUS_ACTIVE) {
                        tm.commit();
                    } else {
                        tm.rollback();
                    }
                }
            } catch (SystemException | RollbackException | HeuristicMixedException | HeuristicRollbackException
                    | NotSupportedException ex) {
                if (!lookupRetryHint(ex, new HashSet<>())) {
                    throw new RuntimeException("Failed to run transaction", ex);
                }
                yieldAndLog(retry, ex);
            }
        }
    }

    private static void yieldAndLog(int retry, Throwable t) {
        Thread.yield(); // give the other transaction a bit more chance to complete
        log.infof("Retrying failed transaction, attempt %d/%d", retry, Util.MAX_TRANSACTION_RETRIES);
        log.trace("This is the exception that caused retry: ", t);
    }

    public static  T getAnnotation(Method method, Class annotationClass) {
        T methodAnnotation = method.getAnnotation(annotationClass);
        if (methodAnnotation != null) {
            return methodAnnotation;
        } else {
            return method.getDeclaringClass().getAnnotation(annotationClass);
        }
    }

    public static String explainCauses(Throwable e) {
        StringBuilder causes = new StringBuilder();
        Set reported = new HashSet<>();
        while (e != null && !reported.contains(e)) {
            if (causes.length() != 0) {
                causes.append(": ");
            }
            causes.append(e.getMessage());
            reported.add(e);
            e = e.getCause();
        }
        return causes.toString();
    }

    public static void executeBlocking(Vertx vertx, Runnable runnable) {
        Runnable wrapped = wrapForBlockingExecution(runnable);
        vertx.executeBlocking(promise -> {
            try {
                wrapped.run();
            } catch (Exception e) {
                log.error("Failed to execute blocking task", e);
            } finally {
                promise.complete();
            }
        }, result -> {
        });
    }

    public static Runnable wrapForBlockingExecution(Runnable runnable) {
        // CDI needs to be propagated - without that the interceptors wouldn't run.
        // Without thread context propagation we would get an exception in Run.findById, though the interceptors would be invoked correctly.
        Runnable withThreadContext = SmallRyeContextManagerProvider.getManager().newThreadContextBuilder()
                .propagated(ThreadContext.CDI).build().contextualRunnable(runnable);
        return () -> {
            // Note: this won't help with accessing the injected security identity
            RolesInterceptor.setCurrentIdentity(CachedSecurityIdentity.ANONYMOUS);
            try {
                withThreadContext.run();
            } finally {
                RolesInterceptor.setCurrentIdentity(null);
            }
        };
    }

    public static Uni executeBlocking(io.vertx.mutiny.core.Vertx vertx, SecurityIdentity identity, Uni uni) {
        return vertx.executeBlocking(Uni.createFrom().voidItem()
                .invoke(() -> RolesInterceptor.setCurrentIdentity(identity))
                .chain(() -> uni)
                .eventually(() -> RolesInterceptor.setCurrentIdentity(null)));
    }

    public static boolean lookupRetryHint(Throwable ex, Set causes) {
        while (ex != null && causes.add(ex)) {
            if (ex instanceof PSQLException) {
                if (ex.getMessage().contains(RETRY_HINT)) {
                    return true;
                }
            } else if (ex instanceof OptimisticLockException) {
                return true;
            }
            for (Throwable suppressed : ex.getSuppressed()) {
                if (lookupRetryHint(suppressed, causes)) {
                    return true;
                }
            }
            ex = ex.getCause();
        }
        return false;
    }

    public static JsonNode parseFingerprint(String fpString) {
        if (fpString == null || fpString.isEmpty()) {
            // any fingerprint
            return null;
        }
        return toJsonNode(URLDecoder.decode(fpString.replace("+", "%2B"), StandardCharsets.UTF_8));
    }

    /*
     * Evaluates a List of Objects, executing Javascript Combination Functions, if defined.
     * Callbacks for JS evaluation exceptions and output logging allow for custom error handling
     */
    static  void evaluateWithCombinationFunction(List inputData,
            Function jsCombinationFunction,
            Function evaluationInputObject,
            BiConsumer jsFuncResultConsumer,
            Consumer nonFuncResultConsumer,
            ExecutionExceptionConsumer onJsEvaluationException,
            Consumer jsOutputConsumer) {
        ByteArrayOutputStream out = new ByteArrayOutputStream();

        for (int i = 0; i < inputData.size(); i++) {
            T element = inputData.get(i);
            String jsFuncBody = jsCombinationFunction.apply(element);
            if (jsFuncBody != null && !jsFuncBody.isBlank()) {
                try (org.graalvm.polyglot.Context context = createContext(out)) {
                    context.enter();
                    try {
                        setupContext(context);
                        StringBuilder jsCode = new StringBuilder("const __obj").append(i).append(" = ")
                                .append(evaluationInputObject.apply(element)).append(";\n");
                        jsCode.append("const __func").append(i).append(" = ").append(jsFuncBody).append(";\n");
                        jsCode.append("__func").append(i).append("(__obj").append(i).append(")");
                        try {
                            Value value = context.eval("js", jsCode);
                            value = resolvePromise(value);
                            jsFuncResultConsumer.accept(element, value);
                        } catch (PolyglotException e) {
                            onJsEvaluationException.accept(element, e, jsCode.toString());
                        }
                    } catch (IOException e) {
                        onJsEvaluationException.accept(null, e, "");
                    } finally {
                        context.leave();
                    }
                }
            } else {
                nonFuncResultConsumer.accept(element);
            }
        }
        if (out.size() > 0) {
            jsOutputConsumer.accept(out.toString(StandardCharsets.UTF_8));
        }
    }

    private static Context createContext(OutputStream out) {
        return Context.newBuilder("js")
                .engine(Engine.newBuilder()
                        .option("engine.WarnInterpreterOnly", "false")
                        .build())
                .allowExperimentalOptions(true)
                .option("js.foreign-object-prototype", "true")
                .option("js.global-property", "true")
                .out(out)
                .err(out)
                .build();
    }

    private static void setupContext(Context context) throws IOException {
        context.getBindings("js").putMember("isInstanceLike", new ProxyJacksonObject.InstanceCheck());
        context.eval("js",
                "Object.defineProperty(Object,Symbol.hasInstance, {\n" +
                        "  value: function myinstanceof(obj) {\n" +
                        "    return isInstanceLike(obj);\n" +
                        "  }\n" +
                        "});");
    }

    public static Value resolvePromise(Value value) {
        if (value.getMetaObject().getMetaSimpleName().equals("Promise") && value.hasMember("then")
                && value.canInvokeMember("then")) {
            List resolved = new ArrayList<>();
            List rejected = new ArrayList<>();
            Object invokeRtrn = value.invokeMember("then", new ProxyExecutable() {
                @Override
                public Object execute(Value... arguments) {
                    resolved.addAll(Arrays.asList(arguments));
                    return arguments;
                }
            }, new ProxyExecutable() {
                @Override
                public Object execute(Value... arguments) {
                    rejected.addAll(Arrays.asList(arguments));
                    return arguments;
                }
            });
            if (!rejected.isEmpty()) {
                value = rejected.get(0);
            } else if (resolved.size() == 1) {
                value = resolved.get(0);
            } else { //resolve.size() > 1, this doesn't happen
                log.error("resolved promise size=" + resolved.size() + ", expected 1 for promise = " + value);
            }
        }
        return value;
    }

    //I SWEAR IF I FIND ANOTHER PLACE THAT PERFORMS THE SAME CALCULATION I WILL BUY MORE SCREWDRIVERS
    static  T evaluateOnce(String function, JsonNode input, Function processResult,
            BiConsumer onException, Consumer onOutput) {
        StringBuilder jsCode = new StringBuilder("const __obj = ").append(input).append(";\n");
        jsCode.append("const __func = ").append(function).append(";\n");
        jsCode.append("__func(__obj)");
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        try (Context context = createContext(out)) {
            context.enter();
            try {
                setupContext(context);
                Value value = context.eval("js", jsCode);
                value = resolvePromise(value);
                //end of the sin
                return processResult.apply(value);
            } catch (PolyglotException e) {
                onException.accept(jsCode.toString(), e);
                return null;
            } catch (IOException e) {
                onException.accept(jsCode.toString(), e);
            } finally {
                if (out.size() > 0) {
                    onOutput.accept(out.toString());
                }
                context.leave();
            }
        }
        return null;
    }

    static boolean evaluateTest(String function, JsonNode input,
            Predicate onNotBoolean, BiConsumer onException, Consumer onOutput) {
        Boolean res = evaluateOnce(makeFilter(function), input, result -> {
            if (result.isBoolean()) {
                return result.asBoolean();
            } else {
                return onNotBoolean.test(result);
            }
        }, onException, onOutput);
        return res != null && res;
    }

    public static String makeFilter(String function) {
        return "__x => (!!(" + function + ")(__x))";
    }

    static public Object runQuery(EntityManager em, String query, Object... params) {
        return runQuery(em, Object.class, query, params);
    }

    static  T runQuery(EntityManager em, Class klass, String query, Object... params) {
        Query q;
        q = klass.equals(Object.class) ? em.createNativeQuery(query) : em.createNativeQuery(query, klass);
        for (int i = 0; i < params.length; ++i) {
            q.setParameter(i + 1, params[i]);
        }
        try {
            return (T) q.getSingleResult();
        } catch (NoResultException e) {
            log.errorf("No results in %s with params: %s", query, Arrays.asList(params));
            throw ServiceException.notFound("No result");
        } catch (Throwable t) {
            log.errorf(t, "Query error in %s with params: %s", query, Arrays.asList(params));
            throw t;
        }
    }

    public static Instant toInstant(Object time) {
        if (time == null) {
            return null;
        } else if (time instanceof Instant) {
            return (Instant) time; //crazier things happen
        } else if (time instanceof JsonNode) {
            JsonNode value = (JsonNode) time;
            if (value.isNumber()) {
                return Instant.ofEpochMilli(value.longValue());
            } else if (value.isTextual()) {
                time = value.asText(); //allow next set of ifs to check the value
            }
        }
        if (time instanceof Number) {
            return Instant.ofEpochMilli(((Number) time).longValue());
        } else {
            String str = time.toString().trim();
            if (str.isBlank()) {
                return null;
            }
            if (str.matches("\\d+")) {
                try {
                    return Instant.ofEpochMilli(Long.parseLong((String) time));
                } catch (NumberFormatException e) {
                    // noop
                }
            }
            //ISO_DATE we add midnight zulu offset
            if (str.matches("\\d{4}-\\d{2}-\\d{2}")) {
                str = str + "T00:00:00Z";
            }
            //ISO_LOCAL_DATE_TIME add zulu offset
            if (str.matches("\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}")) {
                str = str + "Z";
            }
            //ISO_DATE_TIME
            try {
                return ZonedDateTime.parse(str, DateTimeFormatter.ISO_DATE_TIME).toInstant();
            } catch (DateTimeParseException e) {
                log.debug("failed to convert " + time + " to timestamp using " + str);
            }
        }
        return null;//nothing matched
    }

    /**
     * used to check if an input can be cast to a target type in the db and return any error messages
     * Will return a row of all nulls if the input can be cast to the target type.
     */
    //tried pg_input_is_valid but it just returns boolean, no messages
    private static final String CHECK_CAST = "select * from pg_input_error_info(:input,:target)";

    public static record CheckResult(boolean ok, String message, String detail, String hint) {
    }

    /**
     * returns true (in the CheckResult) if the input can be cast to the target type in psql, otherwise it is false and details
     * are included
     *
     * @param input
     * @param target
     * @param em
     * @return
     */
    public static CheckResult castCheck(String input, String target, EntityManager em) {
        List results = null;
        // skip db query if the input is null or blank
        if (input != null && !input.isBlank()) {
            results = em.createNativeQuery(CHECK_CAST).setParameter("input", input).setParameter("target", target)
                    .unwrap(NativeQuery.class)
                    .addScalar("message", String.class)
                    .addScalar("detail", String.class)
                    .addScalar("hint", String.class)
                    .addScalar("sql_error_code", String.class)
                    .getResultList();
        }

        if (results == null) {
            return new CheckResult(
                    false,
                    "",
                    "",
                    "");
        }

        // no results or null result row or no message means it passed. no result and 0 length result should not happen but being defensive
        return results.isEmpty() || results.get(0).length == 0 || results.get(0)[0] == null ? new CheckResult(true, "", "", "")
                : new CheckResult(
                        false,
                        results.get(0)[0] == null ? "" : results.get(0)[0].toString(),
                        results.get(0)[1] == null ? "" : results.get(0)[1].toString(),
                        results.get(0)[2] == null ? "" : results.get(0)[2].toString());
    }

    public static DecomposedJsonPath decomposeJsonPath(String jsonpath) {
        Matcher matcher = JSONPATH_ROOT_PATTERN.matcher(jsonpath);
        if (matcher.find()) {
            String root = matcher.group(1).replaceAll("^\"|\"$", "");
            // Remove root key from the original JSONPath
            String modifiedJsonPath = jsonpath.replaceFirst(Pattern.quote(matcher.group(0)), "\\$");
            return new DecomposedJsonPath(root, modifiedJsonPath);
        }
        return null;
    }

    /**
     * returns null if no filtering, otherwise returns an object for filtering
     *
     * @param input filter string
     * @return JsonNode, original string or null
     */
    public static JsonNode getFilterObject(String input) {
        if (input == null || input.isBlank()) {
            // not a valid filter
            return null;
        }
        JsonNode filterJson = null;
        try {
            filterJson = new ObjectMapper().readTree(input);
        } catch (JsonProcessingException e) {
            // TODO what to do with this error
        }
        if (filterJson != null && filterJson.getNodeType() == JsonNodeType.OBJECT) {
            return filterJson;
        } else {
            // do we need this?? the ObjectMapper().readTree(input) should already parse the object properly
            // TODO validate the jsonpath?
            return JsonNodeFactory.instance.textNode(input);
        }
    }

    interface ExecutionExceptionConsumer {
        void accept(T row, Throwable exception, String code);
    }

    public static void registerTxSynchronization(TransactionManager tm, IntConsumer consumer) {
        try {
            if (tm.getStatus() != Status.STATUS_NO_TRANSACTION) {
                Transaction tx = tm.getTransaction();
                tx.registerSynchronization(new Synchronization() {
                    @Override
                    public void beforeCompletion() {
                        //do nothing
                    }

                    @Override
                    public void afterCompletion(int status) {

                        try {
                            consumer.accept(status);
                        } catch (Exception e) {
                            log.errorf("Tx Synchronization callback failed: %s", e.getMessage());
                        }
                    }
                });
            } else {
                consumer.accept(0);
            }
        } catch (SystemException | RollbackException e) {
            log.errorf("Error occurred in transaction: %s", e.getMessage());
            //         throw new RuntimeException(e);
            consumer.accept(0);
        } catch (Exception e) {
            log.errorf("Error occurred processing consumer: %s", e.getMessage());
        }

    }

    public record DecomposedJsonPath(String root, String jsonpath) {
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy