io.parsingdata.metal.expression.value.BinaryValueExpression Maven / Gradle / Ivy
Show all versions of metal-core Show documentation
/*
* 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.value;
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.data.Selection.reverse;
import static io.parsingdata.metal.expression.value.NotAValue.NOT_A_VALUE;
import java.util.Objects;
import java.util.Optional;
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;
/**
* Base class for {@link ValueExpression}s with two operands.
*
* A BinaryValueExpression implements a ValueExpression that has two operands:
* left
and right
(both {@link ValueExpression}s).
* Both operands are themselves first evaluated. If at least one of the
* operands evaluates to {@link Optional#empty()}, the result of the
* ValueExpression itself will be empty as well.
*
* For lists, values with the same index are evaluated in this manner. If
* lists are of unequal length, the result is a list with evaluated values the
* same size as the shortest list, appended with instances of NOT_A_VALUE to
* match the size of the longest list.
*
* To implement a BinaryValueExpression, only the
* {@link #eval(Value, Value, ParseState, Encoding)} must be implemented,
* handling the case of evaluating two values. This base class takes care of
* evaluating the operands and handling list semantics.
*
* @see UnaryValueExpression
*/
public abstract class BinaryValueExpression extends ImmutableObject implements ValueExpression {
public final ValueExpression left;
public final ValueExpression right;
protected BinaryValueExpression(final ValueExpression left, final ValueExpression right) {
this.left = checkNotNull(left, "left");
this.right = checkNotNull(right, "right");
}
public abstract Optional eval(final Value leftValue, final Value rightValue, final ParseState parseState, final Encoding encoding);
@Override
public ImmutableList eval(final ParseState parseState, final Encoding encoding) {
return evalLists(left.eval(parseState, encoding), right.eval(parseState, encoding), parseState, encoding);
}
private ImmutableList evalLists(final ImmutableList leftValues, final ImmutableList rightValues, final ParseState parseState, final Encoding encoding) {
return reverse(padList(evalLists(leftValues, rightValues, parseState, encoding, new ImmutableList<>()).computeResult(), Math.abs(leftValues.size - rightValues.size)).computeResult());
}
private Trampoline> evalLists(final ImmutableList leftValues, final ImmutableList rightValues, final ParseState parseState, final Encoding encoding, final ImmutableList result) {
if (leftValues.isEmpty() || rightValues.isEmpty()) {
return complete(() -> result);
}
return intermediate(() -> evalLists(leftValues.tail, rightValues.tail, parseState, encoding, result.add(safeEval(leftValues.head, rightValues.head, parseState, encoding))));
}
private Trampoline> padList(final ImmutableList list, final long size) {
if (size <= 0) {
return complete(() -> list);
}
return intermediate(() -> padList(list.add(NOT_A_VALUE), size - 1));
}
private Value safeEval(final Value leftValue, final Value rightValue, final ParseState parseState, final Encoding encoding) {
if (leftValue.equals(NOT_A_VALUE) || rightValue.equals(NOT_A_VALUE)) {
return NOT_A_VALUE;
}
return eval(leftValue, rightValue, parseState, encoding)
.orElse(NOT_A_VALUE);
}
@Override
public String toString() {
return getClass().getSimpleName() + "(" + left + "," + right + ")";
}
@Override
public boolean equals(final Object obj) {
return Util.notNullAndSameClass(this, obj)
&& Objects.equals(left, ((BinaryValueExpression)obj).left)
&& Objects.equals(right, ((BinaryValueExpression)obj).right);
}
@Override
public int immutableHashCode() {
return Objects.hash(getClass(), left, right);
}
}