io.parsingdata.metal.expression.value.Expand 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.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;
/**
* A {@link ValueExpression} that expands a result by copying and concatenating
* it a specified amount of times.
*
* An Expand expression has two operands: bases
(a
* {@link ValueExpression}) and count
(a
* {@link SingleValueExpression}). Both operands are evaluated. Multiple copies
* of the result of evaluating bases
are concatenated. The amount
* of copies equals the result of evaluating count
. If
* count
evaluated to an empty value or NOT_A_VALUE
,
* an {@link IllegalArgumentException} is thrown.
*/
public class Expand extends ImmutableObject implements ValueExpression {
public final ValueExpression bases;
public final SingleValueExpression count;
public Expand(final ValueExpression bases, final SingleValueExpression count) {
this.bases = checkNotNull(bases, "bases");
this.count = checkNotNull(count, "count");
}
@Override
public ImmutableList eval(final ParseState parseState, final Encoding encoding) {
final ImmutableList baseList = bases.eval(parseState, encoding);
if (baseList.isEmpty()) {
return baseList;
}
return count.evalSingle(parseState, encoding)
.filter(countValue -> !countValue.equals(NOT_A_VALUE))
.map(countValue -> expand(baseList, countValue.asNumeric().intValueExact(), new ImmutableList<>()).computeResult())
.orElseThrow(() -> new IllegalArgumentException("Count must evaluate to a non-empty countable value."));
}
private Trampoline> expand(final ImmutableList baseList, final int countValue, final ImmutableList aggregate) {
if (countValue < 1) {
return complete(() -> aggregate);
}
return intermediate(() -> expand(baseList, countValue - 1, aggregate.add(baseList)));
}
@Override
public String toString() {
return getClass().getSimpleName() + "(" + bases + "," + count + ")";
}
@Override
public boolean equals(final Object obj) {
return Util.notNullAndSameClass(this, obj)
&& Objects.equals(bases, ((Expand)obj).bases)
&& Objects.equals(count, ((Expand)obj).count);
}
@Override
public int immutableHashCode() {
return Objects.hash(getClass(), bases, count);
}
}