Skip to content

Commit

Permalink
Introduces PassManager drawer (#2445)
Browse files Browse the repository at this point in the history
* First go at pass manager drawer - draws the passes into boxes based on their flow controllers

* Added in comments and coloring the nodes

* Added in drawing the arguments

* Added in testing

* Refactoring

* Fixed cyclic import

* Made filename non optional

* CHANGELOG

* Added in checks to see if GraphViz is installed - will skip tests if not

* Renames pss to pass_ and moved an import basedon review

* Moved import to fix lint
  • Loading branch information
maddy-tod authored and nonhermitian committed May 23, 2019
1 parent 5a81691 commit 2d14c1e
Show file tree
Hide file tree
Showing 7 changed files with 234 additions and 4 deletions.
11 changes: 7 additions & 4 deletions CHANGELOG.rst
Original file line number Diff line number Diff line change
Expand Up @@ -19,9 +19,16 @@ The format is based on `Keep a Changelog`_.
`UNRELEASED`_
=============

Added
-----

- Introduced a visualization for the Pass Manager. (#2445)

Changed
-------

- Qubits and classical bits are not represented as a tuples anymore, but as
instances of ``Qubit`` and ``Clbit`` respectively.
- The ``pylatexenc`` and ``pillow`` requirements are now optional. These are
only used by the ``latex`` and ``latex_source`` circuit visualization
backends. To continue using them ensure these are installed.
Expand All @@ -43,10 +50,6 @@ Removed
- The ``qiskit.qiskiterror`` module has been removed. Please use
``qiskit.exceptions`` instead. (#2399)

Changed
-------
- Qubits and classical bits are not represented as a tuples anymore, but as
instances of ``Qubit`` and ``Clbit`` respectively.

`0.8.0`_ - 2019-05-02
=====================
Expand Down
5 changes: 5 additions & 0 deletions qiskit/transpiler/passmanager.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
from collections import OrderedDict
from qiskit.dagcircuit import DAGCircuit
from qiskit.converters import circuit_to_dag, dag_to_circuit
from qiskit.visualization import pass_manager_drawer
from .propertyset import PropertySet
from .basepasses import BasePass
from .fencedobjs import FencedPropertySet, FencedDAGCircuit
Expand Down Expand Up @@ -149,6 +150,10 @@ def run(self, circuit):
circuit.name = name
return circuit

def draw(self, filename, style=None):
""" Draw the pass manager"""
pass_manager_drawer(self, filename=filename, style=style)

def _do_pass(self, pass_, dag, options):
"""Do a pass and its "requires".
Expand Down
1 change: 1 addition & 0 deletions qiskit/visualization/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@
from .pulse_visualization import pulse_drawer
from .circuit_visualization import circuit_drawer, qx_color_scheme
from .dag_visualization import dag_drawer
from .pass_manager_visualization import pass_manager_drawer
from .gate_map import plot_gate_map

from .exceptions import VisualizationError
Expand Down
147 changes: 147 additions & 0 deletions qiskit/visualization/pass_manager_visualization.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,147 @@
# -*- coding: utf-8 -*-

# This code is part of Qiskit.
#
# (C) Copyright IBM 2019.
#
# This code is licensed under the Apache License, Version 2.0. You may
# obtain a copy of this license in the LICENSE.txt file in the root directory
# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0.
#
# Any modifications or derivative works of this code must retain this
# copyright notice, and modified files need to carry a notice indicating
# that they have been altered from the originals.
"""
Visualization function for a pass manager. Passes are grouped based on their
flow controller, and coloured based on the type of pass.
"""
import inspect
from qiskit.transpiler.basepasses import AnalysisPass, TransformationPass
DEFAULT_STYLE = {AnalysisPass: 'red',
TransformationPass: 'blue'}


try:
import subprocess
print(subprocess.check_output(['dot', '-V']))
HAS_GRAPHVIZ = True
except FileNotFoundError:
# this is raised when the dot command cannot be found, which means GraphViz
# isn't installed
HAS_GRAPHVIZ = False


def pass_manager_drawer(pass_manager, filename, style=None):
"""
Draws the pass manager.
This function needs `pydot <https://github.com/erocarrera/pydot>`, which in turn needs
Graphviz <https://www.graphviz.org/>` to be installed.
Args:
pass_manager (PassManager): the pass manager to be drawn
filename (str): file path to save image to
style (dict or OrderedDict): keys are the pass classes and the values are
the colors to make them. An example can be seen in the DEFAULT_STYLE. An ordered
dict can be used to ensure a priority coloring when pass falls into multiple
categories. Any values not included in the provided dict will be filled in from
the default dict
Raises:
ImportError: when nxpd or pydot not installed.
"""

try:
import pydot
if not HAS_GRAPHVIZ:
raise ImportError
except ImportError:
raise ImportError("pass_manager_drawer requires pydot and graphviz. "
"Run 'pip install pydot'. "
"Graphviz can be installed using 'brew install graphviz' on Mac"
" or by downloading it from the website.")

passes = pass_manager.passes()

if not style:
style = DEFAULT_STYLE

# create the overall graph
graph = pydot.Dot()

# identifiers for nodes need to be unique, so assign an id
# can't just use python's id in case the exact same pass was
# appended more than once
node_id = 0

prev_node = None

for controller_group in passes:

# label is the name of the flow controller (without the word controller)
label = controller_group['type'].__name__.replace('Controller', '')

# create the subgraph for this controller
subgraph = pydot.Cluster(str(id(controller_group)), label=label)

for pass_ in controller_group['passes']:

# label is the name of the pass
node = pydot.Node(str(node_id),
label=str(type(pass_).__name__),
color=_get_node_color(pass_, style),
shape="rectangle")

subgraph.add_node(node)
node_id += 1

# the arguments that were provided to the pass when it was created
arg_spec = inspect.getfullargspec(pass_.__init__)
# 0 is the args, 1: to remove the self arg
args = arg_spec[0][1:]

num_optional = len(arg_spec[3]) if arg_spec[3] else 0

# add in the inputs to the pass
for arg_index, arg in enumerate(args):
nd_style = 'solid'
# any optional args are dashed
# the num of optional counts from the end towards the start of the list
if arg_index >= (len(args) - num_optional):
nd_style = 'dashed'

input_node = pydot.Node(node_id, label=arg,
color="black",
shape="ellipse",
fontsize=10,
style=nd_style)
subgraph.add_node(input_node)
node_id += 1
subgraph.add_edge(pydot.Edge(input_node, node))

# if there is a previous node, add an edge between them
if prev_node:
subgraph.add_edge(pydot.Edge(prev_node, node))

prev_node = node

graph.add_subgraph(subgraph)

if filename:
# linter says this isn't a method - it is
graph.write_png(filename) # pylint: disable=no-member


def _get_node_color(pss, style):

# look in the user provided dict first
for typ, color in style.items():
if isinstance(pss, typ):
return color

# failing that, look in the default
for typ, color in DEFAULT_STYLE.items():
if isinstance(pss, typ):
return color

return "black"
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
74 changes: 74 additions & 0 deletions test/python/visualization/test_pass_manager_drawer.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
# -*- coding: utf-8 -*-

# This code is part of Qiskit.
#
# (C) Copyright IBM 2019.
#
# This code is licensed under the Apache License, Version 2.0. You may
# obtain a copy of this license in the LICENSE.txt file in the root directory
# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0.
#
# Any modifications or derivative works of this code must retain this
# copyright notice, and modified files need to carry a notice indicating
# that they have been altered from the originals.

"""Tests for pass manager visualization tool."""

import unittest
import os

from qiskit.transpiler.transpile_config import TranspileConfig
from qiskit.transpiler import CouplingMap, Layout
from qiskit import QuantumRegister
from qiskit.transpiler.preset_passmanagers import level_0_pass_manager, level_1_pass_manager
from qiskit.transpiler.passes import SetLayout, CheckMap, EnlargeWithAncilla, RemoveResetInZeroState

from qiskit.visualization.pass_manager_visualization import HAS_GRAPHVIZ
from .visualization import QiskitVisualizationTestCase, path_to_diagram_reference


class TestPassManagerDrawer(QiskitVisualizationTestCase):
"""Qiskit pass manager drawer tests."""

def setUp(self):
coupling = [[0, 1], [1, 2], [2, 3], [3, 4], [4, 5], [5, 6]]
coupling_map = CouplingMap(couplinglist=coupling)
qr = QuantumRegister(7, 'q')
layout = Layout({qr[i]: i for i in range(coupling_map.size())})
self.config = TranspileConfig(optimization_level=1,
basis_gates=['u1', 'u3', 'u2', 'cx'],
initial_layout=layout,
coupling_map=coupling_map,
seed_transpiler=987,
backend_properties=None)

@unittest.skipIf(not HAS_GRAPHVIZ,
'Graphviz not installed.')
def test_pass_manager_drawer_basic(self):
"""Test to see if the drawer draws a normal pass manager correctly"""
filename = self._get_resource_path('current_l0.png')
level_0_pass_manager(self.config).draw(filename=filename)

self.assertImagesAreEqual(filename, path_to_diagram_reference('pass_manager_l0_ref.png'))
os.remove(filename)

@unittest.skipIf(not HAS_GRAPHVIZ,
'Graphviz not installed.')
def test_pass_manager_drawer_style(self):
"""Test to see if the colours are updated when provided by the user"""
# set colours for some passes, but leave others to take the default values
style = {SetLayout: 'cyan',
CheckMap: 'green',
EnlargeWithAncilla: 'pink',
RemoveResetInZeroState: 'grey'}

filename = self._get_resource_path('current_l1.png')
level_1_pass_manager(self.config).draw(filename=filename, style=style)

self.assertImagesAreEqual(filename,
path_to_diagram_reference('pass_manager_l1_style_ref.png'))
os.remove(filename)


if __name__ == '__main__':
unittest.main(verbosity=2)

0 comments on commit 2d14c1e

Please sign in to comment.