/*
* SPDX-FileCopyrightText: 2023 The Refinery Authors
*
* SPDX-License-Identifier: EPL-2.0
*/
package tools.refinery.store.reasoning.scope;
import com.google.ortools.linearsolver.MPConstraint;
import tools.refinery.store.model.ModelStoreBuilder;
import tools.refinery.store.query.ModelQueryBuilder;
import tools.refinery.logic.dnf.AnyQuery;
import tools.refinery.logic.dnf.RelationalQuery;
import tools.refinery.store.query.resultset.ResultSet;
import tools.refinery.store.tuple.Tuple;
import java.util.Collection;
abstract class TypeScopePropagator {
private final BoundScopePropagator adapter;
private final ResultSet allNodes;
private final ResultSet multiNodes;
protected final MPConstraint constraint;
protected TypeScopePropagator(BoundScopePropagator adapter, RelationalQuery allQuery,
RelationalQuery multiQuery) {
this.adapter = adapter;
var queryEngine = adapter.getQueryEngine();
allNodes = queryEngine.getResultSet(allQuery);
multiNodes = queryEngine.getResultSet(multiQuery);
constraint = adapter.makeConstraint();
constraint.setBounds(0, Double.POSITIVE_INFINITY);
var cursor = multiNodes.getAll();
while (cursor.move()) {
var variable = adapter.getVariable(cursor.getKey().get(0));
constraint.setCoefficient(variable, 1);
}
allNodes.addListener(this::allChanged);
multiNodes.addListener(this::multiChanged);
}
protected abstract void doUpdateBounds();
public boolean updateBounds() {
doUpdateBounds();
return constraint.lb() <= constraint.ub();
}
protected int getSingleCount() {
return allNodes.size() - multiNodes.size();
}
private void allChanged(Tuple ignoredKey, Boolean ignoredOldValue, Boolean ignoredNewValue) {
adapter.markAsChanged();
}
private void multiChanged(Tuple key, Boolean ignoredOldValue, Boolean newValue) {
var variable = adapter.getVariable(key.get(0));
constraint.setCoefficient(variable, Boolean.TRUE.equals(newValue) ? 1 : 0);
adapter.markAsChanged();
}
abstract static class Factory {
public abstract TypeScopePropagator createPropagator(BoundScopePropagator adapter);
protected abstract Collection getQueries();
public void configure(ModelStoreBuilder storeBuilder) {
storeBuilder.getAdapter(ModelQueryBuilder.class).queries(getQueries());
}
}
}