| #!/usr/bin/env python3 |
| |
| import unittest |
| from vppapigen import VPPAPI, Option, ParseError, Union, foldup_crcs, \ |
| global_types |
| import vppapigen |
| |
| |
| # TODO |
| # - test parsing of options, typedefs, enums, defines |
| # - test JSON, C output |
| |
| |
| class TestVersion(unittest.TestCase): |
| @classmethod |
| def setUpClass(cls): |
| cls.parser = VPPAPI() |
| |
| def test_version(self): |
| version_string = 'option version = "1.0.0";' |
| r = self.parser.parse_string(version_string) |
| self.assertTrue(isinstance(r[0], Option)) |
| |
| |
| class TestUnion(unittest.TestCase): |
| @classmethod |
| def setUpClass(cls): |
| cls.parser = VPPAPI() |
| |
| def test_union(self): |
| test_string = ''' |
| union foo_union { |
| u32 a; |
| u8 b; |
| }; |
| ''' |
| r = self.parser.parse_string(test_string) |
| self.assertTrue(isinstance(r[0], Union)) |
| |
| def test_union_vla(self): |
| test_string = ''' |
| union foo_union_vla { |
| u32 a; |
| u8 b[a]; |
| }; |
| autoreply define foo { |
| vl_api_foo_union_vla_t v; |
| }; |
| ''' |
| r = self.parser.parse_string(test_string) |
| self.assertTrue(isinstance(r[0], Union)) |
| self.assertTrue(r[0].vla) |
| s = self.parser.process(r) |
| |
| test_string2 = ''' |
| union foo_union_vla2 { |
| u32 a; |
| u8 b[a]; |
| u32 c; |
| }; |
| autoreply define foo2 { |
| vl_api_foo_union_vla2_t v; |
| }; |
| ''' |
| self.assertRaises(ValueError, self.parser.parse_string, test_string2) |
| |
| test_string3 = ''' |
| union foo_union_vla3 { |
| u32 a; |
| u8 b[a]; |
| }; |
| autoreply define foo3 { |
| vl_api_foo_union_vla3_t v; |
| u32 x; |
| }; |
| ''' |
| self.assertRaises(ValueError, self.parser.parse_string, test_string3) |
| |
| |
| class TestTypedef(unittest.TestCase): |
| @classmethod |
| def setUpClass(cls): |
| cls.parser = VPPAPI() |
| |
| def test_duplicatetype(self): |
| test_string = ''' |
| typedef foo1 { u8 dummy; }; |
| typedef foo1 { u8 dummy; }; |
| ''' |
| self.assertRaises(KeyError, self.parser.parse_string, test_string) |
| |
| |
| class TestDefine(unittest.TestCase): |
| @classmethod |
| def setUpClass(cls): |
| cls.parser = VPPAPI() |
| |
| def test_unknowntype(self): |
| test_string = 'define foo { foobar foo;};' |
| with self.assertRaises(ParseError) as ctx: |
| self.parser.parse_string(test_string) |
| self.assertIn('Undefined type: foobar', str(ctx.exception)) |
| |
| test_string = 'define { u8 foo;};' |
| with self.assertRaises(ParseError) as ctx: |
| self.parser.parse_string(test_string) |
| |
| def test_flags(self): |
| test_string = ''' |
| manual_print dont_trace manual_endian define foo { u8 foo; }; |
| define foo_reply {u32 context; i32 retval; }; |
| ''' |
| r = self.parser.parse_string(test_string) |
| self.assertIsNotNone(r) |
| s = self.parser.process(r) |
| self.assertIsNotNone(s) |
| for d in s['Define']: |
| if d.name == 'foo': |
| self.assertTrue(d.dont_trace) |
| self.assertTrue(d.manual_endian) |
| self.assertTrue(d.manual_print) |
| self.assertFalse(d.autoreply) |
| |
| test_string = ''' |
| nonexisting_flag define foo { u8 foo; }; |
| ''' |
| with self.assertRaises(ParseError): |
| self.parser.parse_string(test_string) |
| |
| def test_options(self): |
| test_string = ''' |
| define foo { option deprecated; u8 foo; }; |
| define foo_reply {u32 context; i32 retval; }; |
| ''' |
| r = self.parser.parse_string(test_string) |
| self.assertIsNotNone(r) |
| s = self.parser.process(r) |
| self.assertIsNotNone(s) |
| |
| |
| class TestService(unittest.TestCase): |
| @classmethod |
| def setUpClass(cls): |
| cls.parser = VPPAPI() |
| |
| def test_service(self): |
| test_string = ''' |
| autoreply define show_version { u8 foo;}; |
| service { rpc show_version returns show_version_reply; }; |
| ''' |
| r = self.parser.parse_string(test_string) |
| s = self.parser.process(r) |
| self.assertEqual(s['Service'][0].caller, 'show_version') |
| self.assertEqual(s['Service'][0].reply, 'show_version_reply') |
| |
| |
| def get_crc(apistring, name): |
| vppapigen.global_types = {} |
| parser = vppapigen.VPPAPI() |
| r = parser.parse_string(apistring) |
| s = parser.process(r) |
| foldup_crcs(s['Define']) |
| d = [f for f in s['Define'] if f.name == name] |
| return d[0].crc |
| |
| |
| class TestCRC(unittest.TestCase): |
| def test_crc(self): |
| test_string = ''' |
| typedef list { u8 foo; }; |
| autoreply define foo { u8 foo; vl_api_list_t l;}; |
| ''' |
| crc = get_crc(test_string, 'foo') |
| |
| # modify underlying type |
| test_string = ''' |
| typedef list { u8 foo2; }; |
| autoreply define foo { u8 foo; vl_api_list_t l;}; |
| ''' |
| crc2 = get_crc(test_string, 'foo') |
| self.assertNotEqual(crc, crc2) |
| |
| # two user-defined types |
| test_string = ''' |
| typedef address { u8 foo2; }; |
| typedef list { u8 foo2; vl_api_address_t add; }; |
| autoreply define foo { u8 foo; vl_api_list_t l;}; |
| ''' |
| crc3 = get_crc(test_string, 'foo') |
| |
| test_string = ''' |
| typedef address { u8 foo3; }; |
| typedef list { u8 foo2; vl_api_address_t add; }; |
| autoreply define foo { u8 foo; vl_api_list_t l;}; |
| ''' |
| crc4 = get_crc(test_string, 'foo') |
| self.assertNotEqual(crc3, crc4) |
| |
| test_string = ''' |
| typedef address { u8 foo3; }; |
| typedef list { u8 foo2; vl_api_address_t add; u8 foo3; }; |
| autoreply define foo { u8 foo; vl_api_list_t l;}; |
| ''' |
| crc5 = get_crc(test_string, 'foo') |
| self.assertNotEqual(crc4, crc5) |
| |
| test_string = ''' |
| typedef ip6_address |
| { |
| u8 foo; |
| }; |
| typedef srv6_sid_list |
| { |
| u8 num_sids; |
| u32 weight; |
| u32 sl_index; |
| vl_api_ip6_address_t sids[16]; |
| }; |
| autoreply define sr_policy_add |
| { |
| u32 client_index; |
| u32 context; |
| vl_api_ip6_address_t bsid_addr; |
| u32 weight; |
| bool is_encap; |
| bool is_spray; |
| u32 fib_table; |
| vl_api_srv6_sid_list_t sids; |
| }; |
| ''' |
| |
| crc = get_crc(test_string, 'sr_policy_add') |
| |
| test_string = ''' |
| typedef ip6_address |
| { |
| u8 foo; |
| }; |
| typedef srv6_sid_list |
| { |
| u8 num_sids; |
| u32 weight; |
| vl_api_ip6_address_t sids[16]; |
| }; |
| autoreply define sr_policy_add |
| { |
| u32 client_index; |
| u32 context; |
| vl_api_ip6_address_t bsid_addr; |
| u32 weight; |
| bool is_encap; |
| bool is_spray; |
| u32 fib_table; |
| vl_api_srv6_sid_list_t sids; |
| }; |
| ''' |
| crc2 = get_crc(test_string, 'sr_policy_add') |
| |
| self.assertNotEqual(crc, crc2) |
| |
| |
| class TestEnum(unittest.TestCase): |
| |
| @classmethod |
| def setUpClass(cls): |
| cls.parser = VPPAPI() |
| |
| def test_enum_as_enum(self): |
| test_string = """\ |
| enum tunnel_mode : u8 |
| { |
| /** point-to-point */ |
| TUNNEL_API_MODE_P2P = 0, |
| /** multi-point */ |
| TUNNEL_API_MODE_MP, |
| }; |
| """ |
| r = self.parser.parse_string(test_string) |
| self.assertIsNotNone(r) |
| s = self.parser.process(r) |
| for o in s['types']: |
| if o.type == 'Enum': |
| self.assertEqual(o.name, "tunnel_mode") |
| break |
| else: |
| self.fail() |
| |
| def test_enumflag_as_enum(self): |
| test_string = """\ |
| enum virtio_flags { |
| VIRTIO_API_FLAG_GSO = 1, /* enable gso on the interface */ |
| VIRTIO_API_FLAG_CSUM_OFFLOAD = 2, /* enable checksum offload without gso on the interface */ |
| VIRTIO_API_FLAG_GRO_COALESCE = 4, /* enable packet coalescing on tx side, provided gso enabled */ |
| VIRTIO_API_FLAG_PACKED = 8, /* enable packed ring support, provided it is available from backend */ |
| VIRTIO_API_FLAG_IN_ORDER = 16, /* enable in order support, provided it is available from backend */ |
| VIRTIO_API_FLAG_BUFFERING = 32 [backwards_compatible], /* enable buffering to handle backend jitter/delays */ |
| };""" |
| r = self.parser.parse_string(test_string) |
| self.assertIsNotNone(r) |
| s = self.parser.process(r) |
| for o in s['types']: |
| if o.type == 'Enum': |
| self.assertEqual(o.name, "virtio_flags") |
| break |
| else: |
| self.fail() |
| |
| |
| class TestEnumFlag(unittest.TestCase): |
| |
| @classmethod |
| def setUpClass(cls): |
| cls.parser = VPPAPI() |
| |
| def test_enum_as_enumflag(self): |
| test_string = """\ |
| enumflag tunnel_mode_ef : u8 |
| { |
| /** point-to-point */ |
| TUNNEL_API_MODE_P2P = 0, |
| /** multi-point */ |
| TUNNEL_API_MODE_MP, |
| TUNNEL_API_MODE_FOO, |
| TUNNEL_API_MODE_BAR, |
| };""" |
| with self.assertRaises(TypeError) as ctx: |
| r = self.parser.parse_string(test_string) |
| |
| self.assertTrue(str(ctx.exception).startswith( |
| 'tunnel_mode_ef is not a flag enum.')) |
| |
| def test_enumflag_as_enumflag(self): |
| test_string = """\ |
| enumflag virtio_flags_ef { |
| VIRTIO_API_FLAG_GSO = 1, /* enable gso on the interface */ |
| VIRTIO_API_FLAG_CSUM_OFFLOAD = 2, /* enable checksum offload without gso on the interface */ |
| VIRTIO_API_FLAG_GRO_COALESCE = 4, /* enable packet coalescing on tx side, provided gso enabled */ |
| VIRTIO_API_FLAG_PACKED = 8, /* enable packed ring support, provided it is available from backend */ |
| VIRTIO_API_FLAG_IN_ORDER = 16, /* enable in order support, provided it is available from backend */ |
| VIRTIO_API_FLAG_BUFFERING = 32 [backwards_compatible], /* enable buffering to handle backend jitter/delays */ |
| };""" |
| r = self.parser.parse_string(test_string) |
| self.assertIsNotNone(r) |
| s = self.parser.process(r) |
| for o in s['types']: |
| if o.type == 'EnumFlag': |
| self.assertEqual(o.name, "virtio_flags_ef") |
| break |
| else: |
| self.fail() |
| |
| |
| if __name__ == '__main__': |
| unittest.main(verbosity=2) |