8ea84738ccd4faa83fd581f36e89a37740da0977
[sliver-openvswitch.git] / ovsdb / ovsdb-dot.in
1 #! @PYTHON@
2
3 from datetime import date
4 import ovs.db.error
5 import ovs.db.schema
6 import getopt
7 import os
8 import re
9 import sys
10
11 argv0 = sys.argv[0]
12
13 def printEdge(tableName, type, baseType, label):
14     if baseType.ref_table_name:
15         if type.n_min == 0:
16             if type.n_max == 1:
17                 arity = "?"
18             elif type.n_max == sys.maxint:
19                 arity = "*"
20             else:
21                 arity = "{,%d}" % type.n_max
22         elif type.n_min == 1:
23             if type.n_max == 1:
24                 arity = ""
25             elif type.n_max == sys.maxint:
26                 arity = "+"
27             else:
28                 arity = "{1,%d}" % type.n_max
29
30         options = {}
31         options['label'] = '"%s%s"' % (label, arity)
32         if baseType.ref_type == 'weak':
33             options['constraint'] = 'false'
34             options['style'] = 'dotted'
35         print "\t%s -> %s [%s];" % (
36             tableName,
37             baseType.ref_table_name,
38             ', '.join(['%s=%s' % (k,v) for k,v in options.items()]))
39
40 def schemaToDot(schemaFile):
41     schema = ovs.db.schema.DbSchema.from_json(ovs.json.from_file(schemaFile))
42
43     print "digraph %s {" % schema.name
44     print '\trankdir=LR;'
45     print '\tsize="6.5,4";'
46     print '\tmargin="0";'
47     print "\tnode [shape=box];"
48     print "\tedge [dir=none, arrowhead=none, arrowtail=none];"
49     for tableName, table in schema.tables.iteritems():
50         options = {}
51         if table.is_root:
52             options['style'] = 'bold'
53         print "\t%s [%s];" % (
54             tableName,
55             ', '.join(['%s=%s' % (k,v) for k,v in options.items()]))
56         for columnName, column in table.columns.iteritems():
57             if column.type.value:
58                 printEdge(tableName, column.type, column.type.key, "%s key" % columnName)
59                 printEdge(tableName, column.type, column.type.value, "%s value" % columnName)
60             else:
61                 printEdge(tableName, column.type, column.type.key, columnName)
62     print "}";
63
64 def usage():
65     print """\
66 %(argv0)s: compiles ovsdb schemas to graphviz format
67 Prints a .dot file that "dot" can render to an entity-relationship diagram
68 usage: %(argv0)s [OPTIONS] SCHEMA
69 where SCHEMA is an OVSDB schema in JSON format
70
71 The following options are also available:
72   -h, --help                  display this help message
73   -V, --version               display version information\
74 """ % {'argv0': argv0}
75     sys.exit(0)
76
77 if __name__ == "__main__":
78     try:
79         try:
80             options, args = getopt.gnu_getopt(sys.argv[1:], 'hV',
81                                               ['help', 'version'])
82         except getopt.GetoptError, geo:
83             sys.stderr.write("%s: %s\n" % (argv0, geo.msg))
84             sys.exit(1)
85
86         for key, value in options:
87             if key in ['-h', '--help']:
88                 usage()
89             elif key in ['-V', '--version']:
90                 print "ovsdb-dot (Open vSwitch) @VERSION@"
91             else:
92                 sys.exit(0)
93
94         if len(args) != 1:
95             sys.stderr.write("%s: exactly 1 non-option argument required "
96                              "(use --help for help)\n" % argv0)
97             sys.exit(1)
98
99         schemaToDot(args[0])
100
101     except ovs.db.error.Error, e:
102         sys.stderr.write("%s: %s\n" % (argv0, e.msg))
103         sys.exit(1)
104
105 # Local variables:
106 # mode: python
107 # End: