aboutsummaryrefslogtreecommitdiffstats
path: root/subprojects/store-query-viatra/src/main/java/tools/refinery/store/query/viatra/internal/update/SymbolViewUpdateListener.java
blob: f1a2ac7c2ba0433dd9c378a6635df1fb10cfdbd0 (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
/*
 * SPDX-FileCopyrightText: 2021-2023 The Refinery Authors <https://refinery.tools/>
 *
 * SPDX-License-Identifier: EPL-2.0
 */
package tools.refinery.store.query.viatra.internal.update;

import org.eclipse.viatra.query.runtime.matchers.context.IInputKey;
import org.eclipse.viatra.query.runtime.matchers.context.IQueryRuntimeContextListener;
import org.eclipse.viatra.query.runtime.matchers.tuple.ITuple;
import org.eclipse.viatra.query.runtime.matchers.tuple.Tuple;
import tools.refinery.store.model.Interpretation;
import tools.refinery.store.model.InterpretationListener;
import tools.refinery.store.query.viatra.internal.ViatraModelQueryAdapterImpl;
import tools.refinery.store.query.view.SymbolView;
import tools.refinery.store.query.view.TuplePreservingView;

import java.util.ArrayList;
import java.util.List;

public abstract class SymbolViewUpdateListener<T> implements InterpretationListener<T> {
	private final ViatraModelQueryAdapterImpl adapter;
	private final Interpretation<T> interpretation;
	private final List<RelationViewFilter> filters = new ArrayList<>();

	protected SymbolViewUpdateListener(ViatraModelQueryAdapterImpl adapter, Interpretation<T> interpretation) {
		this.adapter = adapter;
		this.interpretation = interpretation;
	}

	public void addFilter(IInputKey inputKey, ITuple seed, IQueryRuntimeContextListener listener) {
		if (filters.isEmpty()) {
			// First filter to be added, from now on we have to subscribe to model updates.
			interpretation.addListener(this, true);
		}
		filters.add(new RelationViewFilter(inputKey, seed, listener));
	}

	public void removeFilter(IInputKey inputKey, ITuple seed, IQueryRuntimeContextListener listener) {
		if (filters.remove(new RelationViewFilter(inputKey, seed, listener)) && filters.isEmpty()) {
			// Last listener to be added, we don't have be subscribed to model updates anymore.
			interpretation.removeListener(this);
		}
	}

	protected void processUpdate(Tuple tuple, boolean isInsertion) {
		adapter.markAsPending();
		int size = filters.size();
		// Use a for loop instead of a for-each loop to avoid <code>Iterator</code> allocation overhead.
		//noinspection ForLoopReplaceableByForEach
		for (int i = 0; i < size; i++) {
			filters.get(i).update(tuple, isInsertion);
		}
	}

	public static <T> SymbolViewUpdateListener<T> of(ViatraModelQueryAdapterImpl adapter,
													 SymbolView<T> view,
													 Interpretation<T> interpretation) {
		if (view instanceof TuplePreservingView<T> tuplePreservingRelationView) {
			return new TuplePreservingViewUpdateListener<>(adapter, tuplePreservingRelationView,
					interpretation);
		}
		return new TupleChangingViewUpdateListener<>(adapter, view, interpretation);
	}
}