blob: c961f2e85d736a1c2e3bd8df369bc1d7e2739126 [file] [log] [blame]
ShreyaPandita9306c772012-09-28 12:21:40 -04001"""These tests fall under Conformance Test-Suite (OF-SWITCH-1.0.0 TestCases).
2 Refer Documentation -- Detailed testing methodology
3 <Some of test-cases are directly taken from oftest> """
4
5"Test Suite 2 --> Openflow Protocol messages"
6
7
8import logging
9
10import unittest
11import random
12
Rich Lane477f4812012-10-04 22:49:00 -070013from oftest import config
ShreyaPandita9306c772012-09-28 12:21:40 -040014import oftest.controller as controller
Rich Laned7b0ffa2013-03-08 15:53:42 -080015import ofp
ShreyaPandita9306c772012-09-28 12:21:40 -040016import oftest.dataplane as dataplane
ShreyaPandita9306c772012-09-28 12:21:40 -040017import oftest.parse as parse
Rich Laneb90a1c42012-10-05 09:16:05 -070018import oftest.base_tests as base_tests
ShreyaPandita9306c772012-09-28 12:21:40 -040019
Rich Laneda3b5ad2012-10-03 09:05:32 -070020from oftest.testutils import *
ShreyaPandita9306c772012-09-28 12:21:40 -040021from time import sleep
22from FuncUtils import *
23
Rich Laneb90a1c42012-10-05 09:16:05 -070024class FeaturesRequest(base_tests.SimpleProtocol):
ShreyaPandita9306c772012-09-28 12:21:40 -040025
26 """Verify Features_Request-Reply is implemented
27 a) Send OFPT_FEATURES_REQUEST
28 b) Verify OFPT_FEATURES_REPLY is received without errors"""
29
30 def runTest(self):
Rich Lane9a003812012-10-04 17:17:59 -070031 logging.info("Running Features_Request test")
ShreyaPandita9306c772012-09-28 12:21:40 -040032
Rich Lane477f4812012-10-04 22:49:00 -070033 of_ports = config["port_map"].keys()
ShreyaPandita9306c772012-09-28 12:21:40 -040034 of_ports.sort()
35
36 #Clear switch state
Rich Lane32bf9482013-01-03 17:26:30 -080037 delete_all_flows(self.controller)
ShreyaPandita9306c772012-09-28 12:21:40 -040038
Rich Lane9a003812012-10-04 17:17:59 -070039 logging.info("Sending Features_Request")
40 logging.info("Expecting Features_Reply")
ShreyaPandita9306c772012-09-28 12:21:40 -040041
Rich Lane28fa9272013-03-08 16:00:25 -080042 request = ofp.message.features_request()
Rich Lane5c3151c2013-01-03 17:15:41 -080043 self.controller.message_send(request)
ShreyaPandita9306c772012-09-28 12:21:40 -040044
45 (response, pkt) = self.controller.poll(exp_msg=ofp.OFPT_FEATURES_REPLY,
46 timeout=2)
47 self.assertTrue(response is not None,
48 'Did not receive Features Reply')
49
50
Rich Laneb90a1c42012-10-05 09:16:05 -070051class ConfigurationRequest(base_tests.SimpleProtocol):
ShreyaPandita9306c772012-09-28 12:21:40 -040052
53 """Check basic Get Config request is implemented
54 a) Send OFPT_GET_CONFIG_REQUEST
55 b) Verify OFPT_GET_CONFIG_REPLY is received without errors"""
56
57 def runTest(self):
58
Rich Lane9a003812012-10-04 17:17:59 -070059 logging.info("Running Configuration_Request test ")
ShreyaPandita9306c772012-09-28 12:21:40 -040060
Rich Lane477f4812012-10-04 22:49:00 -070061 of_ports = config["port_map"].keys()
ShreyaPandita9306c772012-09-28 12:21:40 -040062 of_ports.sort()
63
64 #Clear switch state
Rich Lane32bf9482013-01-03 17:26:30 -080065 delete_all_flows(self.controller)
ShreyaPandita9306c772012-09-28 12:21:40 -040066
Rich Lane9a003812012-10-04 17:17:59 -070067 logging.info("Sending OFPT_GET_CONFIG_REQUEST ")
68 logging.info("Expecting OFPT_GET_CONFIG_REPLY ")
ShreyaPandita9306c772012-09-28 12:21:40 -040069
Rich Lane28fa9272013-03-08 16:00:25 -080070 request = ofp.message.get_config_request()
Rich Lane5c3151c2013-01-03 17:15:41 -080071 self.controller.message_send(request)
ShreyaPandita9306c772012-09-28 12:21:40 -040072
73 (response, pkt) = self.controller.poll(exp_msg=ofp.OFPT_GET_CONFIG_REPLY,
74 timeout=2)
75 self.assertTrue(response is not None,
76 'Did not receive OFPT_GET_CONFIG_REPLY')
77
Rich Laneb90a1c42012-10-05 09:16:05 -070078class ModifyStateAdd(base_tests.SimpleProtocol):
ShreyaPandita9306c772012-09-28 12:21:40 -040079
80 """Check basic Flow Add request is implemented
81 a) Send OFPT_FLOW_MOD , command = OFPFC_ADD
82 c) Send ofp_table_stats request , verify active_count=1 in reply"""
83
84 def runTest(self):
85
Rich Lane9a003812012-10-04 17:17:59 -070086 logging.info("Running Modify_State_Add test")
ShreyaPandita9306c772012-09-28 12:21:40 -040087
Rich Lane477f4812012-10-04 22:49:00 -070088 of_ports = config["port_map"].keys()
ShreyaPandita9306c772012-09-28 12:21:40 -040089 of_ports.sort()
90
91 #Clear switch state
Rich Lane32bf9482013-01-03 17:26:30 -080092 delete_all_flows(self.controller)
ShreyaPandita9306c772012-09-28 12:21:40 -040093
Rich Lane9a003812012-10-04 17:17:59 -070094 logging.info("Inserting a flow entry")
95 logging.info("Expecting active_count=1 in table_stats_reply")
ShreyaPandita9306c772012-09-28 12:21:40 -040096
97 #Insert a flow entry matching on ingress_port
ShreyaPandita8dab4662012-11-02 13:40:14 -040098 (pkt,match) = wildcard_all_except_ingress(self,of_ports)
ShreyaPandita9306c772012-09-28 12:21:40 -040099
100 # Send Table_Stats_Request and verify flow gets inserted.
ShreyaPandita8dab4662012-11-02 13:40:14 -0400101 verify_tablestats(self,expect_active=1)
ShreyaPandita9306c772012-09-28 12:21:40 -0400102
103
Rich Laneb90a1c42012-10-05 09:16:05 -0700104class ModifyStateDelete(base_tests.SimpleProtocol):
ShreyaPandita9306c772012-09-28 12:21:40 -0400105
106 """Check Basic Flow Delete request is implemented
107 a) Send OFPT_FLOW_MOD, command = OFPFC_ADD
108 b) Send ofp_table_stats request , verify active_count=1 in reply
109 c) Send OFPT_FLOW_MOD, command = OFPFC_DELETE
110 c) Send ofp_table_stats request , verify active_count=0 in reply"""
111
112 def runTest(self):
113
Rich Lane9a003812012-10-04 17:17:59 -0700114 logging.info("Running Modify_State_Delete test")
ShreyaPandita9306c772012-09-28 12:21:40 -0400115
Rich Lane477f4812012-10-04 22:49:00 -0700116 of_ports = config["port_map"].keys()
ShreyaPandita9306c772012-09-28 12:21:40 -0400117 of_ports.sort()
118
119 #Clear switch state
Rich Lane32bf9482013-01-03 17:26:30 -0800120 delete_all_flows(self.controller)
ShreyaPandita9306c772012-09-28 12:21:40 -0400121
Rich Lane9a003812012-10-04 17:17:59 -0700122 logging.info("Inserting a flow entry and then deleting it")
123 logging.info("Expecting the active_count=0 in table_stats_reply")
ShreyaPandita9306c772012-09-28 12:21:40 -0400124
125 #Insert a flow matching on ingress_port
ShreyaPandita8dab4662012-11-02 13:40:14 -0400126 (pkt,match) = wildcard_all_except_ingress(self,of_ports)
ShreyaPandita9306c772012-09-28 12:21:40 -0400127
128 #Verify Flow inserted.
ShreyaPandita8dab4662012-11-02 13:40:14 -0400129 verify_tablestats(self,expect_active=1)
ShreyaPandita9306c772012-09-28 12:21:40 -0400130
131 #Delete the flow
ShreyaPandita8dab4662012-11-02 13:40:14 -0400132 nonstrict_delete(self,match)
ShreyaPandita9306c772012-09-28 12:21:40 -0400133
134 # Send Table_Stats_Request and verify flow deleted.
ShreyaPandita8dab4662012-11-02 13:40:14 -0400135 verify_tablestats(self,expect_active=0)
ShreyaPandita9306c772012-09-28 12:21:40 -0400136
137
138
Rich Laneb90a1c42012-10-05 09:16:05 -0700139class ModifyStateModify(base_tests.SimpleDataPlane):
ShreyaPandita9306c772012-09-28 12:21:40 -0400140
141 """Verify basic Flow Modify request is implemented
142 a) Send OFPT_FLOW_MOD, command = OFPFC_ADD, Action A
143 b) Send OFPT_FLOW_MOD, command = OFPFC_MODIFY , with output action A'
144 c) Send a packet matching the flow, verify packet implements action A' """
145
146 def runTest(self):
147
Rich Lane9a003812012-10-04 17:17:59 -0700148 logging.info("Running Modify_State_Modify test")
ShreyaPandita9306c772012-09-28 12:21:40 -0400149
Rich Lane477f4812012-10-04 22:49:00 -0700150 of_ports = config["port_map"].keys()
ShreyaPandita9306c772012-09-28 12:21:40 -0400151 of_ports.sort()
152
153 #Clear switch state
Rich Lane32bf9482013-01-03 17:26:30 -0800154 delete_all_flows(self.controller)
ShreyaPandita9306c772012-09-28 12:21:40 -0400155
Rich Lane9a003812012-10-04 17:17:59 -0700156 logging.info("Inserting a flow entry and then modifying it")
157 logging.info("Expecting the Test Packet to implement the modified action")
ShreyaPandita9306c772012-09-28 12:21:40 -0400158
159 # Insert a flow matching on ingress_port with action A (output to of_port[1])
ShreyaPandita8dab4662012-11-02 13:40:14 -0400160 (pkt,match) = wildcard_all_except_ingress(self,of_ports)
ShreyaPandita9306c772012-09-28 12:21:40 -0400161
162 # Modify the flow action (output to of_port[2])
ShreyaPandita8dab4662012-11-02 13:40:14 -0400163 modify_flow_action(self,of_ports,match)
ShreyaPandita9306c772012-09-28 12:21:40 -0400164
165 # Send the Test Packet and verify action implemented is A' (output to of_port[2])
ShreyaPandita8dab4662012-11-02 13:40:14 -0400166 send_packet(self,pkt,of_ports[0],of_ports[2])
ShreyaPandita9306c772012-09-28 12:21:40 -0400167
168
Rich Laneb90a1c42012-10-05 09:16:05 -0700169class ReadState(base_tests.SimpleProtocol):
ShreyaPandita9306c772012-09-28 12:21:40 -0400170
171 """Test that a basic Read state request (like flow_stats_get request) does not generate an error
172 a) Send OFPT_FLOW_MOD, command = OFPFC_ADD
173 b) Send ofp_flow_stats request
174 b) Verify switch replies without errors"""
175
176 def runTest(self):
177
Rich Lane9a003812012-10-04 17:17:59 -0700178 logging.info("Running Read_State test")
ShreyaPandita9306c772012-09-28 12:21:40 -0400179
Rich Lane477f4812012-10-04 22:49:00 -0700180 of_ports = config["port_map"].keys()
ShreyaPandita9306c772012-09-28 12:21:40 -0400181 of_ports.sort()
182
183 #Clear switch state
Rich Lane32bf9482013-01-03 17:26:30 -0800184 delete_all_flows(self.controller)
ShreyaPandita9306c772012-09-28 12:21:40 -0400185
Rich Lane9a003812012-10-04 17:17:59 -0700186 logging.info("Inserting a flow entry and then sending flow_stats request")
187 logging.info("Expecting the a flow_stats_reply without errors")
ShreyaPandita9306c772012-09-28 12:21:40 -0400188
189 # Insert a flow with match on ingress_port
ShreyaPandita8dab4662012-11-02 13:40:14 -0400190 (pkt,match ) = wildcard_all_except_ingress(self,of_ports)
ShreyaPandita9306c772012-09-28 12:21:40 -0400191
192 #Verify Flow_Stats request does not generate errors
ShreyaPandita8dab4662012-11-02 13:40:14 -0400193 get_flowstats(self,match)
ShreyaPandita9306c772012-09-28 12:21:40 -0400194
Rich Laneb90a1c42012-10-05 09:16:05 -0700195class PacketOut(base_tests.SimpleDataPlane):
ShreyaPandita9306c772012-09-28 12:21:40 -0400196
197 """Test packet out function
198 a) Send packet out message for each dataplane port.
199 b) Verify the packet appears on the appropriate dataplane port"""
200
201 def runTest(self):
202
Rich Lane9a003812012-10-04 17:17:59 -0700203 logging.info("Running Packet_Out test")
ShreyaPandita9306c772012-09-28 12:21:40 -0400204
Rich Lane477f4812012-10-04 22:49:00 -0700205 of_ports = config["port_map"].keys()
ShreyaPandita9306c772012-09-28 12:21:40 -0400206 of_ports.sort()
207
208 #Clear Switch state
Rich Lane32bf9482013-01-03 17:26:30 -0800209 delete_all_flows(self.controller)
ShreyaPandita9306c772012-09-28 12:21:40 -0400210
Rich Lane9a003812012-10-04 17:17:59 -0700211 logging.info("Sending a packet-out for each dataplane port")
212 logging.info("Expecting the packet on appropriate dataplane port")
ShreyaPandita9306c772012-09-28 12:21:40 -0400213
214 for dp_port in of_ports:
215 for outpkt, opt in [
216 (simple_tcp_packet(), "simple TCP packet"),
217 (simple_eth_packet(), "simple Ethernet packet"),
218 (simple_eth_packet(pktlen=40), "tiny Ethernet packet")]:
219
Rich Lane28fa9272013-03-08 16:00:25 -0800220 msg = ofp.message.packet_out()
ShreyaPandita9306c772012-09-28 12:21:40 -0400221 msg.data = str(outpkt)
Rich Lane9d3cc6b2013-03-08 16:33:08 -0800222 act = ofp.action.output()
ShreyaPandita9306c772012-09-28 12:21:40 -0400223 act.port = dp_port
Rich Lanec495d9e2013-03-08 17:43:36 -0800224 msg.actions.append(act)
ShreyaPandita9306c772012-09-28 12:21:40 -0400225
Rich Lane9a003812012-10-04 17:17:59 -0700226 logging.info("PacketOut to: " + str(dp_port))
Rich Lane5c3151c2013-01-03 17:15:41 -0800227 self.controller.message_send(msg)
ShreyaPandita9306c772012-09-28 12:21:40 -0400228
229 exp_pkt_arg = None
230 exp_port = None
Rich Lane477f4812012-10-04 22:49:00 -0700231 if config["relax"]:
ShreyaPandita9306c772012-09-28 12:21:40 -0400232 exp_pkt_arg = outpkt
233 exp_port = dp_port
234 (of_port, pkt, pkt_time) = self.dataplane.poll(timeout=2,
235 port_number=exp_port,
236 exp_pkt=exp_pkt_arg)
237
238 self.assertTrue(pkt is not None, 'Packet not received')
Rich Lane9a003812012-10-04 17:17:59 -0700239 logging.info("PacketOut: got pkt from " + str(of_port))
ShreyaPandita9306c772012-09-28 12:21:40 -0400240 if of_port is not None:
241 self.assertEqual(of_port, dp_port, "Unexpected receive port")
242 if not dataplane.match_exp_pkt(outpkt, pkt):
Rich Lane9a003812012-10-04 17:17:59 -0700243 logging.debug("Sent %s" % format_packet(outpkt))
244 logging.debug("Resp %s" % format_packet(
ShreyaPandita9306c772012-09-28 12:21:40 -0400245 str(pkt)[:len(str(outpkt))]))
246 self.assertEqual(str(outpkt), str(pkt)[:len(str(outpkt))],
247 'Response packet does not match send packet')
248
249
Rich Laneb90a1c42012-10-05 09:16:05 -0700250class PacketIn(base_tests.SimpleDataPlane):
ShreyaPandita9306c772012-09-28 12:21:40 -0400251
252 """Test basic packet_in function
253 a) Send a simple tcp packet to a dataplane port, without any flow-entry
254 b) Verify that a packet_in event is sent to the controller"""
255
256 def runTest(self):
257
Rich Lane9a003812012-10-04 17:17:59 -0700258 logging.info("Running Packet_In test")
ShreyaPandita9306c772012-09-28 12:21:40 -0400259
Rich Lane477f4812012-10-04 22:49:00 -0700260 of_ports = config["port_map"].keys()
ShreyaPandita9306c772012-09-28 12:21:40 -0400261 of_ports.sort()
262 ingress_port = of_ports[0]
263
264 #Clear Switch state
Rich Lane32bf9482013-01-03 17:26:30 -0800265 delete_all_flows(self.controller)
Rich Lane3a261d52013-01-03 17:45:08 -0800266 do_barrier(self.controller)
ShreyaPandita9306c772012-09-28 12:21:40 -0400267
Rich Lane9a003812012-10-04 17:17:59 -0700268 logging.info("Sending a Simple tcp packet a dataplane port")
269 logging.info("Expecting a packet_in event on the control plane")
ShreyaPandita9306c772012-09-28 12:21:40 -0400270
271 # Send packet on dataplane port and verify packet_in event gets generated.
272 pkt = simple_tcp_packet()
273 self.dataplane.send(ingress_port, str(pkt))
Rich Lane9a003812012-10-04 17:17:59 -0700274 logging.info("Sending packet to dp port " + str(ingress_port) +
ShreyaPandita9306c772012-09-28 12:21:40 -0400275 ", expecting packet_in on control plane" )
276
277 (response, pkt) = self.controller.poll(exp_msg=ofp.OFPT_PACKET_IN,
278 timeout=2)
279 self.assertTrue(response is not None,
280 'Packet in event is not sent to the controller')
281
282
Rich Laneb90a1c42012-10-05 09:16:05 -0700283class Hello(base_tests.SimpleDataPlane):
ShreyaPandita9306c772012-09-28 12:21:40 -0400284
285 """Test Hello messages are implemented
286 a) Create Hello messages from controller
287 b) Verify switch also exchanges hello message -- (Poll the control plane)
288 d) Verify the version field in the hello messages is openflow 1.0.0 """
289
290 def runTest(self):
291
Rich Lane9a003812012-10-04 17:17:59 -0700292 logging.info("Running Hello test")
ShreyaPandita9306c772012-09-28 12:21:40 -0400293
Rich Lane9a003812012-10-04 17:17:59 -0700294 logging.info("Sending Hello")
295 logging.info("Expecting a Hello on the control plane with version--1.0.0")
ShreyaPandita9306c772012-09-28 12:21:40 -0400296
297 #Send Hello message
Rich Lane28fa9272013-03-08 16:00:25 -0800298 request = ofp.message.hello()
ShreyaPandita9306c772012-09-28 12:21:40 -0400299 (response, pkt) = self.controller.poll(exp_msg=ofp.OFPT_HELLO,
300 timeout=1)
301 self.assertTrue(response is not None,
302 'Switch did not exchange hello message in return')
Rich Laneb73808c2013-03-11 15:22:23 -0700303 self.assertTrue(response.version == 0x01, 'switch openflow-version field is not 1.0.0')
ShreyaPandita9306c772012-09-28 12:21:40 -0400304
305
306
Rich Laneb90a1c42012-10-05 09:16:05 -0700307class EchoWithoutBody(base_tests.SimpleProtocol):
ShreyaPandita9306c772012-09-28 12:21:40 -0400308
309 """Test basic echo-reply is implemented
310 a) Send echo-request from the controller side, note echo body is empty here.
311 b) Verify switch responds back with echo-reply with same xid """
312
313 def runTest(self):
314
Rich Lane9a003812012-10-04 17:17:59 -0700315 logging.info("Running Echo_Without_Body test")
ShreyaPandita9306c772012-09-28 12:21:40 -0400316
Rich Lane9a003812012-10-04 17:17:59 -0700317 logging.info("Sending Echo Request")
318 logging.info("Expecting a Echo Reply with version--1.0.0 and same xid")
ShreyaPandita9306c772012-09-28 12:21:40 -0400319
320 # Send echo_request
Rich Lane28fa9272013-03-08 16:00:25 -0800321 request = ofp.message.echo_request()
ShreyaPandita9306c772012-09-28 12:21:40 -0400322 (response, pkt) = self.controller.transact(request)
Rich Laneb73808c2013-03-11 15:22:23 -0700323 self.assertEqual(response.type, ofp.OFPT_ECHO_REPLY,'response is not echo_reply')
324 self.assertEqual(request.xid, response.xid,
ShreyaPandita9306c772012-09-28 12:21:40 -0400325 'response xid != request xid')
Rich Laneb73808c2013-03-11 15:22:23 -0700326 self.assertTrue(response.version == 0x01, 'switch openflow-version field is not 1.0.1')
ShreyaPandita9306c772012-09-28 12:21:40 -0400327 self.assertEqual(len(response.data), 0, 'response data non-empty')
328
329
Rich Laneb90a1c42012-10-05 09:16:05 -0700330class BarrierRequestReply(base_tests.SimpleProtocol):
ShreyaPandita9306c772012-09-28 12:21:40 -0400331
332 """ Check basic Barrier request is implemented
333 a) Send OFPT_BARRIER_REQUEST
334 c) Verify OFPT_BARRIER_REPLY is recieved"""
335
336 def runTest(self):
337
Rich Lane9a003812012-10-04 17:17:59 -0700338 logging.info("Running Barrier_Request_Reply test")
ShreyaPandita9306c772012-09-28 12:21:40 -0400339
Rich Lane9a003812012-10-04 17:17:59 -0700340 logging.info("Sending Barrier Request")
341 logging.info("Expecting a Barrier Reply with same xid")
ShreyaPandita9306c772012-09-28 12:21:40 -0400342
343 #Send Barrier Request
Rich Lane28fa9272013-03-08 16:00:25 -0800344 request = ofp.message.barrier_request()
ShreyaPandita8dab4662012-11-02 13:40:14 -0400345 (response,pkt) = self.controller.transact(request)
Rich Laneb73808c2013-03-11 15:22:23 -0700346 self.assertEqual(response.type, ofp.OFPT_BARRIER_REPLY,'response is not barrier_reply')
347 self.assertEqual(request.xid, response.xid,
ShreyaPandita9306c772012-09-28 12:21:40 -0400348 'response xid != request xid')
349
350
351
352
353