3 ##############################################################################
4 # Copyright (c) 2021 Orange, Inc. and others. All rights reserved.
6 # All rights reserved. This program and the accompanying materials
7 # are made available under the terms of the Apache License, Version 2.0
8 # which accompanies this distribution, and is available at
9 # http://www.apache.org/licenses/LICENSE-2.0
10 ##############################################################################
12 # pylint: disable=no-member
13 # pylint: disable=too-many-public-methods
19 # pylint: disable=wrong-import-order
21 sys.path.append('transportpce_tests/common/')
22 # pylint: disable=wrong-import-position
23 # pylint: disable=import-error
24 import test_utils # nopep8
27 # pylint: disable=too-few-public-methods
30 # pylint: disable=invalid-name
36 class TransportPCEtesting(unittest.TestCase):
39 WAITING = 20 # nominal value is 300
40 NODE_VERSION = '2.2.1'
41 uuid_services = UuidServices()
43 cr_serv_sample_data = {
47 "layer-protocol-name": "PHOTONIC_MEDIA",
48 "service-interface-point": {
49 "service-interface-point-uuid": "b1a0d883-32b8-3b0b-93d6-7ed074f6f107"
51 "administrative-state": "UNLOCKED",
52 "operational-state": "ENABLED",
53 "direction": "BIDIRECTIONAL",
55 "protection-role": "WORK",
56 "local-id": "SPDR-SA1-XPDR1",
59 "value-name": "OpenROADM node id",
60 "value": "SPDR-SA1-XPDR1"
65 "layer-protocol-name": "PHOTONIC_MEDIA",
66 "service-interface-point": {
67 "service-interface-point-uuid": "d1d6305e-179b-346f-b02d-8260aebe1ce8"
69 "administrative-state": "UNLOCKED",
70 "operational-state": "ENABLED",
71 "direction": "BIDIRECTIONAL",
73 "protection-role": "WORK",
74 "local-id": "SPDR-SC1-XPDR1",
77 "value-name": "OpenROADM node id",
78 "value": "SPDR-SC1-XPDR1"
83 "connectivity-constraint": {
84 "service-layer": "PHOTONIC_MEDIA",
85 "service-type": "POINT_TO_POINT_CONNECTIVITY",
86 "service-level": "Some service-level",
87 "requested-capacity": {
94 "state": "Some state"}}
98 # pylint: disable=unsubscriptable-object
99 cls.init_failed = False
100 os.environ['JAVA_MIN_MEM'] = '1024M'
101 os.environ['JAVA_MAX_MEM'] = '4096M'
102 cls.processes = test_utils.start_tpce()
103 # TAPI feature is not installed by default in Karaf
104 if "USE_LIGHTY" not in os.environ or os.environ['USE_LIGHTY'] != 'True':
105 print("installing tapi feature...")
106 result = test_utils.install_karaf_feature("odl-transportpce-tapi")
107 if result.returncode != 0:
108 cls.init_failed = True
109 print("Restarting OpenDaylight...")
110 test_utils.shutdown_process(cls.processes[0])
111 cls.processes[0] = test_utils.start_karaf()
112 test_utils.process_list[0] = cls.processes[0]
113 cls.init_failed = not test_utils.wait_until_log_contains(
114 test_utils.KARAF_LOG, test_utils.KARAF_OK_START_MSG, time_to_wait=60)
116 print("tapi installation feature failed...")
117 test_utils.shutdown_process(cls.processes[0])
119 cls.processes = test_utils.start_sims([('spdra', cls.NODE_VERSION),
120 ('roadma', cls.NODE_VERSION),
121 ('roadmc', cls.NODE_VERSION),
122 ('spdrc', cls.NODE_VERSION)])
125 def tearDownClass(cls):
126 # pylint: disable=not-an-iterable
127 for process in cls.processes:
128 test_utils.shutdown_process(process)
129 print("all processes killed")
134 def test_01_connect_spdrA(self):
135 print("Connecting SPDRA")
136 response = test_utils.mount_tapi_device("SPDR-SA1", ('spdra', self.NODE_VERSION))
137 self.assertEqual(response.status_code,
138 requests.codes.created, test_utils.CODE_SHOULD_BE_201)
141 def test_02_connect_spdrC(self):
142 print("Connecting SPDRC")
143 response = test_utils.mount_tapi_device("SPDR-SC1", ('spdrc', self.NODE_VERSION))
144 self.assertEqual(response.status_code,
145 requests.codes.created, test_utils.CODE_SHOULD_BE_201)
148 def test_03_connect_rdmA(self):
149 print("Connecting ROADMA")
150 response = test_utils.mount_tapi_device("ROADM-A1", ('roadma', self.NODE_VERSION))
151 self.assertEqual(response.status_code,
152 requests.codes.created, test_utils.CODE_SHOULD_BE_201)
155 def test_04_connect_rdmC(self):
156 print("Connecting ROADMC")
157 response = test_utils.mount_tapi_device("ROADM-C1", ('roadmc', self.NODE_VERSION))
158 self.assertEqual(response.status_code,
159 requests.codes.created, test_utils.CODE_SHOULD_BE_201)
162 def test_05_connect_sprdA_1_N1_to_roadmA_PP1(self):
163 response = test_utils.connect_xpdr_to_rdm_request("SPDR-SA1", "1", "1",
164 "ROADM-A1", "1", "SRG1-PP1-TXRX")
165 self.assertEqual(response.status_code, requests.codes.ok)
166 res = response.json()
167 self.assertIn('Xponder Roadm Link created successfully',
168 res["output"]["result"])
171 def test_06_connect_roadmA_PP1_to_spdrA_1_N1(self):
172 response = test_utils.connect_rdm_to_xpdr_request("SPDR-SA1", "1", "1",
173 "ROADM-A1", "1", "SRG1-PP1-TXRX")
174 self.assertEqual(response.status_code, requests.codes.ok)
175 res = response.json()
176 self.assertIn('Roadm Xponder links created successfully',
177 res["output"]["result"])
180 def test_07_connect_sprdC_1_N1_to_roadmC_PP1(self):
181 response = test_utils.connect_xpdr_to_rdm_request("SPDR-SC1", "1", "1",
182 "ROADM-C1", "1", "SRG1-PP1-TXRX")
183 self.assertEqual(response.status_code, requests.codes.ok)
184 res = response.json()
185 self.assertIn('Xponder Roadm Link created successfully',
186 res["output"]["result"])
189 def test_08_connect_roadmC_PP1_to_spdrC_1_N1(self):
190 response = test_utils.connect_rdm_to_xpdr_request("SPDR-SC1", "1", "1",
191 "ROADM-C1", "1", "SRG1-PP1-TXRX")
192 self.assertEqual(response.status_code, requests.codes.ok)
193 res = response.json()
194 self.assertIn('Roadm Xponder links created successfully',
195 res["output"]["result"])
198 def test_09_add_omsAttributes_ROADMA_ROADMC(self):
199 # Config ROADMA-ROADMC oms-attributes
201 "auto-spanloss": "true",
202 "spanloss-base": 11.4,
203 "spanloss-current": 12,
204 "engineered-spanloss": 12.2,
205 "link-concatenation": [{
208 "SRLG-length": 100000,
210 response = test_utils.add_oms_attr_request(
211 "ROADM-A1-DEG2-DEG2-TTP-TXRXtoROADM-C1-DEG1-DEG1-TTP-TXRX", data)
212 self.assertEqual(response.status_code, requests.codes.created)
215 def test_10_add_omsAttributes_ROADMC_ROADMA(self):
216 # Config ROADMC-ROADMA oms-attributes
218 "auto-spanloss": "true",
219 "spanloss-base": 11.4,
220 "spanloss-current": 12,
221 "engineered-spanloss": 12.2,
222 "link-concatenation": [{
225 "SRLG-length": 100000,
227 response = test_utils.add_oms_attr_request(
228 "ROADM-C1-DEG1-DEG1-TTP-TXRXtoROADM-A1-DEG2-DEG2-TTP-TXRX", data)
229 self.assertEqual(response.status_code, requests.codes.created)
232 def test_11_check_otn_topology(self):
233 response = test_utils.get_otn_topo_request()
234 self.assertEqual(response.status_code, requests.codes.ok)
235 res = response.json()
236 nbNode = len(res['network'][0]['node'])
237 self.assertEqual(nbNode, 6, 'There should be 6 otn nodes')
238 self.assertNotIn('ietf-network-topology:link', res['network'][0])
241 def test_12_check_openroadm_topology(self):
242 response = test_utils.get_ordm_topo_request("")
243 self.assertEqual(response.status_code, requests.codes.ok)
244 res = response.json()
245 nbNode = len(res['network'][0]['node'])
246 nbLink = len(res['network'][0]['ietf-network-topology:link'])
247 self.assertEqual(nbNode, 13, 'There should be 13 openroadm nodes')
248 self.assertEqual(nbLink, 22, 'There should be 22 openroadm links')
251 def test_13_get_tapi_topology_details(self):
252 response = test_utils.tapi_get_topology_details_request(
253 "T0 - Full Multi-layer topology")
255 self.assertEqual(response.status_code, requests.codes.ok)
256 res = response.json()
257 nbNode = len(res['output']['topology']['node'])
258 nbLink = len(res['output']['topology']['link'])
259 self.assertEqual(nbNode, 14, 'There should be 14 TAPI nodes')
260 self.assertEqual(nbLink, 15, 'There should be 15 TAPI links')
263 def test_14_check_sip_details(self):
264 response = test_utils.tapi_get_sip_details_request()
265 self.assertEqual(response.status_code, requests.codes.ok)
266 res = response.json()
267 nbSip = len(res['output']['sip'])
268 self.assertEqual(nbSip, 60, 'There should be 60 service interface point')
271 # test create connectivity service from spdrA to spdrC for Photonic_media
272 def test_15_create_connectivity_service_PhotonicMedia(self):
273 response = test_utils.tapi_create_connectivity_request(self.cr_serv_sample_data)
274 time.sleep(self.WAITING)
275 self.assertEqual(response.status_code, requests.codes.ok)
276 res = response.json()
277 self.uuid_services.pm = res['output']['service']['uuid']
278 # pylint: disable=consider-using-f-string
279 print("photonic media service uuid : {}".format(self.uuid_services.pm))
281 input_dict_1 = {'administrative-state': 'LOCKED',
282 'lifecycle-state': 'PLANNED',
283 'operational-state': 'DISABLED',
284 'service-type': 'POINT_TO_POINT_CONNECTIVITY',
285 'service-layer': 'PHOTONIC_MEDIA',
286 'connectivity-direction': 'BIDIRECTIONAL'
288 input_dict_2 = {'value-name': 'OpenROADM node id',
289 'value': 'SPDR-SC1-XPDR1'}
290 input_dict_3 = {'value-name': 'OpenROADM node id',
291 'value': 'SPDR-SA1-XPDR1'}
293 self.assertDictEqual(dict(input_dict_1, **res['output']['service']),
294 res['output']['service'])
295 self.assertDictEqual(dict(input_dict_2, **res['output']['service']['end-point'][0]['name'][0]),
296 res['output']['service']['end-point'][0]['name'][0])
297 self.assertDictEqual(dict(input_dict_3, **res['output']['service']['end-point'][1]['name'][0]),
298 res['output']['service']['end-point'][1]['name'][0])
299 # If the gate fails is because of the waiting time not being enough
300 time.sleep(self.WAITING)
302 def test_16_get_service_PhotonicMedia(self):
303 response = test_utils.get_service_list_request(
304 "services/" + str(self.uuid_services.pm))
305 self.assertEqual(response.status_code, requests.codes.ok)
306 res = response.json()
308 res['services'][0]['administrative-state'], 'inService')
310 res['services'][0]['service-name'], self.uuid_services.pm)
312 res['services'][0]['connection-type'], 'infrastructure')
314 res['services'][0]['lifecycle-state'], 'planned')
317 # test create connectivity service from spdrA to spdrC for odu
318 def test_17_create_connectivity_service_ODU(self):
319 # pylint: disable=line-too-long
320 self.cr_serv_sample_data["input"]["end-point"][0]["layer-protocol-name"] = "ODU"
321 self.cr_serv_sample_data["input"]["end-point"][0]["service-interface-point"]["service-interface-point-uuid"] = "5efda776-f8de-3e0b-9bbd-2c702e210946"
322 self.cr_serv_sample_data["input"]["end-point"][1]["layer-protocol-name"] = "ODU"
323 self.cr_serv_sample_data["input"]["end-point"][1]["service-interface-point"]["service-interface-point-uuid"] = "8116d0af-39fa-3df5-bed2-dd2cd5e8217d"
324 self.cr_serv_sample_data["input"]["connectivity-constraint"]["service-layer"] = "ODU"
325 self.cr_serv_sample_data["input"]["connectivity-constraint"]["service-level"] = self.uuid_services.pm
327 response = test_utils.tapi_create_connectivity_request(self.cr_serv_sample_data)
328 time.sleep(self.WAITING)
329 self.assertEqual(response.status_code, requests.codes.ok)
330 res = response.json()
331 self.uuid_services.odu = res['output']['service']['uuid']
332 # pylint: disable=consider-using-f-string
333 print("odu service uuid : {}".format(self.uuid_services.odu))
335 input_dict_1 = {'administrative-state': 'LOCKED',
336 'lifecycle-state': 'PLANNED',
337 'operational-state': 'DISABLED',
338 'service-type': 'POINT_TO_POINT_CONNECTIVITY',
339 'service-layer': 'ODU',
340 'connectivity-direction': 'BIDIRECTIONAL'
342 input_dict_2 = {'value-name': 'OpenROADM node id',
343 'value': 'SPDR-SC1-XPDR1'}
344 input_dict_3 = {'value-name': 'OpenROADM node id',
345 'value': 'SPDR-SA1-XPDR1'}
347 self.assertDictEqual(dict(input_dict_1, **res['output']['service']),
348 res['output']['service'])
349 self.assertDictEqual(dict(input_dict_2, **res['output']['service']['end-point'][0]['name'][0]),
350 res['output']['service']['end-point'][0]['name'][0])
351 self.assertDictEqual(dict(input_dict_3, **res['output']['service']['end-point'][1]['name'][0]),
352 res['output']['service']['end-point'][1]['name'][0])
353 # If the gate fails is because of the waiting time not being enough
354 time.sleep(self.WAITING)
356 def test_18_get_service_ODU(self):
357 response = test_utils.get_service_list_request(
358 "services/" + str(self.uuid_services.odu))
359 self.assertEqual(response.status_code, requests.codes.ok)
360 res = response.json()
362 res['services'][0]['administrative-state'], 'inService')
364 res['services'][0]['service-name'], self.uuid_services.odu)
366 res['services'][0]['connection-type'], 'infrastructure')
368 res['services'][0]['lifecycle-state'], 'planned')
371 # test create connectivity service from spdrA to spdrC for dsr
372 def test_19_create_connectivity_service_DSR(self):
373 # pylint: disable=line-too-long
374 self.cr_serv_sample_data["input"]["end-point"][0]["layer-protocol-name"] = "DSR"
375 self.cr_serv_sample_data["input"]["end-point"][0]["service-interface-point"]["service-interface-point-uuid"] = "c14797a0-adcc-3875-a1fe-df8949d1a2d7"
376 self.cr_serv_sample_data["input"]["end-point"][1]["layer-protocol-name"] = "DSR"
377 self.cr_serv_sample_data["input"]["end-point"][1]["service-interface-point"]["service-interface-point-uuid"] = "25812ef2-625d-3bf8-af55-5e93946d1c22"
378 self.cr_serv_sample_data["input"]["connectivity-constraint"]["service-layer"] = "DSR"
379 self.cr_serv_sample_data["input"]["connectivity-constraint"]["requested-capacity"]["total-size"]["value"] = "10"
380 self.cr_serv_sample_data["input"]["connectivity-constraint"]["service-level"] = self.uuid_services.odu
382 response = test_utils.tapi_create_connectivity_request(self.cr_serv_sample_data)
383 time.sleep(self.WAITING)
384 self.assertEqual(response.status_code, requests.codes.ok)
385 res = response.json()
386 self.uuid_services.dsr = res['output']['service']['uuid']
387 # pylint: disable=consider-using-f-string
388 print("dsr service uuid : {}".format(self.uuid_services.dsr))
390 input_dict_1 = {'administrative-state': 'LOCKED',
391 'lifecycle-state': 'PLANNED',
392 'operational-state': 'DISABLED',
393 'service-type': 'POINT_TO_POINT_CONNECTIVITY',
394 'service-layer': 'DSR',
395 'connectivity-direction': 'BIDIRECTIONAL'
397 input_dict_2 = {'value-name': 'OpenROADM node id',
398 'value': 'SPDR-SC1-XPDR1'}
399 input_dict_3 = {'value-name': 'OpenROADM node id',
400 'value': 'SPDR-SA1-XPDR1'}
402 self.assertDictEqual(dict(input_dict_1,
403 **res['output']['service']),
404 res['output']['service'])
405 self.assertDictEqual(dict(input_dict_2,
406 **res['output']['service']['end-point'][0]['name'][0]),
407 res['output']['service']['end-point'][0]['name'][0])
408 self.assertDictEqual(dict(input_dict_3,
409 **res['output']['service']['end-point'][1]['name'][0]),
410 res['output']['service']['end-point'][1]['name'][0])
411 # The sleep here is okey as the DSR service creation is very fast
412 time.sleep(self.WAITING)
414 def test_20_get_service_DSR(self):
415 response = test_utils.get_service_list_request(
416 "services/" + str(self.uuid_services.dsr))
417 self.assertEqual(response.status_code, requests.codes.ok)
418 res = response.json()
420 res['services'][0]['administrative-state'], 'inService')
422 res['services'][0]['service-name'], self.uuid_services.dsr)
424 res['services'][0]['connection-type'], 'service')
426 res['services'][0]['lifecycle-state'], 'planned')
429 def test_21_get_connectivity_service_list(self):
430 response = test_utils.tapi_get_service_list_request()
431 self.assertEqual(response.status_code, requests.codes.ok)
432 res = response.json()
433 liste_service = res['output']['service']
434 for ele in liste_service:
435 if ele['uuid'] == self.uuid_services.pm:
436 self.assertEqual(ele['operational-state'], 'ENABLED')
437 self.assertEqual(ele['service-layer'], 'PHOTONIC_MEDIA')
438 nbconnection = len(ele['connection'])
439 self.assertEqual(nbconnection, 3, 'There should be 3 connections')
440 elif ele['uuid'] == self.uuid_services.odu:
441 self.assertEqual(ele['operational-state'], 'ENABLED')
442 self.assertEqual(ele['service-layer'], 'ODU')
443 nbconnection = len(ele['connection'])
444 self.assertEqual(nbconnection, 1, 'There should be 1 connections')
445 elif ele['uuid'] == self.uuid_services.dsr:
446 self.assertEqual(ele['operational-state'], 'ENABLED')
447 self.assertEqual(ele['service-layer'], 'DSR')
448 nbconnection = len(ele['connection'])
449 self.assertEqual(nbconnection, 2, 'There should be 2 connections')
451 self.fail("get connectivity service failed")
454 def test_22_delete_connectivity_service_DSR(self):
455 response = test_utils.tapi_delete_connectivity_request(self.uuid_services.dsr)
456 self.assertEqual(response.status_code, requests.codes.no_content)
457 time.sleep(self.WAITING)
459 def test_23_delete_connectivity_service_ODU(self):
460 response = test_utils.tapi_delete_connectivity_request(self.uuid_services.odu)
461 self.assertEqual(response.status_code, requests.codes.no_content)
462 time.sleep(self.WAITING)
464 def test_24_delete_connectivity_service_PhotonicMedia(self):
465 response = test_utils.tapi_delete_connectivity_request(self.uuid_services.pm)
466 self.assertEqual(response.status_code, requests.codes.no_content)
467 time.sleep(self.WAITING)
469 def test_25_get_no_tapi_services(self):
470 response = test_utils.tapi_get_service_list_request()
471 res = response.json()
473 {"error-type": "rpc", "error-tag": "operation-failed",
474 "error-message": "No services exist in datastore",
475 "error-info": "<severity>error</severity>"},
476 res['errors']['error'])
479 def test_26_get_no_openroadm_services(self):
480 response = test_utils.get_service_list_request("")
481 self.assertEqual(response.status_code, requests.codes.conflict)
482 res = response.json()
484 {"error-type": "application", "error-tag": "data-missing",
485 "error-message": "Request could not be completed because the relevant data model content does not exist"},
486 res['errors']['error'])
489 def test_27_disconnect_spdrA(self):
490 response = test_utils.unmount_device("SPDR-SA1")
491 self.assertEqual(response.status_code, requests.codes.ok,
492 test_utils.CODE_SHOULD_BE_200)
494 def test_28_disconnect_spdrC(self):
495 response = test_utils.unmount_device("SPDR-SC1")
496 self.assertEqual(response.status_code, requests.codes.ok,
497 test_utils.CODE_SHOULD_BE_200)
499 def test_29_disconnect_roadmA(self):
500 response = test_utils.unmount_device("ROADM-A1")
501 self.assertEqual(response.status_code, requests.codes.ok,
502 test_utils.CODE_SHOULD_BE_200)
504 def test_30_disconnect_roadmC(self):
505 response = test_utils.unmount_device("ROADM-C1")
506 self.assertEqual(response.status_code, requests.codes.ok,
507 test_utils.CODE_SHOULD_BE_200)
510 if __name__ == "__main__":
511 unittest.main(verbosity=2)