14f193e34SMarkus Armbruster# *-*- Mode: Python -*-*
2f7160f32SAndrea Bolognani# vim: filetype=python
34f193e34SMarkus Armbruster
4625b251cSEric Blake# This file is a stress test of supported qapi constructs that must
5625b251cSEric Blake# parse and compile correctly.
6625b251cSEric Blake
71554a8faSMarkus Armbruster# Whitelists to permit QAPI rule violations
81554a8faSMarkus Armbruster{ 'pragma': {
95aceeac0SMarkus Armbruster    # Types whose member names may use '_'
105aceeac0SMarkus Armbruster    'member-name-exceptions': [
115aceeac0SMarkus Armbruster        'UserDefA'
125aceeac0SMarkus Armbruster    ],
131554a8faSMarkus Armbruster    # Commands allowed to return a non-dictionary:
14b86df374SMarkus Armbruster    'command-returns-exceptions': [
151554a8faSMarkus Armbruster        'guest-get-time',
161554a8faSMarkus Armbruster        'guest-sync' ] } }
171554a8faSMarkus Armbruster
18748053c9SEric Blake{ 'struct': 'TestStruct',
1987adbbffSMarc-André Lureau  'data': { 'integer': {'type': 'int'}, 'boolean': 'bool', 'string': 'str' } }
20748053c9SEric Blake
214f193e34SMarkus Armbruster# for testing enums
22895a2a80SEric Blake{ 'struct': 'NestedEnumsOne',
2370478cefSEric Blake  'data': { 'enum1': 'EnumOne',   # Intentional forward reference
2470478cefSEric Blake            '*enum2': 'EnumOne', 'enum3': 'EnumOne', '*enum4': 'EnumOne' } }
254f193e34SMarkus Armbruster
26625b251cSEric Blake# An empty enum, although unusual, is currently acceptable
27625b251cSEric Blake{ 'enum': 'MyEnum', 'data': [ ] }
28625b251cSEric Blake
2919767083SEric Blake# Likewise for an empty struct, including an empty base
3019767083SEric Blake{ 'struct': 'Empty1', 'data': { } }
3119767083SEric Blake{ 'struct': 'Empty2', 'base': 'Empty1', 'data': { } }
3219767083SEric Blake
330ced9531SMarkus Armbruster# Likewise for an empty flat union
340ced9531SMarkus Armbruster{ 'union': 'Union',
350ced9531SMarkus Armbruster  'base': { 'type': 'EnumOne' }, 'discriminator': 'type',
360ced9531SMarkus Armbruster  'data': { } }
370ced9531SMarkus Armbruster
3805ebf841SMarkus Armbruster{ 'command': 'user-def-cmd0', 'data': 'Empty2', 'returns': 'Empty2' }
39972a1101SEric Blake
40351d36e4SDaniel P. Berrange# for testing override of default naming heuristic
41351d36e4SDaniel P. Berrange{ 'enum': 'QEnumTwo',
42351d36e4SDaniel P. Berrange  'prefix': 'QENUM_TWO',
43351d36e4SDaniel P. Berrange  'data': [ 'value1', 'value2' ] }
44351d36e4SDaniel P. Berrange
454f193e34SMarkus Armbruster# for testing nested structs
468c3f8e77SMarkus Armbruster{ 'struct': 'UserDefOne',
478c3f8e77SMarkus Armbruster  'base': 'UserDefZero',        # intentional forward reference
4870478cefSEric Blake  'data': { 'string': 'str',
4970478cefSEric Blake            '*enum1': 'EnumOne' } }   # intentional forward reference
5070478cefSEric Blake
5170478cefSEric Blake{ 'enum': 'EnumOne',
52800877bbSAnton Nefedov  'data': [ 'value1', 'value2', 'value3', 'value4' ] }
538c3f8e77SMarkus Armbruster
54895a2a80SEric Blake{ 'struct': 'UserDefZero',
55aabbd472SMarkus Armbruster  'data': { 'integer': 'int' } }
56aabbd472SMarkus Armbruster
576446a592SEric Blake{ 'struct': 'UserDefTwoDictDict',
586446a592SEric Blake  'data': { 'userdef': 'UserDefOne', 'string': 'str' } }
596446a592SEric Blake
606446a592SEric Blake{ 'struct': 'UserDefTwoDict',
616446a592SEric Blake  'data': { 'string1': 'str',
626446a592SEric Blake            'dict2': 'UserDefTwoDictDict',
636446a592SEric Blake            '*dict3': 'UserDefTwoDictDict' } }
646446a592SEric Blake
65895a2a80SEric Blake{ 'struct': 'UserDefTwo',
664f193e34SMarkus Armbruster  'data': { 'string0': 'str',
676446a592SEric Blake            'dict1': 'UserDefTwoDict' } }
684f193e34SMarkus Armbruster
69967c8851SMarc-André Lureau{ 'struct': 'UserDefThree',
70967c8851SMarc-André Lureau  'data': { 'string0': 'str' } }
71967c8851SMarc-André Lureau
729f08c8ecSEric Blake# dummy struct to force generation of array types not otherwise mentioned
739f08c8ecSEric Blake{ 'struct': 'ForceArrays',
74748053c9SEric Blake  'data': { 'unused1':['UserDefOne'], 'unused2':['UserDefTwo'],
75748053c9SEric Blake            'unused3':['TestStruct'] } }
769f08c8ecSEric Blake
774f193e34SMarkus Armbruster# for testing unions
78d220fbcdSEric Blake# Among other things, test that a name collision between branches does
79d220fbcdSEric Blake# not cause any problems (since only one branch can be in use at a time),
80d220fbcdSEric Blake# by intentionally using two branches that both have a C member 'a_b'
81895a2a80SEric Blake{ 'struct': 'UserDefA',
82d220fbcdSEric Blake  'data': { 'boolean': 'bool', '*a_b': 'int' } }
834f193e34SMarkus Armbruster
84895a2a80SEric Blake{ 'struct': 'UserDefB',
85d220fbcdSEric Blake  'data': { 'intb': 'int', '*a-b': 'bool' } }
864f193e34SMarkus Armbruster
878c3f8e77SMarkus Armbruster{ 'union': 'UserDefFlatUnion',
888c3f8e77SMarkus Armbruster  'base': 'UserDefUnionBase',   # intentional forward reference
898c3f8e77SMarkus Armbruster  'discriminator': 'enum1',
9087adbbffSMarc-André Lureau  'data': { 'value1' : {'type': 'UserDefA'},
918c3f8e77SMarkus Armbruster            'value2' : 'UserDefB',
92800877bbSAnton Nefedov            'value3' : 'UserDefB'
93800877bbSAnton Nefedov            # 'value4' defaults to empty
94800877bbSAnton Nefedov  } }
95cb55111bSMichael Roth
96895a2a80SEric Blake{ 'struct': 'UserDefUnionBase',
9780e60a19SMarkus Armbruster  'base': 'UserDefZero',
985223070cSWenchao Xia  'data': { 'string': 'str', 'enum1': 'EnumOne' } }
995223070cSWenchao Xia
10014f00c6cSEric Blake# this variant of UserDefFlatUnion defaults to a union that uses members with
101cb55111bSMichael Roth# allocated types to test corner cases in the cleanup/dealloc visitor
102cb55111bSMichael Roth{ 'union': 'UserDefFlatUnion2',
103ac4338f8SEric Blake  'base': { '*integer': 'int', 'string': 'str', 'enum1': 'QEnumTwo' },
104cb55111bSMichael Roth  'discriminator': 'enum1',
1058c3f8e77SMarkus Armbruster  'data': { 'value1' : 'UserDefC', # intentional forward reference
1069d3524b3SEric Blake            'value2' : 'UserDefB' } }
107cb55111bSMichael Roth
10868d07839SEric Blake{ 'struct': 'WrapAlternate',
10968d07839SEric Blake  'data': { 'alt': 'UserDefAlternate' } }
110ab916fadSEric Blake{ 'alternate': 'UserDefAlternate',
11187adbbffSMarc-André Lureau  'data': { 'udfu': {'type': 'UserDefFlatUnion'}, 'e': 'EnumOne', 'i': 'int',
1124d2d5c41SMarkus Armbruster            'n': 'null' } }
1132c38b600SMarkus Armbruster
1148c3f8e77SMarkus Armbruster{ 'struct': 'UserDefC',
1158c3f8e77SMarkus Armbruster  'data': { 'string1': 'str', 'string2': 'str' } }
1168c3f8e77SMarkus Armbruster
1179c51b441SEric Blake# for testing use of 'number' within alternates
1188168ca8eSMarkus Armbruster{ 'alternate': 'AltEnumBool', 'data': { 'e': 'EnumOne', 'b': 'bool' } }
1198168ca8eSMarkus Armbruster{ 'alternate': 'AltEnumNum', 'data': { 'e': 'EnumOne', 'n': 'number' } }
1208168ca8eSMarkus Armbruster{ 'alternate': 'AltNumEnum', 'data': { 'n': 'number', 'e': 'EnumOne' } }
1218168ca8eSMarkus Armbruster{ 'alternate': 'AltEnumInt', 'data': { 'e': 'EnumOne', 'i': 'int' } }
1229c51b441SEric Blake
123c0644771SMarkus Armbruster# for testing use of 'str' within alternates
124c0644771SMarkus Armbruster{ 'alternate': 'AltStrObj', 'data': { 's': 'str', 'o': 'TestStruct' } }
125c0644771SMarkus Armbruster
126b359f4b2SMarkus Armbruster# for testing lists
127b359f4b2SMarkus Armbruster{ 'union': 'UserDefListUnion',
1284f193e34SMarkus Armbruster  'data': { 'integer': ['int'],
1294f193e34SMarkus Armbruster            's8': ['int8'],
1304f193e34SMarkus Armbruster            's16': ['int16'],
1314f193e34SMarkus Armbruster            's32': ['int32'],
1324f193e34SMarkus Armbruster            's64': ['int64'],
1334f193e34SMarkus Armbruster            'u8': ['uint8'],
1344f193e34SMarkus Armbruster            'u16': ['uint16'],
1354f193e34SMarkus Armbruster            'u32': ['uint32'],
1364f193e34SMarkus Armbruster            'u64': ['uint64'],
1374f193e34SMarkus Armbruster            'number': ['number'],
1384f193e34SMarkus Armbruster            'boolean': ['bool'],
139cb17f79eSEric Blake            'string': ['str'],
14028770e05SMarkus Armbruster            'sizes': ['size'],
1415e12eb98SMarkus Armbruster            'any': ['any'],
1425e12eb98SMarkus Armbruster            'user': ['Status'] } } # intentional forward ref. to sub-module
1434f193e34SMarkus Armbruster
144709395f8SMarkus Armbruster# for testing sub-modules
145709395f8SMarkus Armbruster{ 'include': 'include/sub-module.json' }
146709395f8SMarkus Armbruster
1474f193e34SMarkus Armbruster# testing commands
14805ebf841SMarkus Armbruster{ 'command': 'user-def-cmd', 'data': {} }
14905ebf841SMarkus Armbruster{ 'command': 'user-def-cmd1', 'data': {'ud1a': 'UserDefOne'} }
15005ebf841SMarkus Armbruster{ 'command': 'user-def-cmd2',
15187adbbffSMarc-André Lureau  'data': {'ud1a': {'type': 'UserDefOne'}, '*ud1b': 'UserDefOne'},
152ab22ad96SMarkus Armbruster  'returns': 'UserDefTwo' }
153cae95eaeSEric Blake
154ae6bf766SMarc-André Lureau{ 'command': 'cmd-success-response', 'data': {}, 'success-response': false }
15504f22362SKevin Wolf{ 'command': 'coroutine-cmd', 'data': {}, 'coroutine': true }
156ae6bf766SMarc-André Lureau
157cae95eaeSEric Blake# Returning a non-dictionary requires a name from the whitelist
158cae95eaeSEric Blake{ 'command': 'guest-get-time', 'data': {'a': 'int', '*b': 'int' },
159c2216a8aSMarkus Armbruster  'returns': 'int' }
16028770e05SMarkus Armbruster{ 'command': 'guest-sync', 'data': { 'arg': 'any' }, 'returns': 'any' }
161c818408eSEric Blake{ 'command': 'boxed-struct', 'boxed': true, 'data': 'UserDefZero' }
162b359f4b2SMarkus Armbruster{ 'command': 'boxed-union', 'data': 'UserDefListUnion', 'boxed': true }
163675b214bSMarkus Armbruster{ 'command': 'boxed-empty', 'boxed': true, 'data': 'Empty1' }
1643953e3a5SLaszlo Ersek
165c0698212SMarkus Armbruster# Smoke test on out-of-band and allow-preconfig-test
1667b13f2c2SIgor Mammedov{ 'command': 'test-flags-command', 'allow-oob': true, 'allow-preconfig': true }
1671a1b11dcSPeter Xu
1683953e3a5SLaszlo Ersek# For testing integer range flattening in opts-visitor. The following schema
1693953e3a5SLaszlo Ersek# corresponds to the option format:
1703953e3a5SLaszlo Ersek#
1713953e3a5SLaszlo Ersek# -userdef i64=3-6,i64=-5--1,u64=2,u16=1,u16=7-12
1723953e3a5SLaszlo Ersek#
1733953e3a5SLaszlo Ersek# For simplicity, this example doesn't use [type=]discriminator nor optargs
1743953e3a5SLaszlo Ersek# specific to discriminator values.
175895a2a80SEric Blake{ 'struct': 'UserDefOptions',
1763953e3a5SLaszlo Ersek  'data': {
1773953e3a5SLaszlo Ersek    '*i64' : [ 'int'    ],
1783953e3a5SLaszlo Ersek    '*u64' : [ 'uint64' ],
1793953e3a5SLaszlo Ersek    '*u16' : [ 'uint16' ],
1803953e3a5SLaszlo Ersek    '*i64x':   'int'     ,
1813953e3a5SLaszlo Ersek    '*u64x':   'uint64'  } }
182f6dadb02SWenchao Xia
183f6dadb02SWenchao Xia# testing event
184895a2a80SEric Blake{ 'struct': 'EventStructOne',
18587adbbffSMarc-André Lureau  'data': { 'struct1': {'type': 'UserDefOne'}, 'string': 'str', '*enum2': 'EnumOne' } }
186f6dadb02SWenchao Xia
187f6dadb02SWenchao Xia{ 'event': 'EVENT_A' }
188f6dadb02SWenchao Xia{ 'event': 'EVENT_B',
189f6dadb02SWenchao Xia  'data': { } }
190f6dadb02SWenchao Xia{ 'event': 'EVENT_C',
191f6dadb02SWenchao Xia  'data': { '*a': 'int', '*b': 'UserDefOne', 'c': 'str' } }
192f6dadb02SWenchao Xia{ 'event': 'EVENT_D',
193f6dadb02SWenchao Xia  'data': { 'a' : 'EventStructOne', 'b' : 'str', '*c': 'str', '*enum3': 'EnumOne' } }
194c818408eSEric Blake{ 'event': 'EVENT_E', 'boxed': true, 'data': 'UserDefZero' }
195b22e8658SMarkus Armbruster{ 'event': 'EVENT_F', 'boxed': true, 'data': 'UserDefFlatUnion' }
196675b214bSMarkus Armbruster{ 'event': 'EVENT_G', 'boxed': true, 'data': 'Empty1' }
197fce384b8SEric Blake
198c43567c1SEric Blake# test that we correctly compile downstream extensions, as well as munge
199c43567c1SEric Blake# ticklish names
200f0325536SMarkus Armbruster# also test union and alternate with just one branch
201fce384b8SEric Blake{ 'enum': '__org.qemu_x-Enum', 'data': [ '__org.qemu_x-value' ] }
20283a02706SEric Blake{ 'struct': '__org.qemu_x-Base',
20383a02706SEric Blake  'data': { '__org.qemu_x-member1': '__org.qemu_x-Enum' } }
20483a02706SEric Blake{ 'struct': '__org.qemu_x-Struct', 'base': '__org.qemu_x-Base',
205c43567c1SEric Blake  'data': { '__org.qemu_x-member2': 'str', '*wchar-t': 'int' } }
206bb337290SEric Blake{ 'union': '__org.qemu_x-Union1', 'data': { '__org.qemu_x-branch': 'str' } }
207f0325536SMarkus Armbruster{ 'alternate': '__org.qemu_x-Alt1', 'data': { '__org.qemu_x-branch': 'str' } }
208857af5f0SEric Blake{ 'struct': '__org.qemu_x-Struct2',
209857af5f0SEric Blake  'data': { 'array': ['__org.qemu_x-Union1'] } }
210857af5f0SEric Blake{ 'union': '__org.qemu_x-Union2', 'base': '__org.qemu_x-Base',
211857af5f0SEric Blake  'discriminator': '__org.qemu_x-member1',
212857af5f0SEric Blake  'data': { '__org.qemu_x-value': '__org.qemu_x-Struct2' } }
213d1f07c86SEric Blake{ 'alternate': '__org.qemu_x-Alt',
214f0325536SMarkus Armbruster  'data': { '__org.qemu_x-branch': '__org.qemu_x-Base' } }
215e3c4c3d7SEric Blake{ 'event': '__ORG.QEMU_X-EVENT', 'data': '__org.qemu_x-Struct' }
216e3c4c3d7SEric Blake{ 'command': '__org.qemu_x-command',
217e3c4c3d7SEric Blake  'data': { 'a': ['__org.qemu_x-Enum'], 'b': ['__org.qemu_x-Struct'],
218e3c4c3d7SEric Blake            'c': '__org.qemu_x-Union2', 'd': '__org.qemu_x-Alt' },
219e3c4c3d7SEric Blake  'returns': '__org.qemu_x-Union1' }
220967c8851SMarc-André Lureau
221967c8851SMarc-André Lureau# test 'if' condition handling
222967c8851SMarc-André Lureau
223ccadd6bcSMarc-André Lureau{ 'struct': 'TestIfStruct', 'data':
224ccadd6bcSMarc-André Lureau  { 'foo': 'int',
225*8a9f1e1dSMarc-André Lureau    'bar': { 'type': 'int', 'if': 'TEST_IF_STRUCT_BAR'} },
226*8a9f1e1dSMarc-André Lureau  'if': 'TEST_IF_STRUCT' }
227967c8851SMarc-André Lureau
2286cc32b0eSMarc-André Lureau{ 'enum': 'TestIfEnum', 'data':
229*8a9f1e1dSMarc-André Lureau  [ 'foo', { 'name' : 'bar', 'if': 'TEST_IF_ENUM_BAR' } ],
230*8a9f1e1dSMarc-André Lureau  'if': 'TEST_IF_ENUM' }
231967c8851SMarc-André Lureau
232a2724280SMarc-André Lureau{ 'union': 'TestIfUnion', 'data':
233a2724280SMarc-André Lureau  { 'foo': 'TestStruct',
234*8a9f1e1dSMarc-André Lureau    'bar': { 'type': 'str', 'if': 'TEST_IF_UNION_BAR'} },
235*8a9f1e1dSMarc-André Lureau  'if': { 'all': ['TEST_IF_UNION', 'TEST_IF_STRUCT'] } }
236967c8851SMarc-André Lureau
23705ebf841SMarkus Armbruster{ 'command': 'test-if-union-cmd',
2385aceeac0SMarkus Armbruster  'data': { 'union-cmd-arg': 'TestIfUnion' },
239*8a9f1e1dSMarc-André Lureau  'if': 'TEST_IF_UNION' }
240f8c4fdd6SMarc-André Lureau
2413e270dcaSMarc-André Lureau{ 'alternate': 'TestIfAlternate', 'data':
2423e270dcaSMarc-André Lureau  { 'foo': 'int',
243*8a9f1e1dSMarc-André Lureau    'bar': { 'type': 'TestStruct', 'if': 'TEST_IF_ALT_BAR'} },
244*8a9f1e1dSMarc-André Lureau  'if': { 'all': ['TEST_IF_ALT', 'TEST_IF_STRUCT'] } }
245967c8851SMarc-André Lureau
24605ebf841SMarkus Armbruster{ 'command': 'test-if-alternate-cmd',
2475aceeac0SMarkus Armbruster  'data': { 'alt-cmd-arg': 'TestIfAlternate' },
248*8a9f1e1dSMarc-André Lureau  'if': { 'all': ['TEST_IF_ALT',
249*8a9f1e1dSMarc-André Lureau                  {'not': 'TEST_IF_NOT_ALT'}] } }
250f8c4fdd6SMarc-André Lureau
25105ebf841SMarkus Armbruster{ 'command': 'test-if-cmd',
25205ebf841SMarkus Armbruster  'data': {
25305ebf841SMarkus Armbruster    'foo': 'TestIfStruct',
254*8a9f1e1dSMarc-André Lureau    'bar': { 'type': 'TestIfEnum', 'if': 'TEST_IF_CMD_BAR' } },
255967c8851SMarc-André Lureau  'returns': 'UserDefThree',
256*8a9f1e1dSMarc-André Lureau  'if': { 'all': ['TEST_IF_CMD', 'TEST_IF_STRUCT'] } }
257967c8851SMarc-André Lureau
25805ebf841SMarkus Armbruster{ 'command': 'test-cmd-return-def-three', 'returns': 'UserDefThree' }
259967c8851SMarc-André Lureau
260d4f4cae8SMarkus Armbruster{ 'event': 'TEST_IF_EVENT', 'data':
261ccadd6bcSMarc-André Lureau  { 'foo': 'TestIfStruct',
262*8a9f1e1dSMarc-André Lureau    'bar': { 'type': ['TestIfEnum'], 'if': 'TEST_IF_EVT_BAR' } },
263*8a9f1e1dSMarc-André Lureau  'if': { 'all': ['TEST_IF_EVT', 'TEST_IF_STRUCT'] } }
2648aa3a33eSKevin Wolf
265013b4efcSMarkus Armbruster# test 'features'
2668aa3a33eSKevin Wolf
2678aa3a33eSKevin Wolf{ 'struct': 'FeatureStruct0',
2688aa3a33eSKevin Wolf  'data': { 'foo': 'int' },
2698aa3a33eSKevin Wolf  'features': [] }
2708aa3a33eSKevin Wolf{ 'struct': 'FeatureStruct1',
271f965e8feSMarkus Armbruster  'data': { 'foo': { 'type': 'int', 'features': [ 'deprecated' ] } },
2728aa3a33eSKevin Wolf  'features': [ 'feature1' ] }
2738aa3a33eSKevin Wolf{ 'struct': 'FeatureStruct2',
2748aa3a33eSKevin Wolf  'data': { 'foo': 'int' },
2758aa3a33eSKevin Wolf  'features': [ { 'name': 'feature1' } ] }
2768aa3a33eSKevin Wolf{ 'struct': 'FeatureStruct3',
2778aa3a33eSKevin Wolf  'data': { 'foo': 'int' },
2788aa3a33eSKevin Wolf  'features': [ 'feature1', 'feature2' ] }
2798aa3a33eSKevin Wolf{ 'struct': 'FeatureStruct4',
2808aa3a33eSKevin Wolf  'data': { 'namespace-test': 'int' },
2818aa3a33eSKevin Wolf  'features': [ 'namespace-test', 'int', 'name', 'if' ] }
2828aa3a33eSKevin Wolf
2838aa3a33eSKevin Wolf{ 'struct': 'CondFeatureStruct1',
2848aa3a33eSKevin Wolf  'data': { 'foo': 'int' },
285*8a9f1e1dSMarc-André Lureau  'features': [ { 'name': 'feature1', 'if': 'TEST_IF_FEATURE_1'} ] }
2868aa3a33eSKevin Wolf{ 'struct': 'CondFeatureStruct2',
2878aa3a33eSKevin Wolf  'data': { 'foo': 'int' },
288*8a9f1e1dSMarc-André Lureau  'features': [ { 'name': 'feature1', 'if': 'TEST_IF_FEATURE_1'},
289*8a9f1e1dSMarc-André Lureau                { 'name': 'feature2', 'if': 'TEST_IF_FEATURE_2'} ] }
2908aa3a33eSKevin Wolf{ 'struct': 'CondFeatureStruct3',
2918aa3a33eSKevin Wolf  'data': { 'foo': 'int' },
2925d83b9a1SMarc-André Lureau  'features': [ { 'name': 'feature1',
293*8a9f1e1dSMarc-André Lureau                  'if': { 'all': [ 'TEST_IF_COND_1',
294*8a9f1e1dSMarc-André Lureau                                   'TEST_IF_COND_2'] } } ] }
2953ad64edfSMarc-André Lureau{ 'struct': 'CondFeatureStruct4',
2963ad64edfSMarc-André Lureau  'data': { 'foo': 'int' },
2973ad64edfSMarc-André Lureau  'features': [ { 'name': 'feature1',
298*8a9f1e1dSMarc-André Lureau                  'if': {'any': ['TEST_IF_COND_1',
299*8a9f1e1dSMarc-André Lureau                                 'TEST_IF_COND_2'] } } ] }
300013b4efcSMarkus Armbruster
301013b4efcSMarkus Armbruster{ 'enum': 'FeatureEnum1',
302013b4efcSMarkus Armbruster  'data': [ 'eins', 'zwei', 'drei' ],
303013b4efcSMarkus Armbruster  'features': [ 'feature1' ] }
304013b4efcSMarkus Armbruster
305013b4efcSMarkus Armbruster{ 'union': 'FeatureUnion1',
306013b4efcSMarkus Armbruster  'base': { 'tag': 'FeatureEnum1' },
307013b4efcSMarkus Armbruster  'discriminator': 'tag',
308013b4efcSMarkus Armbruster  'data': { 'eins': 'FeatureStruct1' },
309013b4efcSMarkus Armbruster  'features': [ 'feature1' ] }
310013b4efcSMarkus Armbruster
311013b4efcSMarkus Armbruster{ 'alternate': 'FeatureAlternate1',
312013b4efcSMarkus Armbruster  'data': { 'eins': 'FeatureStruct1' },
313013b4efcSMarkus Armbruster  'features': [ 'feature1' ] }
314013b4efcSMarkus Armbruster
315013b4efcSMarkus Armbruster{ 'command': 'test-features0',
31691fa93e5SMarkus Armbruster  'data': { '*fs0': 'FeatureStruct0',
31791fa93e5SMarkus Armbruster            '*fs1': 'FeatureStruct1',
31891fa93e5SMarkus Armbruster            '*fs2': 'FeatureStruct2',
31991fa93e5SMarkus Armbruster            '*fs3': 'FeatureStruct3',
32091fa93e5SMarkus Armbruster            '*fs4': 'FeatureStruct4',
32191fa93e5SMarkus Armbruster            '*cfs1': 'CondFeatureStruct1',
32291fa93e5SMarkus Armbruster            '*cfs2': 'CondFeatureStruct2',
3233ad64edfSMarc-André Lureau            '*cfs3': 'CondFeatureStruct3',
3243ad64edfSMarc-André Lureau            '*cfs4': 'CondFeatureStruct4' },
32591fa93e5SMarkus Armbruster  'returns': 'FeatureStruct1',
3262e2e0df2SPeter Krempa  'features': [] }
327013b4efcSMarkus Armbruster
3282e2e0df2SPeter Krempa{ 'command': 'test-command-features1',
329f965e8feSMarkus Armbruster  'features': [ 'deprecated' ] }
3302e2e0df2SPeter Krempa{ 'command': 'test-command-features3',
3312e2e0df2SPeter Krempa  'features': [ 'feature1', 'feature2' ] }
3322e2e0df2SPeter Krempa
3332e2e0df2SPeter Krempa{ 'command': 'test-command-cond-features1',
334*8a9f1e1dSMarc-André Lureau  'features': [ { 'name': 'feature1', 'if': 'TEST_IF_FEATURE_1'} ] }
3352e2e0df2SPeter Krempa{ 'command': 'test-command-cond-features2',
336*8a9f1e1dSMarc-André Lureau  'features': [ { 'name': 'feature1', 'if': 'TEST_IF_FEATURE_1'},
337*8a9f1e1dSMarc-André Lureau                { 'name': 'feature2', 'if': 'TEST_IF_FEATURE_2'} ] }
3382e2e0df2SPeter Krempa{ 'command': 'test-command-cond-features3',
3395d83b9a1SMarc-André Lureau  'features': [ { 'name': 'feature1',
340*8a9f1e1dSMarc-André Lureau                  'if': { 'all': [ 'TEST_IF_COND_1',
341*8a9f1e1dSMarc-André Lureau                                   'TEST_IF_COND_2'] } } ] }
342013b4efcSMarkus Armbruster
343d4f4cae8SMarkus Armbruster{ 'event': 'TEST_EVENT_FEATURES0',
344a291a38fSMarkus Armbruster  'data': 'FeatureStruct1' }
345a291a38fSMarkus Armbruster
346d4f4cae8SMarkus Armbruster{ 'event': 'TEST_EVENT_FEATURES1',
347f965e8feSMarkus Armbruster  'features': [ 'deprecated' ] }
348