mirror of
https://github.com/status-im/spiff-arena.git
synced 2025-01-15 12:44:52 +00:00
7c219fd731
git-subtree-dir: SpiffWorkflow git-subtree-split: 63db3e45947ec66b8d0efc2c74064004f8ff482c
82 lines
3.6 KiB
Python
82 lines
3.6 KiB
Python
# -*- coding: utf-8 -*-
|
|
|
|
# This library is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU Lesser General Public
|
|
# License as published by the Free Software Foundation; either
|
|
# version 2.1 of the License, or (at your option) any later version.
|
|
#
|
|
# This library is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
# Lesser General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Lesser General Public
|
|
# License along with this library; if not, write to the Free Software
|
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
|
# 02110-1301 USA
|
|
|
|
# requires: https://github.com/stricaud/gvgen
|
|
import gvgen
|
|
from .base import Serializer
|
|
|
|
|
|
class dotVisualizer(Serializer):
|
|
|
|
def serialize_workflow_spec(self, wf_spec):
|
|
nodes = set()
|
|
linked = set()
|
|
graph = gvgen.GvGen()
|
|
parent = graph.newItem("Workflow")
|
|
|
|
# these built in shapes are available:
|
|
# http://www.graphviz.org/doc/info/shapes.html
|
|
graph.styleAppend("Cancel", "shape", "oval")
|
|
graph.styleAppend("CancelTask", "shape", "oval")
|
|
graph.styleAppend("Choose", "shape", "diamond")
|
|
graph.styleAppend("ExclusiveChoice", "shape", "diamond")
|
|
graph.styleAppend("Execute", "shape", "rect")
|
|
graph.styleAppend("Gate", "shape", "trapezium")
|
|
graph.styleAppend("Join", "shape", "invtriangle")
|
|
graph.styleAppend("Merge", "shape", "invtriangle")
|
|
graph.styleAppend("MultiChoice", "shape", "diamond")
|
|
graph.styleAppend("MultiInstance", "shape", "box")
|
|
graph.styleAppend("ReleaseMutex", "shape", "diamond")
|
|
graph.styleAppend("Simple", "shape", "rect")
|
|
graph.styleAppend("StartTask", "shape", "oval")
|
|
graph.styleAppend("SubWorkflow", "shape", "invhouse")
|
|
graph.styleAppend("ThreadMerge", "shape", "invtriangle")
|
|
graph.styleAppend("ThreadSplit", "shape", "triangle")
|
|
graph.styleAppend("ThreadStart", "shape", "oval")
|
|
graph.styleAppend("Transform", "shape", "rect")
|
|
graph.styleAppend("Trigger", "shape", "oval")
|
|
|
|
# build graph with all the nodes first
|
|
def recurisvelyAddNodes(task_spec):
|
|
if task_spec in nodes:
|
|
return
|
|
task_spec.gv = graph.newItem(task_spec.name, parent)
|
|
# add a default style for this class so that if we don't have one
|
|
# when we apply it doesn't break the GvGen library
|
|
graph.styleAppend(task_spec.__class__.__name__, "ignore", "this")
|
|
graph.styleApply(task_spec.__class__.__name__, task_spec.gv)
|
|
nodes.add(task_spec)
|
|
sub_specs = ([task_spec.spec.start] if hasattr(
|
|
task_spec, 'spec') else []) + task_spec.outputs
|
|
for t in sub_specs:
|
|
recurisvelyAddNodes(t)
|
|
|
|
# then link all the nodes together
|
|
def recursive_linking(task_spec):
|
|
if task_spec in linked:
|
|
return
|
|
linked.add(task_spec)
|
|
sub_specs = ([task_spec.spec.start] if hasattr(
|
|
task_spec, 'spec') else []) + task_spec.outputs
|
|
for i, t in enumerate(sub_specs):
|
|
graph.newLink(task_spec.gv, t.gv)
|
|
recursive_linking(t)
|
|
|
|
recurisvelyAddNodes(wf_spec.start)
|
|
recursive_linking(wf_spec.start)
|
|
return (graph.dot() if graph.dot() else '')
|