All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.mapfish.print.processor.ProcessorDependencyGraph Maven / Gradle / Ivy

package org.mapfish.print.processor;

import com.google.common.collect.BiMap;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.Multimap;
import org.mapfish.print.output.Values;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.ArrayList;
import java.util.Collection;
import java.util.IdentityHashMap;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.RecursiveTask;
import javax.annotation.Nonnull;

import static org.mapfish.print.parser.ParserUtils.FILTER_ONLY_REQUIRED_ATTRIBUTES;
import static org.mapfish.print.parser.ParserUtils.getAttributeNames;

/**
 * Represents a graph of the processors dependencies.  The root nodes can execute in parallel but processors
 * with dependencies must wait for their dependencies to complete before execution.
 * 

*/ public final class ProcessorDependencyGraph { private static final Logger LOGGER = LoggerFactory.getLogger(ProcessorDependencyGraph.class); private final List> roots; ProcessorDependencyGraph() { this.roots = new ArrayList<>(); } static void tryExecuteNodes( final Collection> dependencyNodes, final ProcessorExecutionContext execContext, final boolean notStartedIsOk) { final List> tasks = new ArrayList<>(dependencyNodes.size()); // fork all but 1 dependencies (the first will be ran in current thread) for (final ProcessorGraphNode depNode: dependencyNodes) { final Optional> task = depNode.createTask(execContext); if (task.isPresent()) { tasks.add(task.get()); if (tasks.size() > 1) { task.get().fork(); } } else if (!notStartedIsOk) { LOGGER.error("Failed to start the processor {}", depNode.getProcessor()); } } if (!tasks.isEmpty()) { // compute one task in current thread so as not to waste threads tasks.get(0).compute(); for (ProcessorGraphNode.ProcessorNodeForkJoinTask task: tasks.subList(1, tasks.size())) { task.join(); } } } /** * Create a ForkJoinTask for running in a fork join pool. * * @param values the values to use for getting the required inputs of the processor and putting * the output in. * @return a task ready to be submitted to a fork join pool. */ public ProcessorGraphForkJoinTask createTask(@Nonnull final Values values) { StringBuilder missingAttributes = new StringBuilder(); final Multimap requiredAttributes = getAllRequiredAttributes(); for (String attribute: requiredAttributes.keySet()) { if (!values.containsKey(attribute)) { missingAttributes.append("\n\t* ").append(attribute).append(" <- ") .append(requiredAttributes.get(attribute)); } } if (missingAttributes.length() > 0) { throw new IllegalArgumentException("It has been found that one or more required attributes are " + "missing from the values object:" + missingAttributes + "\n"); } return new ProcessorGraphForkJoinTask(values); } /** * Add a new root node. * * @param node new root node. */ void addRoot(final ProcessorGraphNode node) { this.roots.add(node); } /** * Get all the names of inputs that are required to be in the Values object when this graph is executed. */ @SuppressWarnings("unchecked") public Multimap getAllRequiredAttributes() { Multimap requiredInputs = HashMultimap.create(); for (ProcessorGraphNode root: this.roots) { final BiMap inputMapper = root.getInputMapper(); for (String attr: inputMapper.keySet()) { requiredInputs.put(attr, root.getProcessor()); } final Object inputParameter = root.getProcessor().createInputParameter(); if (inputParameter instanceof Values) { continue; } else if (inputParameter != null) { final Class inputParameterClass = inputParameter.getClass(); final Set requiredAttributesDefinedInInputParameter = getAttributeNames(inputParameterClass, FILTER_ONLY_REQUIRED_ATTRIBUTES); for (String attName: requiredAttributesDefinedInInputParameter) { try { if (inputParameterClass.getField(attName).getType() == Values.class) { continue; } } catch (NoSuchFieldException e) { throw new RuntimeException(e); } String mappedName = ProcessorUtils.getInputValueName( root.getProcessor().getInputPrefix(), inputMapper, attName); requiredInputs.put(mappedName, root.getProcessor()); } } } return requiredInputs; } public List> getRoots() { return this.roots; } @Override public String toString() { StringBuilder builder = new StringBuilder(); builder.append("strict digraph g {\n" + " rankdir=\"LR\";\n" + " node [shape=\"box\"];\n"); toString(builder, 0, "root"); builder.append("}"); return builder.toString(); } /** * Create a string representing the nodes. * * @param builder the builder to add the string to. * @param indent the number of steps of indent for this node * @param parent the parent node */ public void toString(final StringBuilder builder, final int indent, final String parent) { for (ProcessorGraphNode root: this.roots) { root.toString(builder, indent, parent); } } /** * Create a set containing all the processors in the graph. */ public Set> getAllProcessors() { IdentityHashMap, Void> all = new IdentityHashMap<>(); for (ProcessorGraphNode root: this.roots) { for (Processor p: root.getAllProcessors()) { all.put(p, null); } } return all.keySet(); } /** * A ForkJoinTask that will create ForkJoinTasks from each root and run each of them. */ public final class ProcessorGraphForkJoinTask extends RecursiveTask { private final ProcessorExecutionContext execContext; private ProcessorGraphForkJoinTask(@Nonnull final Values values) { this.execContext = new ProcessorExecutionContext(values); } /** * Cancels the complete processor graph of a print task. *

* This is achieved by setting the cancel flag of the execution context, so that every processor can * stop its execution. * * @param mayInterruptIfRunning is ignored */ @Override public boolean cancel(final boolean mayInterruptIfRunning) { this.execContext.cancel(); return super.cancel(mayInterruptIfRunning); } @Override protected Values compute() { return this.execContext.getContext().mdcContext(() -> { final ProcessorDependencyGraph graph = ProcessorDependencyGraph.this; LOGGER.debug("Starting to execute processor graph: \n{}", graph); try { tryExecuteNodes(graph.roots, this.execContext, false); } catch (Throwable ex) { this.execContext.cancel(); // cancel all pending computations in case of error throw ex; } finally { LOGGER.debug("Finished executing processor graph: \n{}", graph); } return this.execContext.getValues(); }); } public Processor.ExecutionContext getExecutionContext() { return this.execContext.getContext(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy