beacd80f687c937476b7188c21b9a34ccc4afe75
[vpp.git] / test / bfd.py
1 from socket import AF_INET, AF_INET6
2 from scapy.all import *
3 from scapy.packet import *
4 from scapy.fields import *
5 from framework import *
6 from vpp_object import *
7 from util import NumericConstant
8
9
10 class BFDDiagCode(NumericConstant):
11     """ BFD Diagnostic Code """
12     no_diagnostic = 0
13     control_detection_time_expired = 1
14     echo_function_failed = 2
15     neighbor_signaled_session_down = 3
16     forwarding_plane_reset = 4
17     path_down = 5
18     concatenated_path_down = 6
19     administratively_down = 7
20     reverse_concatenated_path_down = 8
21
22     desc_dict = {
23         no_diagnostic: "No diagnostic",
24         control_detection_time_expired: "Control Detection Time Expired",
25         echo_function_failed: "Echo Function Failed",
26         neighbor_signaled_session_down: "Neighbor Signaled Session Down",
27         forwarding_plane_reset: "Forwarding Plane Reset",
28         path_down: "Path Down",
29         concatenated_path_down: "Concatenated Path Down",
30         administratively_down: "Administratively Down",
31         reverse_concatenated_path_down: "Reverse Concatenated Path Down",
32     }
33
34     def __init__(self, value):
35         NumericConstant.__init__(self, value)
36
37
38 class BFDState(NumericConstant):
39     """ BFD State """
40     admin_down = 0
41     down = 1
42     init = 2
43     up = 3
44
45     desc_dict = {
46         admin_down: "AdminDown",
47         down: "Down",
48         init: "Init",
49         up: "Up",
50     }
51
52     def __init__(self, value):
53         NumericConstant.__init__(self, value)
54
55
56 class BFD(Packet):
57
58     udp_dport = 3784  #: BFD destination port per RFC 5881
59     udp_sport_min = 49152  #: BFD source port min value per RFC 5881
60     udp_sport_max = 65535  #: BFD source port max value per RFC 5881
61
62     name = "BFD"
63
64     fields_desc = [
65         BitField("version", 1, 3),
66         BitEnumField("diag", 0, 5, BFDDiagCode.desc_dict),
67         BitEnumField("state", 0, 2, BFDState.desc_dict),
68         FlagsField("flags", 0, 6, ['P', 'F', 'C', 'A', 'D', 'M']),
69         XByteField("detect_mult", 0),
70         XByteField("length", 24),
71         BitField("my_discriminator", 0, 32),
72         BitField("your_discriminator", 0, 32),
73         BitField("desired_min_tx_interval", 0, 32),
74         BitField("required_min_rx_interval", 0, 32),
75         BitField("required_min_echo_rx_interval", 0, 32)]
76
77     def mysummary(self):
78         return self.sprintf("BFD(my_disc=%BFD.my_discriminator%,"
79                             "your_disc=%BFD.your_discriminator%)")
80
81 # glue the BFD packet class to scapy parser
82 bind_layers(UDP, BFD, dport=BFD.udp_dport)
83
84
85 class VppBFDUDPSession(VppObject):
86     """ Represents BFD UDP session in VPP """
87
88     @property
89     def test(self):
90         """ Test which created this session """
91         return self._test
92
93     @property
94     def interface(self):
95         """ Interface on which this session lives """
96         return self._interface
97
98     @property
99     def af(self):
100         """ Address family - AF_INET or AF_INET6 """
101         return self._af
102
103     @property
104     def bs_index(self):
105         """ BFD session index from VPP """
106         if self._bs_index is not None:
107             return self._bs_index
108         raise NotConfiguredException("not configured")
109
110     @property
111     def local_addr(self):
112         """ BFD session local address (VPP address) """
113         if self._local_addr is None:
114             return self._interface.local_ip4
115         return self._local_addr
116
117     @property
118     def local_addr_n(self):
119         """ BFD session local address (VPP address) - raw, suitable for API """
120         if self._local_addr is None:
121             return self._interface.local_ip4n
122         return self._local_addr_n
123
124     @property
125     def peer_addr(self):
126         """ BFD session peer address """
127         return self._peer_addr
128
129     @property
130     def peer_addr_n(self):
131         """ BFD session peer address - raw, suitable for API """
132         return self._peer_addr_n
133
134     @property
135     def state(self):
136         """ BFD session state """
137         result = self.test.vapi.bfd_udp_session_dump()
138         session = None
139         for s in result:
140             if s.sw_if_index == self.interface.sw_if_index:
141                 if self.af == AF_INET \
142                         and s.is_ipv6 == 0 \
143                         and self.interface.local_ip4n == s.local_addr[:4] \
144                         and self.interface.remote_ip4n == s.peer_addr[:4]:
145                     session = s
146                     break
147         if session is None:
148             raise Exception(
149                 "Could not find BFD session in VPP response: %s" % repr(result))
150         return session.state
151
152     @property
153     def desired_min_tx(self):
154         return self._desired_min_tx
155
156     @property
157     def required_min_rx(self):
158         return self._required_min_rx
159
160     @property
161     def detect_mult(self):
162         return self._detect_mult
163
164     def __init__(self, test, interface, peer_addr, local_addr=None, af=AF_INET):
165         self._test = test
166         self._interface = interface
167         self._af = af
168         self._local_addr = local_addr
169         self._peer_addr = peer_addr
170         self._peer_addr_n = socket.inet_pton(af, peer_addr)
171         self._bs_index = None
172         self._desired_min_tx = 200000  # 0.2s
173         self._required_min_rx = 200000  # 0.2s
174         self._detect_mult = 3  # 3 packets need to be missed
175
176     def add_vpp_config(self):
177         is_ipv6 = 1 if AF_INET6 == self.af else 0
178         result = self.test.vapi.bfd_udp_add(
179             self._interface.sw_if_index,
180             self.desired_min_tx,
181             self.required_min_rx,
182             self.detect_mult,
183             self.local_addr_n,
184             self.peer_addr_n,
185             is_ipv6=is_ipv6)
186         self._bs_index = result.bs_index
187
188     def query_vpp_config(self):
189         result = self.test.vapi.bfd_udp_session_dump()
190         session = None
191         for s in result:
192             if s.sw_if_index == self.interface.sw_if_index:
193                 if self.af == AF_INET \
194                         and s.is_ipv6 == 0 \
195                         and self.interface.local_ip4n == s.local_addr[:4] \
196                         and self.interface.remote_ip4n == s.peer_addr[:4]:
197                     session = s
198                     break
199         if session is None:
200             return False
201         return True
202
203     def remove_vpp_config(self):
204         if hasattr(self, '_bs_index'):
205             is_ipv6 = 1 if AF_INET6 == self._af else 0
206             self.test.vapi.bfd_udp_del(
207                 self._interface.sw_if_index,
208                 self.local_addr_n,
209                 self.peer_addr_n,
210                 is_ipv6=is_ipv6)
211
212     def object_id(self):
213         return "bfd-udp-%d" % self.bs_index
214
215     def admin_up(self):
216         self.test.vapi.bfd_session_set_flags(self.bs_index, 1)