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

io.parsingdata.metal.expression.comparison.ComparisonExpression Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2013-2024 Netherlands Forensic Institute
 * Copyright 2021-2024 Infix Technologies B.V.
 *
 * 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 io.parsingdata.metal.expression.comparison;

import static io.parsingdata.metal.Trampoline.complete;
import static io.parsingdata.metal.Trampoline.intermediate;
import static io.parsingdata.metal.Util.checkNotNull;
import static io.parsingdata.metal.expression.value.NotAValue.NOT_A_VALUE;

import java.util.Objects;

import io.parsingdata.metal.ImmutableObject;
import io.parsingdata.metal.Trampoline;
import io.parsingdata.metal.Util;
import io.parsingdata.metal.data.ImmutableList;
import io.parsingdata.metal.data.ParseState;
import io.parsingdata.metal.encoding.Encoding;
import io.parsingdata.metal.expression.Expression;
import io.parsingdata.metal.expression.value.Value;
import io.parsingdata.metal.expression.value.ValueExpression;

/**
 * Base class for all ComparisonExpression implementations.
 * 

* Specifies two input {@link ValueExpression}s that are to be evaluated and * their results compared. Handles evaluation of inputs and resulting lists. * The abstract {@link #compare(Value, Value)} method is then called for * individual values. *

* The {@link #value} argument may be null, in which case it is * not evaluated and the output value is substituted with a list containing * only the {@link Value} most recently added to the {@link ParseState}. */ public abstract class ComparisonExpression extends ImmutableObject implements Expression { public final ValueExpression value; public final ValueExpression predicate; protected ComparisonExpression(final ValueExpression value, final ValueExpression predicate) { this.value = value; this.predicate = checkNotNull(predicate, "predicate"); } @Override public boolean eval(final ParseState parseState, final Encoding encoding) { final ImmutableList values = value == null ? parseState.order.current() .map(ImmutableList::create) .orElseGet(ImmutableList::new) : value.eval(parseState, encoding); if (values.isEmpty()) { return false; } final ImmutableList predicates = predicate.eval(parseState, encoding); if (values.size != predicates.size) { return false; } return compare(values, predicates).computeResult(); } private Trampoline compare(final ImmutableList currents, final ImmutableList predicates) { if (currents.head.equals(NOT_A_VALUE) || predicates.head.equals(NOT_A_VALUE)) { return complete(() -> false); } final boolean headResult = compare(currents.head, predicates.head); if (!headResult || currents.tail.isEmpty()) { return complete(() -> headResult); } return intermediate(() -> compare(currents.tail, predicates.tail)); } public abstract boolean compare(final Value left, final Value right); @Override public String toString() { return getClass().getSimpleName() + "(" + (value == null ? "" : value + ",") + predicate + ")"; } @Override public boolean equals(final Object obj) { return Util.notNullAndSameClass(this, obj) && Objects.equals(value, ((ComparisonExpression)obj).value) && Objects.equals(predicate, ((ComparisonExpression)obj).predicate); } @Override public int immutableHashCode() { return Objects.hash(getClass(), value, predicate); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy