Paul Vinciguerra | 6e4c6ad | 2018-11-25 10:35:29 -0800 | [diff] [blame] | 1 | import binascii |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 2 | import socket |
Paul Vinciguerra | 3bce8eb | 2018-11-24 21:46:05 -0800 | [diff] [blame] | 3 | import abc |
Paul Vinciguerra | 00671cf | 2018-11-25 12:47:04 -0800 | [diff] [blame] | 4 | |
Paul Vinciguerra | 3bce8eb | 2018-11-24 21:46:05 -0800 | [diff] [blame] | 5 | import six |
Paul Vinciguerra | 6c74617 | 2018-11-26 09:57:21 -0800 | [diff] [blame] | 6 | from six import moves |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 7 | |
Ole Troan | 7f99183 | 2018-12-06 17:35:12 +0100 | [diff] [blame] | 8 | from util import Host, mk_ll_addr |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 9 | from vpp_papi import mac_ntop, VppEnum |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 10 | from ipaddress import IPv4Network, IPv6Network |
Neale Ranns | 097fa66 | 2018-05-01 05:17:55 -0700 | [diff] [blame] | 11 | |
| 12 | try: |
| 13 | text_type = unicode |
| 14 | except NameError: |
| 15 | text_type = str |
Jan | 65209ed | 2016-12-05 23:29:17 +0100 | [diff] [blame] | 16 | |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 17 | |
Paul Vinciguerra | 3bce8eb | 2018-11-24 21:46:05 -0800 | [diff] [blame] | 18 | @six.add_metaclass(abc.ABCMeta) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 19 | class VppInterface(object): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 20 | """Generic VPP interface.""" |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 21 | |
| 22 | @property |
| 23 | def sw_if_index(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 24 | """Interface index assigned by VPP.""" |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 25 | return self._sw_if_index |
| 26 | |
| 27 | @property |
| 28 | def remote_mac(self): |
Klement Sekera | da505f6 | 2017-01-04 12:58:53 +0100 | [diff] [blame] | 29 | """MAC-address of the remote interface "connected" to this interface""" |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 30 | return self._remote_hosts[0].mac |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 31 | |
| 32 | @property |
| 33 | def local_mac(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 34 | """MAC-address of the VPP interface.""" |
Ole Troan | 6ed154f | 2019-10-15 19:31:55 +0200 | [diff] [blame] | 35 | return str(self._local_mac) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 36 | |
| 37 | @property |
Klement Sekera | 611864f | 2018-09-26 11:19:00 +0200 | [diff] [blame] | 38 | def local_addr(self): |
| 39 | return self._local_addr |
| 40 | |
| 41 | @property |
| 42 | def remote_addr(self): |
| 43 | return self._remote_addr |
| 44 | |
| 45 | @property |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 46 | def local_ip4(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 47 | """Local IPv4 address on VPP interface (string).""" |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 48 | return self._local_ip4 |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 49 | |
| 50 | @local_ip4.setter |
| 51 | def local_ip4(self, value): |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 52 | self._local_ip4 = value |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 53 | |
| 54 | @property |
| 55 | def local_ip4_prefix_len(self): |
| 56 | """Local IPv4 prefix length """ |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 57 | return self._local_ip4_len |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 58 | |
| 59 | @local_ip4_prefix_len.setter |
| 60 | def local_ip4_prefix_len(self, value): |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 61 | self._local_ip4_len = value |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 62 | |
| 63 | @property |
| 64 | def local_ip4_prefix(self): |
| 65 | """Local IPv4 prefix """ |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 66 | return ("%s/%d" % (self._local_ip4, self._local_ip4_len)) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 67 | |
| 68 | @property |
| 69 | def local_ip4n(self): |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 70 | """DEPRECATED """ |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 71 | """Local IPv4 address - raw, suitable as API parameter.""" |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 72 | return socket.inet_pton(socket.AF_INET, self._local_ip4) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 73 | |
| 74 | @property |
| 75 | def remote_ip4(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 76 | """IPv4 address of remote peer "connected" to this interface.""" |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 77 | return self._remote_hosts[0].ip4 |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 78 | |
| 79 | @property |
| 80 | def remote_ip4n(self): |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 81 | """DEPRECATED """ |
| 82 | """Local IPv6 address - raw, suitable as API parameter.""" |
| 83 | return socket.inet_pton(socket.AF_INET, self._remote_hosts[0].ip4) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 84 | |
| 85 | @property |
| 86 | def local_ip6(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 87 | """Local IPv6 address on VPP interface (string).""" |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 88 | return self._local_ip6 |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 89 | |
| 90 | @local_ip6.setter |
| 91 | def local_ip6(self, value): |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 92 | self._local_ip6 |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 93 | |
| 94 | @property |
| 95 | def local_ip6_prefix_len(self): |
| 96 | """Local IPv6 prefix length """ |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 97 | return self._local_ip6_len |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 98 | |
| 99 | @local_ip6_prefix_len.setter |
| 100 | def local_ip6_prefix_len(self, value): |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 101 | self._local_ip6_len = value |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 102 | |
| 103 | @property |
| 104 | def local_ip6_prefix(self): |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 105 | """Local IPv4 prefix """ |
| 106 | return ("%s/%d" % (self._local_ip6, self._local_ip6_len)) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 107 | |
| 108 | @property |
| 109 | def local_ip6n(self): |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 110 | """DEPRECATED """ |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 111 | """Local IPv6 address - raw, suitable as API parameter.""" |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 112 | return socket.inet_pton(socket.AF_INET6, self._local_ip6) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 113 | |
| 114 | @property |
| 115 | def remote_ip6(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 116 | """IPv6 address of remote peer "connected" to this interface.""" |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 117 | return self._remote_hosts[0].ip6 |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 118 | |
| 119 | @property |
| 120 | def remote_ip6n(self): |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 121 | """DEPRECATED """ |
| 122 | """Local IPv6 address - raw, suitable as API parameter.""" |
| 123 | return socket.inet_pton(socket.AF_INET6, self._remote_hosts[0].ip6) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 124 | |
| 125 | @property |
Juraj Sloboda | c037423 | 2018-02-01 15:18:49 +0100 | [diff] [blame] | 126 | def local_ip6_ll(self): |
Klement Sekera | 611864f | 2018-09-26 11:19:00 +0200 | [diff] [blame] | 127 | """Local IPv6 link-local address on VPP interface (string).""" |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 128 | return self._local_ip6_ll |
Juraj Sloboda | c037423 | 2018-02-01 15:18:49 +0100 | [diff] [blame] | 129 | |
| 130 | @property |
| 131 | def local_ip6n_ll(self): |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 132 | """DEPRECATED """ |
| 133 | """Local IPv6 link-local address on VPP interface (string).""" |
| 134 | return socket.inet_pton(socket.AF_INET6, self._local_ip6_ll.address) |
Juraj Sloboda | c037423 | 2018-02-01 15:18:49 +0100 | [diff] [blame] | 135 | |
| 136 | @property |
| 137 | def remote_ip6_ll(self): |
| 138 | """Link-local IPv6 address of remote peer |
| 139 | "connected" to this interface.""" |
| 140 | return self._remote_ip6_ll |
| 141 | |
| 142 | @property |
| 143 | def remote_ip6n_ll(self): |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 144 | """DEPRECATED """ |
| 145 | """Local IPv6 link-local address on VPP interface (string).""" |
| 146 | return socket.inet_pton(socket.AF_INET6, self._remote_ip6_ll) |
Juraj Sloboda | c037423 | 2018-02-01 15:18:49 +0100 | [diff] [blame] | 147 | |
| 148 | @property |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 149 | def name(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 150 | """Name of the interface.""" |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 151 | return self._name |
| 152 | |
| 153 | @property |
| 154 | def dump(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 155 | """RAW result of sw_interface_dump for this interface.""" |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 156 | return self._dump |
| 157 | |
| 158 | @property |
| 159 | def test(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 160 | """Test case creating this interface.""" |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 161 | return self._test |
| 162 | |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 163 | @property |
| 164 | def remote_hosts(self): |
| 165 | """Remote hosts list""" |
| 166 | return self._remote_hosts |
| 167 | |
| 168 | @remote_hosts.setter |
| 169 | def remote_hosts(self, value): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 170 | """ |
| 171 | :param list value: List of remote hosts. |
| 172 | """ |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 173 | self._remote_hosts = value |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 174 | self._hosts_by_mac = {} |
| 175 | self._hosts_by_ip4 = {} |
| 176 | self._hosts_by_ip6 = {} |
| 177 | for host in self._remote_hosts: |
| 178 | self._hosts_by_mac[host.mac] = host |
| 179 | self._hosts_by_ip4[host.ip4] = host |
| 180 | self._hosts_by_ip6[host.ip6] = host |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 181 | |
| 182 | def host_by_mac(self, mac): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 183 | """ |
Matej Klotton | c5bf07f | 2016-11-23 15:27:17 +0100 | [diff] [blame] | 184 | :param mac: MAC address to find host by. |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 185 | :return: Host object assigned to interface. |
| 186 | """ |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 187 | return self._hosts_by_mac[mac] |
| 188 | |
| 189 | def host_by_ip4(self, ip): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 190 | """ |
| 191 | :param ip: IPv4 address to find host by. |
| 192 | :return: Host object assigned to interface. |
| 193 | """ |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 194 | return self._hosts_by_ip4[ip] |
| 195 | |
| 196 | def host_by_ip6(self, ip): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 197 | """ |
| 198 | :param ip: IPv6 address to find host by. |
| 199 | :return: Host object assigned to interface. |
| 200 | """ |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 201 | return self._hosts_by_ip6[ip] |
| 202 | |
| 203 | def generate_remote_hosts(self, count=1): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 204 | """Generate and add remote hosts for the interface. |
| 205 | |
| 206 | :param int count: Number of generated remote hosts. |
| 207 | """ |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 208 | self._remote_hosts = [] |
| 209 | self._hosts_by_mac = {} |
| 210 | self._hosts_by_ip4 = {} |
| 211 | self._hosts_by_ip6 = {} |
Klement Sekera | 7bb873a | 2016-11-18 07:38:42 +0100 | [diff] [blame] | 212 | for i in range( |
| 213 | 2, count + 2): # 0: network address, 1: local vpp address |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 214 | mac = "02:%02x:00:00:ff:%02x" % (self.sw_if_index, i) |
| 215 | ip4 = "172.16.%u.%u" % (self.sw_if_index, i) |
Klement Sekera | 46a87ad | 2017-01-02 08:22:23 +0100 | [diff] [blame] | 216 | ip6 = "fd01:%x::%x" % (self.sw_if_index, i) |
Neale Ranns | 2a3ea49 | 2017-04-19 05:24:40 -0700 | [diff] [blame] | 217 | ip6_ll = mk_ll_addr(mac) |
| 218 | host = Host(mac, ip4, ip6, ip6_ll) |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 219 | self._remote_hosts.append(host) |
| 220 | self._hosts_by_mac[mac] = host |
| 221 | self._hosts_by_ip4[ip4] = host |
| 222 | self._hosts_by_ip6[ip6] = host |
| 223 | |
Paul Vinciguerra | 3bce8eb | 2018-11-24 21:46:05 -0800 | [diff] [blame] | 224 | @abc.abstractmethod |
Matej Klotton | c5bf07f | 2016-11-23 15:27:17 +0100 | [diff] [blame] | 225 | def __init__(self, test): |
| 226 | self._test = test |
| 227 | |
| 228 | self._remote_hosts = [] |
| 229 | self._hosts_by_mac = {} |
| 230 | self._hosts_by_ip4 = {} |
| 231 | self._hosts_by_ip6 = {} |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 232 | |
Neale Ranns | 3b81a1e | 2018-09-06 09:50:26 -0700 | [diff] [blame] | 233 | def set_mac(self, mac): |
Ole Troan | 8006c6a | 2018-12-17 12:02:26 +0100 | [diff] [blame] | 234 | self._local_mac = str(mac) |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 235 | self._local_ip6_ll = mk_ll_addr(self._local_mac) |
Neale Ranns | 3b81a1e | 2018-09-06 09:50:26 -0700 | [diff] [blame] | 236 | self.test.vapi.sw_interface_set_mac_address( |
Ole Troan | 8006c6a | 2018-12-17 12:02:26 +0100 | [diff] [blame] | 237 | self.sw_if_index, mac.packed) |
Neale Ranns | 3b81a1e | 2018-09-06 09:50:26 -0700 | [diff] [blame] | 238 | |
Klement Sekera | 31da2e3 | 2018-06-24 22:49:55 +0200 | [diff] [blame] | 239 | def set_sw_if_index(self, sw_if_index): |
| 240 | self._sw_if_index = sw_if_index |
| 241 | |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 242 | self.generate_remote_hosts() |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 243 | |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 244 | self._local_ip4 = "172.16.%u.1" % self.sw_if_index |
| 245 | self._local_ip4_len = 24 |
Neale Ranns | 24b170a | 2017-08-15 05:33:11 -0700 | [diff] [blame] | 246 | self._local_ip4_subnet = "172.16.%u.0" % self.sw_if_index |
Neale Ranns | 24b170a | 2017-08-15 05:33:11 -0700 | [diff] [blame] | 247 | self._local_ip4_bcast = "172.16.%u.255" % self.sw_if_index |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 248 | self.has_ip4_config = False |
| 249 | self.ip4_table_id = 0 |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 250 | |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 251 | self._local_ip6 = "fd01:%x::1" % self.sw_if_index |
| 252 | self._local_ip6_len = 64 |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 253 | self.has_ip6_config = False |
| 254 | self.ip6_table_id = 0 |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 255 | |
Klement Sekera | 611864f | 2018-09-26 11:19:00 +0200 | [diff] [blame] | 256 | self._local_addr = {socket.AF_INET: self.local_ip4, |
| 257 | socket.AF_INET6: self.local_ip6} |
Klement Sekera | 611864f | 2018-09-26 11:19:00 +0200 | [diff] [blame] | 258 | self._remote_addr = {socket.AF_INET: self.remote_ip4, |
| 259 | socket.AF_INET6: self.remote_ip6} |
Klement Sekera | 611864f | 2018-09-26 11:19:00 +0200 | [diff] [blame] | 260 | |
Paul Vinciguerra | 7a99823 | 2019-06-07 15:01:12 -0400 | [diff] [blame] | 261 | r = self.test.vapi.sw_interface_dump(sw_if_index=self.sw_if_index) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 262 | for intf in r: |
| 263 | if intf.sw_if_index == self.sw_if_index: |
Ole Troan | 6ed154f | 2019-10-15 19:31:55 +0200 | [diff] [blame] | 264 | self._name = intf.interface_name |
| 265 | self._local_mac = intf.l2_address |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 266 | self._dump = intf |
| 267 | break |
| 268 | else: |
| 269 | raise Exception( |
| 270 | "Could not find interface with sw_if_index %d " |
| 271 | "in interface dump %s" % |
Paul Vinciguerra | 6c74617 | 2018-11-26 09:57:21 -0800 | [diff] [blame] | 272 | (self.sw_if_index, moves.reprlib.repr(r))) |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 273 | self._local_ip6_ll = mk_ll_addr(self.local_mac) |
Juraj Sloboda | c037423 | 2018-02-01 15:18:49 +0100 | [diff] [blame] | 274 | self._remote_ip6_ll = mk_ll_addr(self.remote_mac) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 275 | |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 276 | def config_ip4(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 277 | """Configure IPv4 address on the VPP interface.""" |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 278 | self.test.vapi.sw_interface_add_del_address( |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 279 | sw_if_index=self.sw_if_index, prefix=self.local_ip4_prefix) |
Neale Ranns | 177bbdc | 2016-11-15 09:46:51 +0000 | [diff] [blame] | 280 | self.has_ip4_config = True |
| 281 | |
| 282 | def unconfig_ip4(self): |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 283 | """Remove IPv4 address on the VPP interface.""" |
Neale Ranns | 177bbdc | 2016-11-15 09:46:51 +0000 | [diff] [blame] | 284 | try: |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 285 | if self.has_ip4_config: |
Neale Ranns | 177bbdc | 2016-11-15 09:46:51 +0000 | [diff] [blame] | 286 | self.test.vapi.sw_interface_add_del_address( |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 287 | sw_if_index=self.sw_if_index, |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 288 | prefix=self.local_ip4_prefix, is_add=0) |
Neale Ranns | 177bbdc | 2016-11-15 09:46:51 +0000 | [diff] [blame] | 289 | except AttributeError: |
| 290 | self.has_ip4_config = False |
| 291 | self.has_ip4_config = False |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 292 | |
Neale Ranns | baf2e90 | 2017-02-25 04:20:00 -0800 | [diff] [blame] | 293 | def configure_ipv4_neighbors(self): |
Jan | e546d3b | 2016-12-08 13:10:03 +0100 | [diff] [blame] | 294 | """For every remote host assign neighbor's MAC to IPv4 addresses. |
| 295 | |
| 296 | :param vrf_id: The FIB table / VRF ID. (Default value = 0) |
| 297 | """ |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 298 | for host in self._remote_hosts: |
Neale Ranns | 3702930 | 2018-08-10 05:30:06 -0700 | [diff] [blame] | 299 | self.test.vapi.ip_neighbor_add_del(self.sw_if_index, |
| 300 | host.mac, |
| 301 | host.ip4) |
Matej Klotton | 0178d52 | 2016-11-04 11:11:44 +0100 | [diff] [blame] | 302 | |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 303 | def config_ip6(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 304 | """Configure IPv6 address on the VPP interface.""" |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 305 | self.test.vapi.sw_interface_add_del_address( |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 306 | sw_if_index=self.sw_if_index, prefix=self.local_ip6_prefix) |
Neale Ranns | 177bbdc | 2016-11-15 09:46:51 +0000 | [diff] [blame] | 307 | self.has_ip6_config = True |
| 308 | |
| 309 | def unconfig_ip6(self): |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 310 | """Remove IPv6 address on the VPP interface.""" |
Neale Ranns | 177bbdc | 2016-11-15 09:46:51 +0000 | [diff] [blame] | 311 | try: |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 312 | if self.has_ip6_config: |
Neale Ranns | 177bbdc | 2016-11-15 09:46:51 +0000 | [diff] [blame] | 313 | self.test.vapi.sw_interface_add_del_address( |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 314 | sw_if_index=self.sw_if_index, |
Neale Ranns | efd7bc2 | 2019-11-11 08:32:34 +0000 | [diff] [blame] | 315 | prefix=self.local_ip6_prefix, is_add=0) |
Neale Ranns | 177bbdc | 2016-11-15 09:46:51 +0000 | [diff] [blame] | 316 | except AttributeError: |
| 317 | self.has_ip6_config = False |
| 318 | self.has_ip6_config = False |
| 319 | |
Neale Ranns | baf2e90 | 2017-02-25 04:20:00 -0800 | [diff] [blame] | 320 | def configure_ipv6_neighbors(self): |
Jan Gelety | 057bb8c | 2016-12-20 17:32:45 +0100 | [diff] [blame] | 321 | """For every remote host assign neighbor's MAC to IPv6 addresses. |
| 322 | |
| 323 | :param vrf_id: The FIB table / VRF ID. (Default value = 0) |
| 324 | """ |
Klement Sekera | 46a87ad | 2017-01-02 08:22:23 +0100 | [diff] [blame] | 325 | for host in self._remote_hosts: |
Neale Ranns | 3702930 | 2018-08-10 05:30:06 -0700 | [diff] [blame] | 326 | self.test.vapi.ip_neighbor_add_del(self.sw_if_index, |
| 327 | host.mac, |
| 328 | host.ip6) |
Klement Sekera | 46a87ad | 2017-01-02 08:22:23 +0100 | [diff] [blame] | 329 | |
Neale Ranns | 177bbdc | 2016-11-15 09:46:51 +0000 | [diff] [blame] | 330 | def unconfig(self): |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 331 | """Unconfigure IPv6 and IPv4 address on the VPP interface.""" |
Neale Ranns | 177bbdc | 2016-11-15 09:46:51 +0000 | [diff] [blame] | 332 | self.unconfig_ip4() |
| 333 | self.unconfig_ip6() |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 334 | |
Neale Ranns | 8fe8cc2 | 2016-11-01 10:05:08 +0000 | [diff] [blame] | 335 | def set_table_ip4(self, table_id): |
| 336 | """Set the interface in a IPv4 Table. |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 337 | |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 338 | .. note:: Must be called before configuring IP4 addresses. |
| 339 | """ |
| 340 | self.ip4_table_id = table_id |
Neale Ranns | 8fe8cc2 | 2016-11-01 10:05:08 +0000 | [diff] [blame] | 341 | self.test.vapi.sw_interface_set_table( |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 342 | self.sw_if_index, 0, self.ip4_table_id) |
Neale Ranns | 8fe8cc2 | 2016-11-01 10:05:08 +0000 | [diff] [blame] | 343 | |
| 344 | def set_table_ip6(self, table_id): |
| 345 | """Set the interface in a IPv6 Table. |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 346 | |
| 347 | .. note:: Must be called before configuring IP6 addresses. |
| 348 | """ |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 349 | self.ip6_table_id = table_id |
Neale Ranns | 8fe8cc2 | 2016-11-01 10:05:08 +0000 | [diff] [blame] | 350 | self.test.vapi.sw_interface_set_table( |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 351 | self.sw_if_index, 1, self.ip6_table_id) |
Neale Ranns | 8fe8cc2 | 2016-11-01 10:05:08 +0000 | [diff] [blame] | 352 | |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 353 | def disable_ipv6_ra(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 354 | """Configure IPv6 RA suppress on the VPP interface.""" |
Ole Troan | e1ade68 | 2019-03-04 23:55:43 +0100 | [diff] [blame] | 355 | self.test.vapi.sw_interface_ip6nd_ra_config( |
| 356 | sw_if_index=self.sw_if_index, |
| 357 | suppress=1) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 358 | |
Neale Ranns | 32e1c01 | 2016-11-22 17:07:28 +0000 | [diff] [blame] | 359 | def ip6_ra_config(self, no=0, suppress=0, send_unicast=0): |
Neale Ranns | 7515228 | 2017-01-09 01:00:45 -0800 | [diff] [blame] | 360 | """Configure IPv6 RA suppress on the VPP interface.""" |
Ole Troan | e1ade68 | 2019-03-04 23:55:43 +0100 | [diff] [blame] | 361 | self.test.vapi.sw_interface_ip6nd_ra_config( |
| 362 | sw_if_index=self.sw_if_index, |
| 363 | is_no=no, |
| 364 | suppress=suppress, |
| 365 | send_unicast=send_unicast) |
Neale Ranns | 7515228 | 2017-01-09 01:00:45 -0800 | [diff] [blame] | 366 | |
Paul Vinciguerra | ab05508 | 2019-06-06 14:07:55 -0400 | [diff] [blame] | 367 | def ip6_ra_prefix(self, prefix, is_no=0, |
Neale Ranns | 87df12d | 2017-02-18 08:16:41 -0800 | [diff] [blame] | 368 | off_link=0, no_autoconfig=0, use_default=0): |
Paul Vinciguerra | ab05508 | 2019-06-06 14:07:55 -0400 | [diff] [blame] | 369 | """Configure IPv6 RA suppress on the VPP interface. |
| 370 | |
| 371 | prefix can be a string in the format of '<address>/<length_in_bits>' |
| 372 | or ipaddress.ipnetwork object (if strict.)""" |
| 373 | |
Ole Troan | e1ade68 | 2019-03-04 23:55:43 +0100 | [diff] [blame] | 374 | self.test.vapi.sw_interface_ip6nd_ra_prefix( |
Ole Troan | a5b2eec | 2019-03-11 19:23:25 +0100 | [diff] [blame] | 375 | sw_if_index=self.sw_if_index, |
Paul Vinciguerra | ab05508 | 2019-06-06 14:07:55 -0400 | [diff] [blame] | 376 | prefix=prefix, |
Ole Troan | a5b2eec | 2019-03-11 19:23:25 +0100 | [diff] [blame] | 377 | use_default=use_default, |
| 378 | off_link=off_link, no_autoconfig=no_autoconfig, |
| 379 | is_no=is_no) |
Neale Ranns | 87df12d | 2017-02-18 08:16:41 -0800 | [diff] [blame] | 380 | |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 381 | def admin_up(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 382 | """Put interface ADMIN-UP.""" |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 383 | self.test.vapi.sw_interface_set_flags( |
| 384 | self.sw_if_index, |
| 385 | flags=VppEnum.vl_api_if_status_flags_t.IF_STATUS_API_FLAG_ADMIN_UP) |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 386 | |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 387 | def admin_down(self): |
| 388 | """Put interface ADMIN-down.""" |
Klement Sekera | da505f6 | 2017-01-04 12:58:53 +0100 | [diff] [blame] | 389 | self.test.vapi.sw_interface_set_flags(self.sw_if_index, |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 390 | flags=0) |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 391 | |
Neale Ranns | 6348074 | 2019-03-13 06:41:52 -0700 | [diff] [blame] | 392 | def link_up(self): |
| 393 | """Put interface link-state-UP.""" |
| 394 | self.test.vapi.cli("test interface link-state %s up" % self.name) |
| 395 | |
| 396 | def link_down(self): |
| 397 | """Put interface link-state-down.""" |
| 398 | self.test.vapi.cli("test interface link-state %s down" % self.name) |
| 399 | |
Neale Ranns | 7515228 | 2017-01-09 01:00:45 -0800 | [diff] [blame] | 400 | def ip6_enable(self): |
| 401 | """IPv6 Enable interface""" |
Ole Troan | e1ade68 | 2019-03-04 23:55:43 +0100 | [diff] [blame] | 402 | self.test.vapi.sw_interface_ip6_enable_disable(self.sw_if_index, |
Klement Sekera | da505f6 | 2017-01-04 12:58:53 +0100 | [diff] [blame] | 403 | enable=1) |
Neale Ranns | 7515228 | 2017-01-09 01:00:45 -0800 | [diff] [blame] | 404 | |
| 405 | def ip6_disable(self): |
| 406 | """Put interface ADMIN-DOWN.""" |
Ole Troan | e1ade68 | 2019-03-04 23:55:43 +0100 | [diff] [blame] | 407 | self.test.vapi.sw_interface_ip6_enable_disable(self.sw_if_index, |
Klement Sekera | da505f6 | 2017-01-04 12:58:53 +0100 | [diff] [blame] | 408 | enable=0) |
Neale Ranns | 7515228 | 2017-01-09 01:00:45 -0800 | [diff] [blame] | 409 | |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 410 | def add_sub_if(self, sub_if): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 411 | """Register a sub-interface with this interface. |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 412 | |
| 413 | :param sub_if: sub-interface |
Klement Sekera | f62ae12 | 2016-10-11 11:47:09 +0200 | [diff] [blame] | 414 | """ |
| 415 | if not hasattr(self, 'sub_if'): |
| 416 | self.sub_if = sub_if |
| 417 | else: |
| 418 | if isinstance(self.sub_if, list): |
| 419 | self.sub_if.append(sub_if) |
| 420 | else: |
| 421 | self.sub_if = sub_if |
Neale Ranns | 8fe8cc2 | 2016-11-01 10:05:08 +0000 | [diff] [blame] | 422 | |
| 423 | def enable_mpls(self): |
Matej Klotton | 86d87c4 | 2016-11-11 11:38:55 +0100 | [diff] [blame] | 424 | """Enable MPLS on the VPP interface.""" |
Ole Troan | 9a47537 | 2019-03-05 16:58:24 +0100 | [diff] [blame] | 425 | self.test.vapi.sw_interface_set_mpls_enable(self.sw_if_index) |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 426 | |
Neale Ranns | 180279b | 2017-03-16 15:49:09 -0400 | [diff] [blame] | 427 | def disable_mpls(self): |
| 428 | """Enable MPLS on the VPP interface.""" |
Ole Troan | 9a47537 | 2019-03-05 16:58:24 +0100 | [diff] [blame] | 429 | self.test.vapi.sw_interface_set_mpls_enable(self.sw_if_index, 0) |
Neale Ranns | 180279b | 2017-03-16 15:49:09 -0400 | [diff] [blame] | 430 | |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 431 | def is_ip4_entry_in_fib_dump(self, dump): |
| 432 | for i in dump: |
Neale Ranns | 097fa66 | 2018-05-01 05:17:55 -0700 | [diff] [blame] | 433 | n = IPv4Network(text_type("%s/%d" % (self.local_ip4, |
| 434 | self.local_ip4_prefix_len))) |
| 435 | if i.route.prefix == n and \ |
| 436 | i.route.table_id == self.ip4_table_id: |
Matej Klotton | 8d8a1da | 2016-12-22 11:06:56 +0100 | [diff] [blame] | 437 | return True |
| 438 | return False |
Neale Ranns | 39f9d8b | 2017-02-16 21:57:05 -0800 | [diff] [blame] | 439 | |
| 440 | def set_unnumbered(self, ip_sw_if_index): |
| 441 | """ Set the interface to unnumbered via ip_sw_if_index """ |
Ole Troan | 9a47537 | 2019-03-05 16:58:24 +0100 | [diff] [blame] | 442 | self.test.vapi.sw_interface_set_unnumbered(ip_sw_if_index, |
| 443 | self.sw_if_index) |
Neale Ranns | 39f9d8b | 2017-02-16 21:57:05 -0800 | [diff] [blame] | 444 | |
Neale Ranns | 37be736 | 2017-02-21 17:30:26 -0800 | [diff] [blame] | 445 | def unset_unnumbered(self, ip_sw_if_index): |
Neale Ranns | 4b919a5 | 2017-03-11 05:55:21 -0800 | [diff] [blame] | 446 | """ Unset the interface to unnumbered via ip_sw_if_index """ |
Ole Troan | 9a47537 | 2019-03-05 16:58:24 +0100 | [diff] [blame] | 447 | self.test.vapi.sw_interface_set_unnumbered(ip_sw_if_index, |
| 448 | self.sw_if_index, is_add=0) |
Neale Ranns | 37be736 | 2017-02-21 17:30:26 -0800 | [diff] [blame] | 449 | |
Neale Ranns | 39f9d8b | 2017-02-16 21:57:05 -0800 | [diff] [blame] | 450 | def set_proxy_arp(self, enable=1): |
| 451 | """ Set the interface to enable/disable Proxy ARP """ |
| 452 | self.test.vapi.proxy_arp_intfc_enable_disable( |
| 453 | self.sw_if_index, |
| 454 | enable) |
Klement Sekera | 75e7d13 | 2017-09-20 08:26:30 +0200 | [diff] [blame] | 455 | |
| 456 | def query_vpp_config(self): |
Paul Vinciguerra | 7a99823 | 2019-06-07 15:01:12 -0400 | [diff] [blame] | 457 | dump = self.test.vapi.sw_interface_dump(sw_if_index=self.sw_if_index) |
Klement Sekera | 75e7d13 | 2017-09-20 08:26:30 +0200 | [diff] [blame] | 458 | return self.is_interface_config_in_dump(dump) |
| 459 | |
Juraj Sloboda | b3f9050 | 2018-10-04 15:15:16 +0200 | [diff] [blame] | 460 | def get_interface_config_from_dump(self, dump): |
Klement Sekera | 75e7d13 | 2017-09-20 08:26:30 +0200 | [diff] [blame] | 461 | for i in dump: |
| 462 | if i.interface_name.rstrip(' \t\r\n\0') == self.name and \ |
Ole Troan | 9a47537 | 2019-03-05 16:58:24 +0100 | [diff] [blame] | 463 | i.sw_if_index == self.sw_if_index: |
Juraj Sloboda | b3f9050 | 2018-10-04 15:15:16 +0200 | [diff] [blame] | 464 | return i |
Klement Sekera | 75e7d13 | 2017-09-20 08:26:30 +0200 | [diff] [blame] | 465 | else: |
Juraj Sloboda | b3f9050 | 2018-10-04 15:15:16 +0200 | [diff] [blame] | 466 | return None |
| 467 | |
| 468 | def is_interface_config_in_dump(self, dump): |
| 469 | return self.get_interface_config_from_dump(dump) is not None |
| 470 | |
| 471 | def assert_interface_state(self, admin_up_down, link_up_down, |
| 472 | expect_event=False): |
| 473 | if expect_event: |
| 474 | event = self.test.vapi.wait_for_event(timeout=1, |
| 475 | name='sw_interface_event') |
| 476 | self.test.assert_equal(event.sw_if_index, self.sw_if_index, |
| 477 | "sw_if_index") |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 478 | self.test.assert_equal((event.flags & 1), admin_up_down, |
Juraj Sloboda | b3f9050 | 2018-10-04 15:15:16 +0200 | [diff] [blame] | 479 | "admin state") |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 480 | self.test.assert_equal((event.flags & 2), link_up_down, |
Juraj Sloboda | b3f9050 | 2018-10-04 15:15:16 +0200 | [diff] [blame] | 481 | "link state") |
| 482 | dump = self.test.vapi.sw_interface_dump() |
| 483 | if_state = self.get_interface_config_from_dump(dump) |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 484 | self.test.assert_equal((if_state.flags & 1), admin_up_down, |
Juraj Sloboda | b3f9050 | 2018-10-04 15:15:16 +0200 | [diff] [blame] | 485 | "admin state") |
Jakub Grajciar | 053204a | 2019-03-18 13:17:53 +0100 | [diff] [blame] | 486 | self.test.assert_equal((if_state.flags & 2), link_up_down, |
Juraj Sloboda | b3f9050 | 2018-10-04 15:15:16 +0200 | [diff] [blame] | 487 | "link state") |
Neale Ranns | 93cc3ee | 2018-10-10 07:22:51 -0700 | [diff] [blame] | 488 | |
| 489 | def __str__(self): |
| 490 | return self.name |
Neale Ranns | 2ac885c | 2019-03-20 18:24:43 +0000 | [diff] [blame] | 491 | |
| 492 | def get_rx_stats(self): |
| 493 | c = self.test.statistics.get_counter("^/if/rx$") |
| 494 | return c[0][self.sw_if_index] |
| 495 | |
| 496 | def get_tx_stats(self): |
| 497 | c = self.test.statistics.get_counter("^/if/tx$") |
| 498 | return c[0][self.sw_if_index] |
Ole Troan | eb284a1 | 2019-10-09 13:33:19 +0200 | [diff] [blame] | 499 | |
| 500 | def set_l3_mtu(self, mtu): |
| 501 | self.test.vapi.sw_interface_set_mtu(self.sw_if_index, [mtu, 0, 0, 0]) |
| 502 | |
| 503 | def set_ip4_mtu(self, mtu): |
| 504 | self.test.vapi.sw_interface_set_mtu(self.sw_if_index, [0, mtu, 0, 0]) |
| 505 | |
| 506 | def set_ip6_mtu(self, mtu): |
| 507 | self.test.vapi.sw_interface_set_mtu(self.sw_if_index, [0, 0, mtu, 0]) |
| 508 | |
| 509 | def set_mpls_mtu(self, mtu): |
| 510 | self.test.vapi.sw_interface_set_mtu(self.sw_if_index, [0, 0, 0, mtu]) |