3 # Render Qemu Block Graph
5 # Copyright (c) 2018 Virtuozzo International GmbH. All rights reserved.
7 # This program is free software; you can redistribute it and/or modify
8 # it under the terms of the GNU General Public License as published by
9 # the Free Software Foundation; either version 2 of the License, or
10 # (at your option) any later version.
12 # This program is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU General Public License for more details.
17 # You should have received a copy of the GNU General Public License
18 # along with this program. If not, see <http://www.gnu.org/licenses/>.
25 from graphviz
import Digraph
27 sys
.path
.append(os
.path
.join(os
.path
.dirname(__file__
), '..', 'python'))
28 from qemu
.qmp
import (
35 s
= 'w' if 'write' in arr
else '_'
36 s
+= 'r' if 'consistent-read' in arr
else '_'
37 s
+= 'u' if 'write-unchanged' in arr
else '_'
38 s
+= 'g' if 'graph-mod' in arr
else '_'
39 s
+= 's' if 'resize' in arr
else '_'
43 def render_block_graph(qmp
, filename
, format
='png'):
45 Render graph in text (dot) representation into "@filename" and
46 representation in @format into "@filename.@format"
49 bds_nodes
= qmp
.command('query-named-block-nodes')
50 bds_nodes
= {n
['node-name']: n
for n
in bds_nodes
}
52 job_nodes
= qmp
.command('query-block-jobs')
53 job_nodes
= {n
['device']: n
for n
in job_nodes
}
55 block_graph
= qmp
.command('x-debug-query-block-graph')
57 graph
= Digraph(comment
='Block Nodes Graph')
59 graph
.node('permission symbols:\l'
61 ' r - consistent-Read\l'
62 ' u - write - Unchanged\l'
65 'edge label scheme:\l'
68 ' <shared_perm>\l', shape
='none')
70 for n
in block_graph
['nodes']:
71 if n
['type'] == 'block-driver':
72 info
= bds_nodes
[n
['name']]
73 label
= n
['name'] + ' [' + info
['drv'] + ']'
74 if info
['drv'] == 'file':
75 label
+= '\n' + os
.path
.basename(info
['file'])
77 elif n
['type'] == 'block-job':
78 info
= job_nodes
[n
['name']]
79 label
= info
['type'] + ' job (' + n
['name'] + ')'
82 assert n
['type'] == 'block-backend'
83 label
= n
['name'] if n
['name'] else 'unnamed blk'
86 graph
.node(str(n
['id']), label
, shape
=shape
)
88 for e
in block_graph
['edges']:
89 label
= '%s\l%s\l%s\l' % (e
['name'], perm(e
['perm']),
90 perm(e
['shared-perm']))
91 graph
.edge(str(e
['parent']), str(e
['child']), label
=label
)
93 graph
.render(filename
)
97 def __init__(self
, name
):
100 def command(self
, cmd
):
101 # only supports qmp commands without parameters
103 ar
= ['virsh', 'qemu-monitor-command', self
.name
, json
.dumps(m
)]
105 reply
= json
.loads(subprocess
.check_output(ar
))
108 raise QMPResponseError(reply
)
110 return reply
['return']
113 if __name__
== '__main__':
117 if os
.path
.exists(obj
):
119 qmp
= QEMUMonitorProtocol(obj
)
122 # assume libvirt guest name
123 qmp
= LibvirtGuest(obj
)
125 render_block_graph(qmp
, out
)