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

org.deltafi.common.rules.RuleEvaluator Maven / Gradle / Ivy

/*
 *    DeltaFi - Data transformation and enrichment platform
 *
 *    Copyright 2021-2023 DeltaFi Contributors 
 *
 *    Licensed under the Apache License, Version 2.0 (the "License");
 *    you may not use this file except in compliance with the License.
 *    You may obtain a copy of the License at
 *
 *        http://www.apache.org/licenses/LICENSE-2.0
 *
 *    Unless required by applicable law or agreed to in writing, software
 *    distributed under the License is distributed on an "AS IS" BASIS,
 *    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *    See the License for the specific language governing permissions and
 *    limitations under the License.
 */
package org.deltafi.common.rules;

import lombok.extern.slf4j.Slf4j;
import org.deltafi.common.types.Content;
import org.deltafi.common.types.DeltaFileFlow;
import org.springframework.expression.EvaluationContext;
import org.springframework.expression.Expression;
import org.springframework.expression.spel.standard.SpelExpressionParser;
import org.springframework.expression.spel.support.SimpleEvaluationContext;
import org.springframework.stereotype.Service;

import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;

@Slf4j
@Service
public class RuleEvaluator {
    private static final ImmutableDeltaFileFlow EXAMPLE_DELTA_FILE = new ImmutableDeltaFileFlow(Map.of("a", "b"), List.of(new Content("", "")));

    private final Map expressionCache = new ConcurrentHashMap<>();

    /**
     * Validate the condition loads and can be evaluated against a sample DeltaFile
     * @param condition to validate
     * @throws IllegalArgumentException if the condition results in an exception
     */
    public void validateCondition(String condition) throws IllegalArgumentException {
        try {
            doEvaluateCondition(condition, EXAMPLE_DELTA_FILE);
        } catch (Exception e) {
            String exceptionMessage = e.getMessage();
            String message = exceptionMessage != null ? "invalid condition `" + condition + "`: " + exceptionMessage :
                    "invalid condition `" + condition + "`";
            throw new IllegalArgumentException(message);
        }
    }

    /**
     * Evaluate the condition against the given DeltaFile. A null condition
     * will result in true. An invalid condition results in false.
     * @param condition SPeL expression to evaluate
     * @param deltaFileFlow used in the evaluation
     * @return the result of evaluating the condition
     */
    public boolean evaluateCondition(String condition, DeltaFileFlow deltaFileFlow) {
        try {
            return doEvaluateCondition(condition, new ImmutableDeltaFileFlow(deltaFileFlow));
        } catch (Exception e) {
            log.error("Failed to evaluate condition", e);
            return false;
        }
    }

    boolean doEvaluateCondition(String condition, ImmutableDeltaFileFlow deltaFileFlow) {
        if (condition == null) {
            return true;
        }

        EvaluationContext spelContext = SimpleEvaluationContext.forReadOnlyDataBinding()
                .withRootObject(deltaFileFlow)
                .withInstanceMethods()
                .build();

        Expression expression = expressionCache.computeIfAbsent(condition, this::newExpression);
        Boolean output = expression.getValue(spelContext, Boolean.class);

        return Boolean.TRUE.equals(output);
    }

    private Expression newExpression(String condition) {
        SpelExpressionParser parser = new SpelExpressionParser();
        return parser.parseExpression(condition);
    }

    /**
     * Holds a copy of the DeltaFileFlow metadata and content list to prevent
     * SpEL expressions from modifying the original DeltaFile
     * @param metadata to use when evaluating conditions
     * @param content to use when evaluating conditions
     */
    public record ImmutableDeltaFileFlow(Map metadata, List content) {
        public ImmutableDeltaFileFlow(DeltaFileFlow deltaFileFlow) {
            this(deltaFileFlow.getImmutableMetadata(), deltaFileFlow.getImmutableContent());
        }

        /**
         * Helper that can be used to check if DeltaFile contains content with the given mediaType
         * @param mediaType to find in the list of content
         * @return true if there is a content with the given mediaType
         */
        public boolean hasMediaType(String mediaType) {
            return this.content.stream().anyMatch(content -> Objects.equals(mediaType, content.getMediaType()));
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy