3 # QAPI parser test harness
5 # Copyright (c) 2013 Red Hat Inc.
8 # Markus Armbruster <armbru@redhat.com>
10 # This work is licensed under the terms of the GNU GPL, version 2 or later.
11 # See the COPYING file in the top-level directory.
19 from io
import StringIO
21 from qapi
.error
import QAPIError
22 from qapi
.schema
import QAPISchema
, QAPISchemaVisitor
25 class QAPISchemaTestVisitor(QAPISchemaVisitor
):
27 def visit_module(self
, name
):
28 print('module %s' % name
)
30 def visit_include(self
, name
, info
):
31 print('include %s' % name
)
33 def visit_enum_type(self
, name
, info
, ifcond
, features
, members
, prefix
):
34 print('enum %s' % name
)
36 print(' prefix %s' % prefix
)
38 print(' member %s' % m
.name
)
39 self
._print
_if
(m
.ifcond
, indent
=8)
40 self
._print
_features
(m
.features
, indent
=8)
41 self
._print
_if
(ifcond
)
42 self
._print
_features
(features
)
44 def visit_array_type(self
, name
, info
, ifcond
, element_type
):
46 return # suppress built-in arrays
47 print('array %s %s' % (name
, element_type
.name
))
48 self
._print
_if
(ifcond
)
50 def visit_object_type(self
, name
, info
, ifcond
, features
,
51 base
, members
, variants
):
52 print('object %s' % name
)
54 print(' base %s' % base
.name
)
56 print(' member %s: %s optional=%s'
57 % (m
.name
, m
.type.name
, m
.optional
))
58 self
._print
_if
(m
.ifcond
, 8)
59 self
._print
_features
(m
.features
, indent
=8)
60 self
._print
_variants
(variants
)
61 self
._print
_if
(ifcond
)
62 self
._print
_features
(features
)
64 def visit_alternate_type(self
, name
, info
, ifcond
, features
, variants
):
65 print('alternate %s' % name
)
66 self
._print
_variants
(variants
)
67 self
._print
_if
(ifcond
)
68 self
._print
_features
(features
)
70 def visit_command(self
, name
, info
, ifcond
, features
,
71 arg_type
, ret_type
, gen
, success_response
, boxed
,
72 allow_oob
, allow_preconfig
, coroutine
):
73 print('command %s %s -> %s'
74 % (name
, arg_type
and arg_type
.name
,
75 ret_type
and ret_type
.name
))
76 print(' gen=%s success_response=%s boxed=%s oob=%s preconfig=%s%s'
77 % (gen
, success_response
, boxed
, allow_oob
, allow_preconfig
,
78 " coroutine=True" if coroutine
else ""))
79 self
._print
_if
(ifcond
)
80 self
._print
_features
(features
)
82 def visit_event(self
, name
, info
, ifcond
, features
, arg_type
, boxed
):
83 print('event %s %s' % (name
, arg_type
and arg_type
.name
))
84 print(' boxed=%s' % boxed
)
85 self
._print
_if
(ifcond
)
86 self
._print
_features
(features
)
89 def _print_variants(variants
):
91 print(' tag %s' % variants
.tag_member
.name
)
92 for v
in variants
.variants
:
93 print(' case %s: %s' % (v
.name
, v
.type.name
))
94 QAPISchemaTestVisitor
._print
_if
(v
.ifcond
, indent
=8)
97 def _print_if(ifcond
, indent
=4):
98 # TODO Drop this hack after replacing OrderedDict by plain
99 # dict (requires Python 3.7)
100 def _massage(subcond
):
101 if isinstance(subcond
, str):
103 if isinstance(subcond
, list):
104 return [_massage(val
) for val
in subcond
]
105 return {key
: _massage(val
) for key
, val
in subcond
.items()}
107 if ifcond
.is_present():
108 print('%sif %s' % (' ' * indent
, _massage(ifcond
.ifcond
)))
111 def _print_features(cls
, features
, indent
=4):
114 print('%sfeature %s' % (' ' * indent
, f
.name
))
115 cls
._print
_if
(f
.ifcond
, indent
+ 4)
118 def test_frontend(fname
):
119 schema
= QAPISchema(fname
)
120 schema
.visit(QAPISchemaTestVisitor())
122 for doc
in schema
.docs
:
124 print('doc symbol=%s' % doc
.symbol
)
126 print('doc freeform')
127 print(' body=\n%s' % doc
.body
.text
)
128 for arg
, section
in doc
.args
.items():
129 print(' arg=%s\n%s' % (arg
, section
.text
))
130 for feat
, section
in doc
.features
.items():
131 print(' feature=%s\n%s' % (feat
, section
.text
))
132 for section
in doc
.sections
:
133 print(' section=%s\n%s' % (section
.name
, section
.text
))
136 def open_test_result(dir_name
, file_name
, update
):
137 mode
= 'r+' if update
else 'r'
139 fp
= open(os
.path
.join(dir_name
, file_name
), mode
)
140 except FileNotFoundError
:
143 fp
= open(os
.path
.join(dir_name
, file_name
), 'w+')
147 def test_and_diff(test_name
, dir_name
, update
):
148 sys
.stdout
= StringIO()
150 test_frontend(os
.path
.join(dir_name
, test_name
+ '.json'))
151 except QAPIError
as err
:
152 errstr
= str(err
) + '\n'
154 errstr
= errstr
.replace(dir_name
+ '/', '')
155 actual_err
= errstr
.splitlines(True)
159 actual_out
= sys
.stdout
.getvalue().splitlines(True)
161 sys
.stdout
= sys
.__stdout
__
164 outfp
= open_test_result(dir_name
, test_name
+ '.out', update
)
165 errfp
= open_test_result(dir_name
, test_name
+ '.err', update
)
166 expected_out
= outfp
.readlines()
167 expected_err
= errfp
.readlines()
168 except OSError as err
:
169 print("%s: can't open '%s': %s"
170 % (sys
.argv
[0], err
.filename
, err
.strerror
),
174 if actual_out
== expected_out
and actual_err
== expected_err
:
177 print("%s %s" % (test_name
, 'UPDATE' if update
else 'FAIL'),
179 out_diff
= difflib
.unified_diff(expected_out
, actual_out
, outfp
.name
)
180 err_diff
= difflib
.unified_diff(expected_err
, actual_err
, errfp
.name
)
181 sys
.stdout
.writelines(out_diff
)
182 sys
.stdout
.writelines(err_diff
)
190 outfp
.writelines(actual_out
)
193 errfp
.writelines(actual_err
)
194 except OSError as err
:
195 print("%s: can't write '%s': %s"
196 % (sys
.argv
[0], err
.filename
, err
.strerror
),
204 parser
= argparse
.ArgumentParser(
205 description
='QAPI schema tester')
206 parser
.add_argument('-d', '--dir', action
='store', default
='',
207 help="directory containing tests")
208 parser
.add_argument('-u', '--update', action
='store_true',
209 help="update expected test results")
210 parser
.add_argument('tests', nargs
='*', metavar
='TEST', action
='store')
211 args
= parser
.parse_args()
215 (dir_name
, base_name
) = os
.path
.split(t
)
216 dir_name
= dir_name
or args
.dir
217 test_name
= os
.path
.splitext(base_name
)[0]
218 status |
= test_and_diff(test_name
, dir_name
, args
.update
)
223 if __name__
== '__main__':