From 886d05ee644b39f99b0162a07ee02b1597d9e54b Mon Sep 17 00:00:00 2001 From: Anton Arefiev Date: Wed, 22 Feb 2017 17:26:12 +0200 Subject: [PATCH] Add fsm dot diagram generator Add a tool which generates dot diagram from inspector state machine using atomaton pydot convertor. Also add generated svg diagram to docs. Change-Id: I021812288f1833a6ebad2f90cbe862d40bd8d503 --- doc/source/images/states.svg | 174 +++++++++++++++++++++++++++++++++++ doc/source/workflow.rst | 13 ++- tools/states_to_dot.py | 94 +++++++++++++++++++ tox.ini | 5 + 4 files changed, 285 insertions(+), 1 deletion(-) create mode 100644 doc/source/images/states.svg create mode 100755 tools/states_to_dot.py diff --git a/doc/source/images/states.svg b/doc/source/images/states.svg new file mode 100644 index 000000000..236ca411f --- /dev/null +++ b/doc/source/images/states.svg @@ -0,0 +1,174 @@ + + + + + + +Ironic Inspector states + + +enrolling + +enrolling + + +error + +error + + +enrolling->error + + +error + + +processing + +processing + + +enrolling->processing + + +process + + +error->error + + +abort + + +error->error + + +error + + +starting + +starting + + +error->starting + + +start + + +reapplying + +reapplying + + +error->reapplying + + +reapply + + +processing->error + + +error + + +finished + +finished + + +processing->finished + + +finish + + +starting->error + + +error + + +starting->starting + + +start + + +waiting + +waiting + + +starting->waiting + + +wait + + +reapplying->error + + +error + + +reapplying->reapplying + + +reapply + + +reapplying->finished + + +finish + + +finished->starting + + +start + + +finished->reapplying + + +reapply + + +finished->finished + + +finish + + +waiting->error + + +timeout + + +waiting->error + + +abort + + +waiting->processing + + +process + + +waiting->starting + + +start + + + diff --git a/doc/source/workflow.rst b/doc/source/workflow.rst index 3061b44b2..dbf9a33c8 100644 --- a/doc/source/workflow.rst +++ b/doc/source/workflow.rst @@ -56,6 +56,17 @@ Usual hardware introspection flow is as follows: Starting DHCP server and configuring PXE boot environment is not part of this package and should be done separately. +State machine diagram +===================== + +The diagram below shows the introspection states that an **ironic-inspector** +FSM goes through during the node introspection, discovery and reprocessing. +The diagram also shows events that trigger state transitions. + +.. figure:: ./images/states.svg + :width: 660px + :align: center + :alt: ironic-inspector state machine diagram + .. _Ironic inspection documentation: http://docs.openstack.org/developer/ironic/deploy/install-guide.html#hardware-inspection .. _Ironic: https://wiki.openstack.org/wiki/Ironic - diff --git a/tools/states_to_dot.py b/tools/states_to_dot.py new file mode 100755 index 000000000..bde791c2e --- /dev/null +++ b/tools/states_to_dot.py @@ -0,0 +1,94 @@ +#!/usr/bin/env python +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +import optparse + +from automaton.converters import pydot + +from ironic_inspector import introspection_state as states + + +def print_header(text): + print("*" * len(text)) + print(text) + print("*" * len(text)) + + +def main(): + parser = optparse.OptionParser() + parser.add_option("-f", "--file", dest="filename", + help="write output to FILE", metavar="FILE") + parser.add_option("-T", "--format", dest="format", + help="output in given format (default: png)", + default='png') + parser.add_option("--no-labels", dest="labels", + help="do not include labels", + action='store_false', default=True) + (options, args) = parser.parse_args() + if options.filename is None: + options.filename = 'states.%s' % options.format + + def node_attrs(state): + """Attributes used for drawing the nodes (states). + + The user can perform actions on introspection states, we distinguish + the error states from the other states by highlighting the node. + Error stable states are labelled with red. + + This is a callback method used by pydot.convert(). + + :param state: name of state + :returns: A dictionary with graphic attributes used for displaying + the state. + # """ + attrs = {} + attrs['fontcolor'] = 'red' if 'error' in state else 'gray' + return attrs + + def edge_attrs(start_state, event, end_state): + """Attributes used for drawing the edges (transitions). + + This is a callback method used by pydot.convert(). + + :param start_state: name of the start state + :param event: the event, a string + :param end_state: name of the end state (unused) + :returns: A dictionary with graphic attributes used for displaying + the transition. + """ + if not options.labels: + return {} + + attrs = {} + attrs['fontsize'] = 10 + attrs['label'] = event + if end_state is 'error': + attrs['fontcolor'] = 'red' + return attrs + + source = states.FSM + graph_name = '"Ironic Inspector states"' + graph_attrs = {'size': 0} + dot_graph = pydot.convert( + source, graph_name, graph_attrs=graph_attrs, + node_attrs_cb=node_attrs, edge_attrs_cb=edge_attrs, + add_start_state=False) + + dot_graph.write(options.filename, format=options.format) + + print(dot_graph.to_string()) + print_header("Created %s at '%s'" % (options.format, options.filename)) + + +if __name__ == '__main__': + main() diff --git a/tox.ini b/tox.ini index 9c350f67b..72cf0e054 100644 --- a/tox.ini +++ b/tox.ini @@ -45,6 +45,11 @@ commands = envdir = {toxworkdir}/venv commands = oslo-config-generator --config-file config-generator.conf +[testenv:genstates] +deps = {[testenv]deps} + pydot2 +commands = {toxinidir}/tools/states_to_dot.py -f {toxinidir}/doc/source/images/states.svg --format svg + [flake8] max-complexity=15 # [H106] Don’t put vim configuration in source files.