VPP-1508 python3 tests: .encode('hex')
[vpp.git] / test / vpp_interface.py
1 import binascii
2 import socket
3 from abc import abstractmethod, ABCMeta
4
5 from six import moves
6
7 from util import Host, mk_ll_addr, mactobinary
8
9
10 class VppInterface(object):
11     """Generic VPP interface."""
12     __metaclass__ = ABCMeta
13
14     @property
15     def sw_if_index(self):
16         """Interface index assigned by VPP."""
17         return self._sw_if_index
18
19     @property
20     def remote_mac(self):
21         """MAC-address of the remote interface "connected" to this interface"""
22         return self._remote_hosts[0].mac
23
24     @property
25     def local_mac(self):
26         """MAC-address of the VPP interface."""
27         return self._local_mac
28
29     @property
30     def local_addr(self):
31         return self._local_addr
32
33     @property
34     def remote_addr(self):
35         return self._remote_addr
36
37     @property
38     def local_addr_n(self):
39         return self._local_addr_n
40
41     @property
42     def remote_addr_n(self):
43         return self._remote_addr_n
44
45     @property
46     def local_ip4(self):
47         """Local IPv4 address on VPP interface (string)."""
48         return self._local_ip4
49
50     @property
51     def local_ip4n(self):
52         """Local IPv4 address - raw, suitable as API parameter."""
53         return socket.inet_pton(socket.AF_INET, self._local_ip4)
54
55     @property
56     def remote_ip4(self):
57         """IPv4 address of remote peer "connected" to this interface."""
58         return self._remote_hosts[0].ip4
59
60     @property
61     def remote_ip4n(self):
62         """IPv4 address of remote peer - raw, suitable as API parameter."""
63         return socket.inet_pton(socket.AF_INET, self.remote_ip4)
64
65     @property
66     def local_ip6(self):
67         """Local IPv6 address on VPP interface (string)."""
68         return self._local_ip6
69
70     @property
71     def local_ip6n(self):
72         """Local IPv6 address - raw, suitable as API parameter."""
73         return socket.inet_pton(socket.AF_INET6, self.local_ip6)
74
75     @property
76     def remote_ip6(self):
77         """IPv6 address of remote peer "connected" to this interface."""
78         return self._remote_hosts[0].ip6
79
80     @property
81     def remote_ip6n(self):
82         """IPv6 address of remote peer - raw, suitable as API parameter"""
83         return socket.inet_pton(socket.AF_INET6, self.remote_ip6)
84
85     @property
86     def local_ip6_ll(self):
87         """Local IPv6 link-local address on VPP interface (string)."""
88         return self._local_ip6_ll
89
90     @property
91     def local_ip6n_ll(self):
92         """Local IPv6 link-local address - raw, suitable as API parameter."""
93         return self._local_ip6n_ll
94
95     @property
96     def remote_ip6_ll(self):
97         """Link-local IPv6 address of remote peer
98         "connected" to this interface."""
99         return self._remote_ip6_ll
100
101     @property
102     def remote_ip6n_ll(self):
103         """Link-local IPv6 address of remote peer
104         - raw, suitable as API parameter"""
105         return self._remote_ip6n_ll
106
107     @property
108     def name(self):
109         """Name of the interface."""
110         return self._name
111
112     @property
113     def dump(self):
114         """RAW result of sw_interface_dump for this interface."""
115         return self._dump
116
117     @property
118     def test(self):
119         """Test case creating this interface."""
120         return self._test
121
122     @property
123     def remote_hosts(self):
124         """Remote hosts list"""
125         return self._remote_hosts
126
127     @remote_hosts.setter
128     def remote_hosts(self, value):
129         """
130         :param list value: List of remote hosts.
131         """
132         self._remote_hosts = value
133         self._hosts_by_mac = {}
134         self._hosts_by_ip4 = {}
135         self._hosts_by_ip6 = {}
136         for host in self._remote_hosts:
137             self._hosts_by_mac[host.mac] = host
138             self._hosts_by_ip4[host.ip4] = host
139             self._hosts_by_ip6[host.ip6] = host
140
141     def host_by_mac(self, mac):
142         """
143         :param mac: MAC address to find host by.
144         :return: Host object assigned to interface.
145         """
146         return self._hosts_by_mac[mac]
147
148     def host_by_ip4(self, ip):
149         """
150         :param ip: IPv4 address to find host by.
151         :return: Host object assigned to interface.
152         """
153         return self._hosts_by_ip4[ip]
154
155     def host_by_ip6(self, ip):
156         """
157         :param ip: IPv6 address to find host by.
158         :return: Host object assigned to interface.
159         """
160         return self._hosts_by_ip6[ip]
161
162     def generate_remote_hosts(self, count=1):
163         """Generate and add remote hosts for the interface.
164
165         :param int count: Number of generated remote hosts.
166         """
167         self._remote_hosts = []
168         self._hosts_by_mac = {}
169         self._hosts_by_ip4 = {}
170         self._hosts_by_ip6 = {}
171         for i in range(
172                 2, count + 2):  # 0: network address, 1: local vpp address
173             mac = "02:%02x:00:00:ff:%02x" % (self.sw_if_index, i)
174             ip4 = "172.16.%u.%u" % (self.sw_if_index, i)
175             ip6 = "fd01:%x::%x" % (self.sw_if_index, i)
176             ip6_ll = mk_ll_addr(mac)
177             host = Host(mac, ip4, ip6, ip6_ll)
178             self._remote_hosts.append(host)
179             self._hosts_by_mac[mac] = host
180             self._hosts_by_ip4[ip4] = host
181             self._hosts_by_ip6[ip6] = host
182
183     @abstractmethod
184     def __init__(self, test):
185         self._test = test
186
187         self._remote_hosts = []
188         self._hosts_by_mac = {}
189         self._hosts_by_ip4 = {}
190         self._hosts_by_ip6 = {}
191
192     def set_mac(self, mac):
193         self._local_mac = mac
194         self._local_ip6_ll = mk_ll_addr(mac)
195         self.test.vapi.sw_interface_set_mac_address(
196             self.sw_if_index,
197             mactobinary(self._local_mac))
198
199     def set_sw_if_index(self, sw_if_index):
200         self._sw_if_index = sw_if_index
201
202         self.generate_remote_hosts()
203
204         self._local_ip4 = "172.16.%u.1" % self.sw_if_index
205         self._local_ip4n = socket.inet_pton(socket.AF_INET, self.local_ip4)
206         self._local_ip4_subnet = "172.16.%u.0" % self.sw_if_index
207         self._local_ip4n_subnet = socket.inet_pton(socket.AF_INET,
208                                                    self._local_ip4_subnet)
209         self._local_ip4_bcast = "172.16.%u.255" % self.sw_if_index
210         self._local_ip4n_bcast = socket.inet_pton(socket.AF_INET,
211                                                   self._local_ip4_bcast)
212         self.local_ip4_prefix_len = 24
213         self.has_ip4_config = False
214         self.ip4_table_id = 0
215
216         self._local_ip6 = "fd01:%x::1" % self.sw_if_index
217         self._local_ip6n = socket.inet_pton(socket.AF_INET6, self.local_ip6)
218         self.local_ip6_prefix_len = 64
219         self.has_ip6_config = False
220         self.ip6_table_id = 0
221
222         self._local_addr = {socket.AF_INET: self.local_ip4,
223                             socket.AF_INET6: self.local_ip6}
224         self._local_addr_n = {socket.AF_INET: self.local_ip4n,
225                               socket.AF_INET6: self.local_ip6n}
226         self._remote_addr = {socket.AF_INET: self.remote_ip4,
227                              socket.AF_INET6: self.remote_ip6}
228         self._remote_addr_n = {socket.AF_INET: self.remote_ip4n,
229                                socket.AF_INET6: self.remote_ip6n}
230
231         r = self.test.vapi.sw_interface_dump()
232         for intf in r:
233             if intf.sw_if_index == self.sw_if_index:
234                 self._name = intf.interface_name.split(b'\0', 1)[0]
235                 self._local_mac = \
236                     ':'.join(binascii.hexlify(intf.l2_address)[i:i + 2]
237                              for i in range(0, 12, 2))
238                 self._dump = intf
239                 break
240         else:
241             raise Exception(
242                 "Could not find interface with sw_if_index %d "
243                 "in interface dump %s" %
244                 (self.sw_if_index, moves.reprlib.repr(r)))
245         self._local_ip6_ll = mk_ll_addr(self.local_mac)
246         self._local_ip6n_ll = socket.inet_pton(socket.AF_INET6,
247                                                self.local_ip6_ll)
248         self._remote_ip6_ll = mk_ll_addr(self.remote_mac)
249         self._remote_ip6n_ll = socket.inet_pton(socket.AF_INET6,
250                                                 self.remote_ip6_ll)
251
252     def config_ip4(self):
253         """Configure IPv4 address on the VPP interface."""
254         self.test.vapi.sw_interface_add_del_address(
255             self.sw_if_index, self.local_ip4n, self.local_ip4_prefix_len)
256         self.has_ip4_config = True
257
258     def unconfig_ip4(self):
259         """Remove IPv4 address on the VPP interface."""
260         try:
261             if self.has_ip4_config:
262                 self.test.vapi.sw_interface_add_del_address(
263                     self.sw_if_index,
264                     self.local_ip4n,
265                     self.local_ip4_prefix_len,
266                     is_add=0)
267         except AttributeError:
268             self.has_ip4_config = False
269         self.has_ip4_config = False
270
271     def configure_ipv4_neighbors(self):
272         """For every remote host assign neighbor's MAC to IPv4 addresses.
273
274         :param vrf_id: The FIB table / VRF ID. (Default value = 0)
275         """
276         for host in self._remote_hosts:
277             macn = host.mac.replace(":", "").decode('hex')
278             ipn = host.ip4n
279             self.test.vapi.ip_neighbor_add_del(
280                 self.sw_if_index, macn, ipn)
281
282     def config_ip6(self):
283         """Configure IPv6 address on the VPP interface."""
284         self.test.vapi.sw_interface_add_del_address(
285             self.sw_if_index, self._local_ip6n, self.local_ip6_prefix_len,
286             is_ipv6=1)
287         self.has_ip6_config = True
288
289     def unconfig_ip6(self):
290         """Remove IPv6 address on the VPP interface."""
291         try:
292             if self.has_ip6_config:
293                 self.test.vapi.sw_interface_add_del_address(
294                     self.sw_if_index,
295                     self.local_ip6n,
296                     self.local_ip6_prefix_len,
297                     is_ipv6=1, is_add=0)
298         except AttributeError:
299             self.has_ip6_config = False
300         self.has_ip6_config = False
301
302     def configure_ipv6_neighbors(self):
303         """For every remote host assign neighbor's MAC to IPv6 addresses.
304
305         :param vrf_id: The FIB table / VRF ID. (Default value = 0)
306         """
307         for host in self._remote_hosts:
308             macn = host.mac.replace(":", "").decode('hex')
309             ipn = host.ip6n
310             self.test.vapi.ip_neighbor_add_del(
311                 self.sw_if_index, macn, ipn, is_ipv6=1)
312
313     def unconfig(self):
314         """Unconfigure IPv6 and IPv4 address on the VPP interface."""
315         self.unconfig_ip4()
316         self.unconfig_ip6()
317
318     def set_table_ip4(self, table_id):
319         """Set the interface in a IPv4 Table.
320
321         .. note:: Must be called before configuring IP4 addresses.
322         """
323         self.ip4_table_id = table_id
324         self.test.vapi.sw_interface_set_table(
325             self.sw_if_index, 0, self.ip4_table_id)
326
327     def set_table_ip6(self, table_id):
328         """Set the interface in a IPv6 Table.
329
330         .. note:: Must be called before configuring IP6 addresses.
331         """
332         self.ip6_table_id = table_id
333         self.test.vapi.sw_interface_set_table(
334             self.sw_if_index, 1, self.ip6_table_id)
335
336     def disable_ipv6_ra(self):
337         """Configure IPv6 RA suppress on the VPP interface."""
338         self.test.vapi.sw_interface_ra_suppress(self.sw_if_index)
339
340     def ip6_ra_config(self, no=0, suppress=0, send_unicast=0):
341         """Configure IPv6 RA suppress on the VPP interface."""
342         self.test.vapi.ip6_sw_interface_ra_config(self.sw_if_index,
343                                                   no,
344                                                   suppress,
345                                                   send_unicast)
346
347     def ip6_ra_prefix(self, address, address_length, is_no=0,
348                       off_link=0, no_autoconfig=0, use_default=0):
349         """Configure IPv6 RA suppress on the VPP interface."""
350         self.test.vapi.ip6_sw_interface_ra_prefix(self.sw_if_index,
351                                                   address,
352                                                   address_length,
353                                                   is_no=is_no,
354                                                   off_link=off_link,
355                                                   no_autoconfig=no_autoconfig,
356                                                   use_default=use_default)
357
358     def admin_up(self):
359         """Put interface ADMIN-UP."""
360         self.test.vapi.sw_interface_set_flags(self.sw_if_index,
361                                               admin_up_down=1)
362
363     def admin_down(self):
364         """Put interface ADMIN-down."""
365         self.test.vapi.sw_interface_set_flags(self.sw_if_index,
366                                               admin_up_down=0)
367
368     def ip6_enable(self):
369         """IPv6 Enable interface"""
370         self.test.vapi.ip6_sw_interface_enable_disable(self.sw_if_index,
371                                                        enable=1)
372
373     def ip6_disable(self):
374         """Put interface ADMIN-DOWN."""
375         self.test.vapi.ip6_sw_interface_enable_disable(self.sw_if_index,
376                                                        enable=0)
377
378     def add_sub_if(self, sub_if):
379         """Register a sub-interface with this interface.
380
381         :param sub_if: sub-interface
382         """
383         if not hasattr(self, 'sub_if'):
384             self.sub_if = sub_if
385         else:
386             if isinstance(self.sub_if, list):
387                 self.sub_if.append(sub_if)
388             else:
389                 self.sub_if = sub_if
390
391     def enable_mpls(self):
392         """Enable MPLS on the VPP interface."""
393         self.test.vapi.sw_interface_enable_disable_mpls(
394             self.sw_if_index)
395
396     def disable_mpls(self):
397         """Enable MPLS on the VPP interface."""
398         self.test.vapi.sw_interface_enable_disable_mpls(
399             self.sw_if_index, 0)
400
401     def is_ip4_entry_in_fib_dump(self, dump):
402         for i in dump:
403             if i.address == self.local_ip4n and \
404                i.address_length == self.local_ip4_prefix_len and \
405                i.table_id == self.ip4_table_id:
406                 return True
407         return False
408
409     def set_unnumbered(self, ip_sw_if_index):
410         """ Set the interface to unnumbered via ip_sw_if_index """
411         self.test.vapi.sw_interface_set_unnumbered(
412             self.sw_if_index,
413             ip_sw_if_index)
414
415     def unset_unnumbered(self, ip_sw_if_index):
416         """ Unset the interface to unnumbered via ip_sw_if_index """
417         self.test.vapi.sw_interface_set_unnumbered(
418             self.sw_if_index,
419             ip_sw_if_index,
420             is_add=0)
421
422     def set_proxy_arp(self, enable=1):
423         """ Set the interface to enable/disable Proxy ARP """
424         self.test.vapi.proxy_arp_intfc_enable_disable(
425             self.sw_if_index,
426             enable)
427
428     def query_vpp_config(self):
429         dump = self.test.vapi.sw_interface_dump()
430         return self.is_interface_config_in_dump(dump)
431
432     def get_interface_config_from_dump(self, dump):
433         for i in dump:
434             if i.interface_name.rstrip(' \t\r\n\0') == self.name and \
435                i.sw_if_index == self.sw_if_index:
436                 return i
437         else:
438             return None
439
440     def is_interface_config_in_dump(self, dump):
441         return self.get_interface_config_from_dump(dump) is not None
442
443     def assert_interface_state(self, admin_up_down, link_up_down,
444                                expect_event=False):
445         if expect_event:
446             event = self.test.vapi.wait_for_event(timeout=1,
447                                                   name='sw_interface_event')
448             self.test.assert_equal(event.sw_if_index, self.sw_if_index,
449                                    "sw_if_index")
450             self.test.assert_equal(event.admin_up_down, admin_up_down,
451                                    "admin state")
452             self.test.assert_equal(event.link_up_down, link_up_down,
453                                    "link state")
454         dump = self.test.vapi.sw_interface_dump()
455         if_state = self.get_interface_config_from_dump(dump)
456         self.test.assert_equal(if_state.admin_up_down, admin_up_down,
457                                "admin state")
458         self.test.assert_equal(if_state.link_up_down, link_up_down,
459                                "link state")
460
461     def __str__(self):
462         return self.name