graphql.util.TreeParallelTransformer Maven / Gradle / Ivy
package graphql.util;
import graphql.Internal;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CountedCompleter;
import java.util.concurrent.ForkJoinPool;
import static graphql.Assert.assertNotEmpty;
import static graphql.Assert.assertNotNull;
import static graphql.Assert.assertTrue;
import static graphql.util.NodeZipper.ModificationType.REPLACE;
import static graphql.util.TraversalControl.ABORT;
import static graphql.util.TraversalControl.CONTINUE;
import static graphql.util.TraversalControl.QUIT;
@Internal
public class TreeParallelTransformer {
private final Map, Object> rootVars = new ConcurrentHashMap<>();
private final ForkJoinPool forkJoinPool;
private final NodeAdapter nodeAdapter;
private Object sharedContextData;
private TreeParallelTransformer(Object sharedContextData,
ForkJoinPool forkJoinPool,
NodeAdapter nodeAdapter) {
this.sharedContextData = sharedContextData;
this.forkJoinPool = forkJoinPool;
this.nodeAdapter = nodeAdapter;
}
public static TreeParallelTransformer parallelTransformer(NodeAdapter nodeAdapter) {
return parallelTransformer(nodeAdapter, ForkJoinPool.commonPool());
}
public static TreeParallelTransformer parallelTransformer(NodeAdapter nodeAdapter, ForkJoinPool forkJoinPool) {
return new TreeParallelTransformer<>(null, forkJoinPool, nodeAdapter);
}
public TreeParallelTransformer rootVars(Map, Object> rootVars) {
this.rootVars.putAll(assertNotNull(rootVars));
return this;
}
public TreeParallelTransformer rootVar(Class> key, Object value) {
rootVars.put(key, value);
return this;
}
public T transform(T root, TraverserVisitor super T> visitor) {
return transformImpl(root, visitor);
}
public DefaultTraverserContext newRootContext(Map, Object> vars) {
return newContextImpl(null, null, vars, null, true);
}
public T transformImpl(T root, TraverserVisitor super T> visitor) {
assertNotNull(root);
assertNotNull(visitor);
DefaultTraverserContext rootContext = newRootContext(rootVars);
DefaultTraverserContext context = newContext(root, rootContext, null);
EnterAction enterAction = new EnterAction(null, context, visitor);
T result = (T) forkJoinPool.invoke(enterAction);
return result;
}
private class EnterAction extends CountedCompleter {
private DefaultTraverserContext currentContext;
private TraverserVisitor super T> visitor;
private List children;
private List> myZippers = new LinkedList<>();
private T result;
private EnterAction(CountedCompleter parent, DefaultTraverserContext currentContext, TraverserVisitor super T> visitor) {
super(parent);
this.currentContext = currentContext;
this.visitor = visitor;
}
@Override
public void compute() {
currentContext.setPhase(TraverserContext.Phase.ENTER);
currentContext.setVar(List.class, myZippers);
TraversalControl traversalControl = visitor.enter(currentContext);
assertNotNull(traversalControl, () -> "result of enter must not be null");
assertTrue(QUIT != traversalControl, () -> "can't return QUIT for parallel traversing");
if (traversalControl == ABORT) {
this.children = Collections.emptyList();
tryComplete();
return;
}
assertTrue(traversalControl == CONTINUE);
this.children = pushAll(currentContext);
if (children.size() == 0) {
tryComplete();
return;
}
setPendingCount(children.size() - 1);
for (int i = 1; i < children.size(); i++) {
new EnterAction(this, children.get(i), visitor).fork();
}
new EnterAction(this, children.get(0), visitor).compute();
}
@Override
public void onCompletion(CountedCompleter caller) {
if (currentContext.isDeleted()) {
this.result = null;
return;
}
List> childZippers = new LinkedList<>();
for (DefaultTraverserContext childContext : this.children) {
childZippers.addAll((Collection extends NodeZipper>) childContext.getVar(List.class));
}
if (childZippers.size() > 0) {
NodeZipper newNode = moveUp((T) currentContext.thisNode(), childZippers);
myZippers.add(newNode);
this.result = (T) newNode.getCurNode();
} else if (currentContext.isChanged()) {
NodeZipper newNode = new NodeZipper(currentContext.thisNode(), currentContext.getBreadcrumbs(), nodeAdapter);
myZippers.add(newNode);
this.result = (T) currentContext.thisNode();
} else {
this.result = (T) currentContext.thisNode();
}
}
@Override
public T getRawResult() {
return result;
}
private NodeZipper moveUp(T parent, List> sameParent) {
assertNotEmpty(sameParent, () -> "expected at least one zipper");
Map> childrenMap = new HashMap<>(nodeAdapter.getNamedChildren(parent));
Map indexCorrection = new HashMap<>();
sameParent.sort((zipper1, zipper2) -> {
int index1 = zipper1.getBreadcrumbs().get(0).getLocation().getIndex();
int index2 = zipper2.getBreadcrumbs().get(0).getLocation().getIndex();
if (index1 != index2) {
return Integer.compare(index1, index2);
}
NodeZipper.ModificationType modificationType1 = zipper1.getModificationType();
NodeZipper.ModificationType modificationType2 = zipper2.getModificationType();
// same index can never be deleted and changed at the same time
if (modificationType1 == modificationType2) {
return 0;
}
// always first replacing the node
if (modificationType1 == REPLACE) {
return -1;
}
// and then INSERT_BEFORE before INSERT_AFTER
return modificationType1 == NodeZipper.ModificationType.INSERT_BEFORE ? -1 : 1;
});
for (NodeZipper zipper : sameParent) {
NodeLocation location = zipper.getBreadcrumbs().get(0).getLocation();
Integer ixDiff = indexCorrection.getOrDefault(location.getName(), 0);
int ix = location.getIndex() + ixDiff;
String name = location.getName();
List childList = new ArrayList<>(childrenMap.get(name));
switch (zipper.getModificationType()) {
case REPLACE:
childList.set(ix, zipper.getCurNode());
break;
case DELETE:
childList.remove(ix);
indexCorrection.put(name, ixDiff - 1);
break;
case INSERT_BEFORE:
childList.add(ix, zipper.getCurNode());
indexCorrection.put(name, ixDiff + 1);
break;
case INSERT_AFTER:
childList.add(ix + 1, zipper.getCurNode());
indexCorrection.put(name, ixDiff + 1);
break;
}
childrenMap.put(name, childList);
}
T newNode = nodeAdapter.withNewChildren(parent, childrenMap);
List> newBreadcrumbs = sameParent.get(0).getBreadcrumbs().subList(1, sameParent.get(0).getBreadcrumbs().size());
return new NodeZipper<>(newNode, newBreadcrumbs, nodeAdapter);
}
}
private List pushAll(TraverserContext traverserContext) {
Map>> childrenContextMap = new LinkedHashMap<>();
LinkedList contexts = new LinkedList<>();
if (!traverserContext.isDeleted()) {
Map> childrenMap = this.nodeAdapter.getNamedChildren(traverserContext.thisNode());
childrenMap.keySet().forEach(key -> {
List children = childrenMap.get(key);
for (int i = children.size() - 1; i >= 0; i--) {
T child = assertNotNull(children.get(i), () -> String.format("null child for key %s", key));
NodeLocation nodeLocation = new NodeLocation(key, i);
DefaultTraverserContext context = newContext(child, traverserContext, nodeLocation);
contexts.push(context);
childrenContextMap.computeIfAbsent(key, notUsed -> new ArrayList<>());
childrenContextMap.get(key).add(0, context);
}
});
}
return contexts;
}
private DefaultTraverserContext newContext(T o, TraverserContext parent, NodeLocation position) {
return newContextImpl(o, parent, new LinkedHashMap<>(), position, false);
}
private DefaultTraverserContext newContextImpl(T curNode,
TraverserContext parent,
Map, Object> vars,
NodeLocation nodeLocation,
boolean isRootContext) {
assertNotNull(vars);
return new DefaultTraverserContext<>(curNode, parent, null, vars, sharedContextData, nodeLocation, isRootContext, true);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy