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

com.att.research.xacmlatt.pdp.policy.expressions.QuantifiedExpression Maven / Gradle / Ivy

/*
 * Copyright (c) 2021, salesforce.com, inc.
 * All rights reserved.
 * SPDX-License-Identifier: MIT
 * For full license text, see the LICENSE file in the repo root or https://opensource.org/licenses/MIT
 */
package com.att.research.xacmlatt.pdp.policy.expressions;

import com.att.research.xacml.api.AttributeValue;
import com.att.research.xacml.api.trace.Traceable;
import com.att.research.xacml.std.StdStatus;
import com.att.research.xacml.std.StdStatusCode;
import com.att.research.xacml.std.trace.StdTraceEvent;
import com.att.research.xacmlatt.pdp.eval.EvaluationContext;
import com.att.research.xacmlatt.pdp.eval.EvaluationException;
import com.att.research.xacmlatt.pdp.policy.*;
import com.google.common.collect.Iterators;

import java.util.Iterator;

/**
 * QuantifiedExpression extends {@link Expression} to provide the base behavior specified in section 5 of the
 * 
 * XACML v3.0 Related and Nested Entities Profile Version 1.0 specification.
 *
 * @author ygrignon
 */
public abstract class QuantifiedExpression extends Expression implements LexicalEnvironment {
    protected static ExpressionResult ER_CONTINUE_PROCESSING = null;

    private LexicalEnvironment lexicalEnvironment;
    private VariableDefinition quantifiedVariable;
    private ThreadLocalExpression domainValue;
    private Expression domainExpression;
    private Expression iterantExpression;

    /**
     * Constructs a quantified expression within a {@link LexicalEnvironment}.
     * @param lexicalEnvironment The parent lexical environment.
     */
    protected QuantifiedExpression(LexicalEnvironment lexicalEnvironment) {
        this.lexicalEnvironment = lexicalEnvironment;
        this.domainValue = new ThreadLocalExpression();
        this.quantifiedVariable = new VariableDefinition();
        this.quantifiedVariable.setExpression(this.domainValue);
    }

    public Expression getDomainExpression() {
        return this.domainExpression;
    }

    public void setDomainExpression(Expression domainExpression) {
        this.domainExpression = domainExpression;
    }

    public Expression getIterantExpression() {
        return this.iterantExpression;
    }

    public void setIterantExpression(Expression iterantExpression) {
        this.iterantExpression = iterantExpression;
    }

    public String getVariableId() {
        return this.quantifiedVariable.getId();
    }

    public void setVariableId(String variableId) {
        this.quantifiedVariable.setId(variableId);
    }

    protected VariableDefinition getQuantifiedVariable() {
        return this.quantifiedVariable;
    }

    protected LexicalEnvironment getOuterLexicalEnvironment() {
        return this.lexicalEnvironment;
    }

    @Override
    public Iterator getVariableDefinitions() {
        return Iterators.concat(
                Iterators.singletonIterator(getQuantifiedVariable()),
                getOuterLexicalEnvironment().getVariableDefinitions());
    }

    /**
     * Gets the VariableDefinition for the given String variable identifier.
     *
     * @param variableId the String variable identifier
     * @return the VariableDefinition with the given String identifier or null if not found
     */
    public VariableDefinition getVariableDefinition(String variableId) {
        if (variableId != null) {
            if (variableId.equals(this.getVariableId())) {
                return getQuantifiedVariable();
            } else {
                return getOuterLexicalEnvironment().getVariableDefinition(variableId);
            }
        }
        return null;
    }

    /**
     * Evaluates this QuantifiedExpression in the given {@link EvaluationContext}.
     *
     * @param evaluationContext the EvaluationContext in which to evaluate this QuantifiedExpression
     * @param policyDefaults the {@link PolicyDefaults} to use in evaluating this QuantifiedExpression
     * @return an {@link ExpressionResult}
     * @throws EvaluationException if the evaluation failed.
     */
    @Override
    public ExpressionResult evaluate(EvaluationContext evaluationContext, PolicyDefaults policyDefaults) throws EvaluationException {
        if (!this.validate()) {
            return ExpressionResult.newError(new StdStatus(this.getStatusCode(), this.getStatusMessage()));
        }

        // Evaluate the domain expression
        ExpressionResult domainResult = getDomainExpression().evaluate(evaluationContext, policyDefaults);
        assert domainResult != null;
        if (!domainResult.isOk()) {
            return ExpressionResult.newError(domainResult.getStatus());
        }
        if (!domainResult.isBag()) {
            return ExpressionResult.newError(new StdStatus(StdStatusCode.STATUS_CODE_PROCESSING_ERROR, "Domain didn't produce a bag"));
        }

        // Evaluate the iterant expression for each domain values
        return processDomainResult(domainResult.getBag(), null, evaluationContext, policyDefaults);
    }

