aboutsummaryrefslogtreecommitdiffstats
path: root/subprojects/store-query-interpreter/src/main/java/tools/refinery/store/query/interpreter/internal/matcher/InterpretedRelationalMatcher.java
blob: 9d9c30428afd783d6677279c79aa0163a94cb681 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
/*
 * SPDX-FileCopyrightText: 2021-2023 The Refinery Authors <https://refinery.tools/>
 *
 * SPDX-License-Identifier: EPL-2.0
 */
package tools.refinery.store.query.interpreter.internal.matcher;

import tools.refinery.interpreter.matchers.context.IQueryRuntimeContext;
import tools.refinery.interpreter.matchers.tuple.TupleMask;
import tools.refinery.interpreter.matchers.tuple.Tuples;
import tools.refinery.interpreter.rete.index.Indexer;
import tools.refinery.interpreter.rete.matcher.RetePatternMatcher;
import tools.refinery.store.map.Cursor;
import tools.refinery.store.map.Cursors;
import tools.refinery.logic.dnf.RelationalQuery;
import tools.refinery.store.query.interpreter.internal.QueryInterpreterAdapterImpl;
import tools.refinery.store.tuple.Tuple;

/**
 * Directly access the tuples inside a Refinery Interpreter pattern matcher.<p>
 * This class neglects calling
 * {@link IQueryRuntimeContext#wrapTuple(tools.refinery.interpreter.matchers.tuple.Tuple)}
 * and
 * {@link IQueryRuntimeContext#unwrapTuple(tools.refinery.interpreter.matchers.tuple.Tuple)},
 * because {@link tools.refinery.store.query.interpreter.internal.context.RelationalRuntimeContext} provides a trivial
 * implementation for these methods.
 * Using this class with any other runtime context may lead to undefined behavior.
 */
public class InterpretedRelationalMatcher extends AbstractInterpretedMatcher<Boolean> {
	private final TupleMask emptyMask;
	private final TupleMask identityMask;
	private final Indexer emptyMaskIndexer;

	public InterpretedRelationalMatcher(QueryInterpreterAdapterImpl adapter, RelationalQuery query,
										RawPatternMatcher rawPatternMatcher) {
		super(adapter, query, rawPatternMatcher);
		int arity = query.arity();
		emptyMask = TupleMask.empty(arity);
		identityMask = TupleMask.identity(arity);
		if (backend instanceof RetePatternMatcher reteBackend) {
			emptyMaskIndexer = reteBackend.getInternalIndexer(emptyMask);
		} else {
			emptyMaskIndexer = null;
		}
	}

	@Override
	public Boolean get(Tuple parameters) {
		var tuple = MatcherUtils.toViatraTuple(parameters);
		if (emptyMaskIndexer == null) {
			return backend.hasMatch(identityMask, tuple);
		}
		var matches = emptyMaskIndexer.get(Tuples.staticArityFlatTupleOf());
		return matches != null && matches.contains(tuple);
	}

	@Override
	public Cursor<Tuple, Boolean> getAll() {
		if (emptyMaskIndexer == null) {
			var allMatches = backend.getAllMatches(emptyMask, Tuples.staticArityFlatTupleOf());
			return new RelationalCursor(allMatches.iterator());
		}
		var matches = emptyMaskIndexer.get(Tuples.staticArityFlatTupleOf());
		return matches == null ? Cursors.empty() : new RelationalCursor(matches.stream().iterator());
	}

	@Override
	public int size() {
		if (emptyMaskIndexer == null) {
			return backend.countMatches(emptyMask, Tuples.staticArityFlatTupleOf());
		}
		var matches = emptyMaskIndexer.get(Tuples.staticArityFlatTupleOf());
		return matches == null ? 0 : matches.size();
	}

	@Override
	public void update(tools.refinery.interpreter.matchers.tuple.Tuple updateElement, boolean isInsertion) {
		var key = MatcherUtils.toRefineryTuple(updateElement);
		notifyChange(key, !isInsertion, isInsertion);
	}
}