    /**
     * Evaluates the iterant expression over each value of the domain. Execution might terminate early if a final result
     * is determined prior to evaluating every domain values. Subclasses must override this method in order to make sure
     * an actual result is provided when the end of the domain is reached.
     *
     * @param domain The domain.
     * @param resultBag The result bag provided by some subclass override. This is simply passed through to
     *     {@link #processIterantResult(AttributeValue, ExpressionResult, Bag)}.
     * @param evaluationContext the EvaluationContext in which to evaluate the iterant expression.
     * @param policyDefaults the {@link PolicyDefaults} to use in evaluating the iterant expression.
     * @return the final result provided by {@link #processIterantResult(AttributeValue, ExpressionResult, Bag)} if
     *     there is one, {@link #ER_CONTINUE_PROCESSING} if the end of the domain has been reached.
     * @throws EvaluationException if the iterant expression evaluation failed.
     */
    protected ExpressionResult processDomainResult(Bag domain, Bag resultBag, EvaluationContext evaluationContext, PolicyDefaults policyDefaults) throws EvaluationException {
        if (domain != null) {
            try {
                for (AttributeValue attributeValue : domain.getAttributeValueList()) {
                    // Set the quantified variable to the current domain value
                    domainValue.set(ExpressionResult.newSingle(attributeValue));

                    // Execute the iterant expression for the current domain value
                    ExpressionResult iterantResult = getIterantExpression().evaluate(evaluationContext, policyDefaults);
                    assert iterantResult != null;
                    if (evaluationContext.isTracing()) {
                        evaluationContext.trace(new StdTraceEvent<>(attributeValue.getValue().toString(), this, iterantResult));
                    }
                    if (!iterantResult.isOk()) {
                        return iterantResult;
                    }

                    // Apply the quantified expression behavior to the iterant expression result
                    ExpressionResult result = processIterantResult(attributeValue, iterantResult, resultBag);
                    if (result != ER_CONTINUE_PROCESSING) {
                        return result;
                    }
                }
            }
            finally {
                domainValue.clear();
            }
        }
        return ER_CONTINUE_PROCESSING;
    }

    /**
     * Apply the quantified expression behavior for this iteration of the evaluation.
     *
     * @param domainAttributeValue The domain value.
     * @param iterantResult The iterant expression result.
     * @param resultBag The result bag.
     * @return the final expression result if one is reached, {{@link #ER_CONTINUE_PROCESSING}} if evaluation should continue.
     */
    protected abstract ExpressionResult processIterantResult(AttributeValue domainAttributeValue, ExpressionResult iterantResult, Bag resultBag);

    @Override
    protected boolean validateComponent() {
        if (this.getDomainExpression() == null) {
            this.setStatus(StdStatusCode.STATUS_CODE_SYNTAX_ERROR, "Missing domain expression");
            return false;
        }
        if (this.getIterantExpression() == null) {
            this.setStatus(StdStatusCode.STATUS_CODE_SYNTAX_ERROR, "Missing iterant expression");
            return false;
        }
        if (this.getVariableId() == null) {
            this.setStatus(StdStatusCode.STATUS_CODE_SYNTAX_ERROR, "Missing VariableId");
            return false;
        }
        if (this.getOuterLexicalEnvironment().getVariableDefinition(getVariableId()) != null) {
            this.setStatus(StdStatusCode.STATUS_CODE_SYNTAX_ERROR, "Duplicate VariableId");
            return false;
        }
        this.setStatus(StdStatusCode.STATUS_CODE_OK, null);
        return true;
    }

    @Override
    public String getTraceId() {
        return this.getVariableId();
    }

    @Override
    public Traceable getCause() {
        return getOuterLexicalEnvironment();
    }

    /**
     * An Expression that returns the set result for a given thread.
     */
    private static class ThreadLocalExpression extends Expression {
        private ThreadLocal tl = new ThreadLocal<>();

        /**
         * Sets the result for the current thread.
         * @param result The result.
         */
        public void set(ExpressionResult result) {
            tl.set(result);
        }

        /**
         * Clears the result for the current thread.
         */
        public void clear() {
            tl.remove();
        }

        /**
         * Returns the result for the current thread.
         * @param evaluationContext the EvaluationContext in which to evaluate this Expression
         * @param policyDefaults the {@link com.att.research.xacmlatt.pdp.policy.PolicyDefaults} to use in evaluating this Expression
         * @return the result for this thread.
         */
        @Override
        public ExpressionResult evaluate(EvaluationContext evaluationContext, PolicyDefaults policyDefaults) {
            ExpressionResult result = tl.get();
            return result != null ? result : ExpressionResult.newError(new StdStatus(StdStatusCode.STATUS_CODE_PROCESSING_ERROR, "No result set for current thread"));
        }

        @Override
        protected boolean validateComponent() {
            return tl.get() != null;
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy