Add new network after loading a template - cypress test
[vid.git] / vid-webpack-master / cypress / integration / iFrames / instantiation-templates.e2e.ts
1 import * as _ from "lodash";
2 import {PropertyPath} from "lodash";
3
4 describe('Drawing Board: Instantiation Templates', function () {
5
6   describe('Instantiation templates ', () => {
7
8     beforeEach(() => {
9       cy.clearSessionStorage();
10       cy.setTestApiParamToGR();
11       cy.initAAIMock();
12       cy.initGetAAISubDetails();
13       cy.initVidMock();
14       cy.initDrawingBoardUserPermission();
15       cy.login();
16
17       mockAsyncBulkResponse();
18     });
19
20     afterEach(() => {
21       cy.screenshot();
22     });
23
24     describe('Load Page and Deploy', () => {
25
26       it(`Given a stored template - when click "deploy" - then a coherent request should be sent upon deploy`, () => {
27
28         loadDrawingBoardWithRecreateMode();
29
30         // Then...
31         cy.getElementByDataTestsId("node-vProbe_NC_VNF 0").should('be.visible');
32         assertThatBodyFromDeployRequestEqualsToTemplateFromBackEnd();
33       });
34
35       it('Given a template - User can remove existing VNF', () => {
36
37         loadDrawingBoardWithRecreateMode();
38
39         removeVNFWithVFModules('node-21ae311e-432f-4c54-b855-446d0b8ded72-vProbe_NC_VNF 0');
40
41         cy.getDrawingBoardDeployBtn().click();
42         cy.wait('@expectedPostAsyncInstantiation').then(xhr => {
43           cy.deepCompare(bodyOf(xhr).vnfs, {});
44         });
45
46       });
47
48       it('Given a template - User can add new VNF', () => {
49         loadDrawingBoardWithRecreateMode();
50         // add new node
51         addNewNode('node-vProbe_NC_VNF 0-add-btn')
52           .fillVnfPopup()
53           .getDrawingBoardDeployBtn().click()
54           .wait('@expectedPostAsyncInstantiation').then(xhr => {
55             const vnfRequest = bodyOf(xhr).vnfs['vProbe_NC_VNF 0_1'];
56
57             expect(vnfRequest.action).equals("Create");
58             expect(vnfRequest.rollbackOnFailure).equals("true");
59             expect(vnfRequest.originalName).equals("vProbe_NC_VNF 0");
60             expect(vnfRequest.productFamilyId).equals("a9a77d5a-123e-4ca2-9eb9-0b015d2ee0fb");
61             expect(vnfRequest.lcpCloudRegionId).equals("hvf6");
62             expect(vnfRequest.lineOfBusiness).equals("zzz1");
63             expect(vnfRequest.platformName).equals("xxx1");
64             expect(vnfRequest.tenantId).equals("229bcdc6eaeb4ca59d55221141d01f8e");
65         });
66       });
67
68       it('Given a template - User can Duplicate VNF', () => {
69         const numberOfDuplicate: number = 4;
70         loadDrawingBoardWithRecreateMode();
71         nodeAction('node-21ae311e-432f-4c54-b855-446d0b8ded72-vProbe_NC_VNF 0', 'Duplicate')
72           .getElementByDataTestsId('duplicate-amount-vfmodules').select(numberOfDuplicate.toString())
73           .getTagElementContainsText('button', 'Duplicate').click()
74           .getDrawingBoardDeployBtn().click()
75           .wait('@expectedPostAsyncInstantiation').then(xhr => {
76             expect(Object.keys(bodyOf(xhr).vnfs).length).equals(numberOfDuplicate + 1);
77         });
78       });
79
80       it('Given a stored template - when "edit" vnf and vfmodules are opened - then template’s details are visible as expected and deploy without changes', () => {
81
82         loadDrawingBoardWithRecreateMode();
83
84         // Then...
85         editNode("node-21ae311e-432f-4c54-b855-446d0b8ded72-vProbe_NC_VNF 0")
86           .getElementByDataTestsId("instanceName").should('have.value', 'hvf6arlba007')
87           .getElementByDataTestsId("productFamily").should('contain', 'Emanuel')
88           .getElementByDataTestsId("tenant").should('contain', 'DN5242-Nov21-T1')
89           .getElementByDataTestsId("lcpRegion").should('contain', 'hvf6')
90           .getElementByDataTestsId("lineOfBusiness").should('contain', 'zzz1')
91           .getElementByDataTestsId("rollback").should('contain', 'Rollback')
92           .checkPlatformValue('xxx1')
93           .getElementByDataTestsId("cancelButton").click();
94
95         editNode("node-c5b26cc1-a66f-4b69-aa23-6abc7c647c88-vprobe_nc_vnf0..VprobeNcVnf..FE_base_module..module-0")
96           .getElementByDataTestsId("instanceName").should('have.value', 'hvf6arlba007_lba_Base_01')
97           .getElementByDataTestsId("lcpRegion").should('contain', 'hvf6')
98           .getElementByDataTestsId("tenant").should('contain', 'DN5242-Nov21-T1')
99           .getElementByDataTestsId("rollback").should('contain', 'Rollback')
100           .getElementByDataTestsId("cancelButton").click();
101
102         editNode("node-c09e4530-8fd8-418f-9483-2f57ce927b05-vprobe_nc_vnf0..VprobeNcVnf..FE_Add_On_Module_vlbagent_eph..module-1")
103           .getElementByDataTestsId("instanceName").should('have.value', 'my_hvf6arlba007_lba_dj_01')
104           .getElementByDataTestsId("volumeGroupName").should('have.value', 'my_special_hvf6arlba007_lba_dj_01_vol')
105           .getElementByDataTestsId("lcpRegion").should('contain', 'hvf6')
106           .getElementByDataTestsId("tenant").should('contain', 'DN5242-Nov21-T1')
107           .getElementByDataTestsId("rollback").should('contain', 'Rollback')
108           .getElementByDataTestsId("sdncPreLoad").should('have.value', 'on')
109           .getElementByDataTestsId("cancelButton").click();
110
111         assertThatBodyFromDeployRequestEqualsToTemplateFromBackEnd([
112           {path: [...vnfPath, "vnfStoreKey"], value: "vProbe_NC_VNF 0"}, // side-effect
113         ]);
114       });
115
116       it(`Given a stored template - when "edit" service is opened - then template’s details are visible as expected`, function () {
117
118         loadDrawingBoardWithRecreateMode();
119
120         cy.openServiceContextMenu()
121           .getElementByDataTestsId("context-menu-header-edit-item").click()
122           .getElementByDataTestsId("instanceName").should('have.value', 'vProbe_NC_Service_DG_new_SI')
123           .getElementByDataTestsId("subscriberName").should('contain', 'SILVIA ROBBINS')
124           .getElementByDataTestsId("serviceType").should('contain', 'TYLER SILVIA')
125           .getElementByDataTestsId("owningEntity").should('contain', 'WayneHolland')
126           .getElementByDataTestsId("project").should('contain', 'WATKINS')
127           .getElementByDataTestsId("rollback").should('contain', 'Rollback');
128
129       });
130
131       it(`Given a stored template - add one VfModule, edit its details, and deploy - deploy is added with the vfModule details`, () => {
132         loadDrawingBoardWithRecreateMode();
133
134         let newVfModuleName = "new.vfmodule.name";
135         let module1ModelId = "VprobeNcVnf..FE_Add_On_Module_vlbagent_eph..module-1";
136         let module1CustomizationId = `vprobe_nc_vnf0..${module1ModelId}`;
137
138         // Click target VNF on right tree
139         cy.getElementByDataTestsId('node-21ae311e-432f-4c54-b855-446d0b8ded72-vProbe_NC_VNF 0').click();
140
141         // Click [+] vfModule on left tree
142         cy.drawingBoardPressAddButtonByElementName(`node-${module1CustomizationId}`)
143           .click({force: true});
144
145         editNode(`node-c09e4530-8fd8-418f-9483-2f57ce927b05-${module1CustomizationId}`, 1);
146           cy.clearInput("instanceName");
147           cy.typeToInput("instanceName", newVfModuleName);
148           cy.selectDropdownOptionByText('lcpRegion', 'hvf6');
149           cy.selectDropdownOptionByText('tenant', 'DN5242-Nov21-T1');
150           cy.getElementByDataTestsId('form-set').click();
151
152         // Then...
153         cy.getReduxState().then((state) => {
154           let vfModules_1Path = [
155             ...vnfPath, "vfModules", module1CustomizationId,
156           ];
157
158           let serviceInstanceElementOnRedux = state.service.serviceInstance[(templateWithVnfSetup.serviceModelId)];
159           let latestVfModule_1Path = findPathOfLatestVfModule(serviceInstanceElementOnRedux, vfModules_1Path);
160
161           // This is a funny merge, as values are already there, but that way ensures
162           // the values that selected are really deployed, while limiting the cost of
163           // maintenance, by taking other vfModule's fields as granted.
164           let latestVfModule_1ExpectedValue = _.merge(
165             _.get(serviceInstanceElementOnRedux, latestVfModule_1Path),
166             {
167               instanceName: newVfModuleName,
168               volumeGroupName: `${newVfModuleName}_vol`,
169               lcpCloudRegionId: "hvf6",
170               tenantId: "4914ab0ab3a743e58f0eefdacc1dde77",
171             }
172           );
173
174           assertThatBodyFromDeployRequestEqualsToTemplateFromBackEnd([
175             {path: [...vnfPath, "vnfStoreKey"], value: "vProbe_NC_VNF 0"},   // side-effect
176             {path: ["existingNames", newVfModuleName], value: ""},
177             {path: ["existingNames", `${newVfModuleName}_vol`], value: ""},
178             {path: latestVfModule_1Path, value: latestVfModule_1ExpectedValue},
179             {path: ["validationCounter"], value: null},  // side-effect
180           ]);
181         });
182
183       });
184
185       it('Given a template - User can remove existing vfmodule', function () {
186
187         loadDrawingBoardWithRecreateMode();
188         nodeAction('node-c09e4530-8fd8-418f-9483-2f57ce927b05-vprobe_nc_vnf0..VprobeNcVnf..FE_Add_On_Module_vlbagent_eph..module-1', 'Remove');
189         let removed_vfModule_Path = [
190           ...vnfPath, "vfModules",
191           "vprobe_nc_vnf0..VprobeNcVnf..FE_Add_On_Module_vlbagent_eph..module-1",
192         ];
193
194         assertThatBodyFromDeployRequestEqualsToTemplateFromBackEnd([
195           {path: [...vnfPath, "vnfStoreKey"], value: "vProbe_NC_VNF 0"}, // side-effect
196           {path: [...removed_vfModule_Path], value: undefined},
197         ]);
198       });
199
200       [
201         {desc: "with changes", modifySomeValues: true},
202         {desc: "without changes", modifySomeValues: false},
203       ].forEach((testCase) => {
204
205         it(`Given a stored template - edit service vnf and vfmodule ${testCase.desc} - deploy request should be ${testCase.desc}`, function () {
206
207           loadDrawingBoardWithRecreateMode();
208
209           //edit service
210           cy.openServiceContextMenu();
211           cy.getElementByDataTestsId("context-menu-header-edit-item").click();
212           if (testCase.modifySomeValues) {
213             cy.clearInput("instanceName");
214             cy.typeToInput("instanceName", "different.instance.name");
215           }
216           cy.getElementByDataTestsId('form-set').click();
217
218           // edit vnf
219           editNode("node-21ae311e-432f-4c54-b855-446d0b8ded72-vProbe_NC_VNF 0");
220           if (testCase.modifySomeValues) {
221             cy.selectPlatformValue('platform');
222             cy.selectDropdownOptionByText("tenant", "CESAR-100-D-spjg61909");
223           }
224           cy.getElementByDataTestsId('form-set').click();
225
226           //edit vf module
227           editNode("node-c5b26cc1-a66f-4b69-aa23-6abc7c647c88-vprobe_nc_vnf0..VprobeNcVnf..FE_base_module..module-0");
228           if (testCase.modifySomeValues) {
229             cy.getElementByDataTestsId('sdncPreLoad').click();
230           }
231           cy.getElementByDataTestsId('form-set').click();
232
233           // Then...
234           let vfModule_0Path = [
235             ...vnfPath, "vfModules",
236             "vprobe_nc_vnf0..VprobeNcVnf..FE_base_module..module-0",
237             "vprobe_nc_vnf0..VprobeNcVnf..FE_base_module..module-0ahubg",
238           ];
239
240           assertThatBodyFromDeployRequestEqualsToFile(testCase.modifySomeValues ? [
241             {path: ["instanceName"], value: "different.instance.name"},
242             {path: ["existingNames", "vprobe_nc_service_dg_new_si"], value: undefined},
243             {path: ["existingNames", "different.instance.name"], value: ""},
244
245             {path: [...vnfPath, "platformName"], value: "xxx1,platform"},
246             {path: [...vnfPath, "tenantId"], value: "f2f3830e4c984d45bcd00e1a04158a79"},
247
248             {path: [...vfModule_0Path, "sdncPreLoad"], value: true},
249           ] : []);
250         })
251
252       });
253
254       it(`Given a stored template of Network - - it is loaded`,  () => {
255
256         loadDrawingBoardWithRecreateModeNetwork();
257
258         // Then...
259         cy.getElementByDataTestsId("node-SR-IOV Provider 2-1").should('be.visible');
260         cy.getElementByDataTestsId("node-SR-IOV Provider 2-2").should('be.visible');
261         assertThatBodyFromDeployRequestEqualsToTemplateFromBackEnd_network();
262       });
263
264       it(`Given a stored template of Network - User can remove existing network`,  () => {
265
266         loadDrawingBoardWithRecreateModeNetwork();
267
268         nodeAction('node-01f4c475-3f89-4f00-a2f4-39a873dba0ae-SR-IOV Provider 2-1', 'Remove');
269         let removed_network_Path = [
270           "networks", "SR-IOV Provider 2-1",
271         ];
272
273         let removed_network_counter_Path = [
274           "existingNetworksCounterMap", "f6b6d141-0d4c-427d-ad35-797f3d1abe71",
275         ];
276
277         assertThatBodyFromDeployRequestEqualsToTemplateFromBackEnd_network([
278           {path: removed_network_Path, value: undefined},
279           {path: removed_network_counter_Path, value: 0},
280         ]);
281       });
282
283       it('Given a template - User can add a new network', () => {
284
285         loadDrawingBoardWithRecreateModeNetwork();
286
287         // add new node
288         addNewNode('node-SR-IOV Provider 2-1-add-btn')
289         .fillNetworkPopup()
290         .getDrawingBoardDeployBtn().click()
291         .wait('@expectedPostAsyncInstantiation').then(xhr => {
292           const networkRequest = bodyOf(xhr).networks['SR-IOV Provider 2-1_1'];
293
294           expect(networkRequest.action).equals("Create");
295           expect(networkRequest.rollbackOnFailure).equals("true");
296           expect(networkRequest.originalName).equals("SR-IOV Provider 2-1");
297           expect(networkRequest.productFamilyId).equals("a9a77d5a-123e-4ca2-9eb9-0b015d2ee0fb");
298           expect(networkRequest.lcpCloudRegionId).equals("hvf6");
299           expect(networkRequest.lineOfBusiness).equals("zzz1");
300           expect(networkRequest.platformName).equals("xxx1");
301           expect(networkRequest.tenantId).equals("229bcdc6eaeb4ca59d55221141d01f8e");
302         });
303       });
304     });
305   });
306 });
307
308 let apiTestResources = '../vid-automation/src/test/resources/asyncInstantiation/';
309
310 const templateWithVnfSetup = {
311   serviceModelId: '6cfeeb18-c2b0-49df-987a-da47493c8e38',
312   instanceTemplateFile: apiTestResources + 'templates__instance_template.json',
313   instanceTemplateSetWithoutModifyFile: apiTestResources + 'templates__instance_from_template__set_without_modify1.json',
314   serviceModelFile: '../support/jsonBuilders/mocks/jsons/instantiationTemplates/templates__service_model.json',
315 };
316
317 const templateWithNetworkSetup = {
318   serviceModelId: 'a1a14610-ee40-4049-8007-0608a20dd1fa',
319   instanceTemplateFile: apiTestResources + 'templates__instance_template_network.json',
320   serviceModelFile: '../support/jsonBuilders/mocks/jsons/instantiationTemplates/templates__service_model_network.json',
321 };
322
323 const vnfPath = [
324   "vnfs", "vProbe_NC_VNF 0"
325 ];
326
327 function loadDrawingBoardWithRecreateMode() {
328   loadDrawingBoardWithRecreateModeInternal(
329     '../../' + templateWithVnfSetup.instanceTemplateFile,
330     templateWithVnfSetup.serviceModelId,
331     templateWithVnfSetup.serviceModelFile);
332 }
333
334 function loadDrawingBoardWithRecreateModeNetwork() {
335   loadDrawingBoardWithRecreateModeInternal(
336     '../../' + templateWithNetworkSetup.instanceTemplateFile,
337     templateWithNetworkSetup.serviceModelId,
338     templateWithNetworkSetup.serviceModelFile);
339 }
340
341 function loadDrawingBoardWithRecreateModeInternal(instanceTemplate: string, serviceModelIdToLoad: any, serviceModel: string) {
342   const templateUuid = "46390edd-7100-46b2-9f18-419bd24fb60b";
343
344   const drawingBoardAction = `RECREATE`;
345   const templateTopologyEndpoint = "templateTopology";
346   cy.route(`**/rest/models/services/${serviceModelIdToLoad}`,
347     'fixture:' + serviceModel)
348     .as('serviceModel');
349
350   cy.route(`**/instantiationTemplates/${templateTopologyEndpoint}/${templateUuid}`,
351     'fixture:' + instanceTemplate)
352     .as('templateTopology');
353
354   // When...
355
356   cy.openIframe(`app/ui/#/servicePlanning/${drawingBoardAction}` +
357     `?jobId=${templateUuid}` +
358     `&serviceModelId=${serviceModelIdToLoad}`);
359
360   cy.wait('@serviceModel');
361   cy.wait('@templateTopology');
362 }
363
364 function nodeAction(dataTestId: string, action: string, index ?: number) {
365   return cy.drawingBoardTreeOpenContextMenuByElementDataTestId(dataTestId, index)
366     .drawingBoardTreeClickOnContextMenuOptionByName(action)
367 }
368
369 function editNode(dataTestId: string, index ?: number) {
370   return nodeAction(dataTestId, 'Edit', index);
371 }
372
373 function addNewNode(dataTestId: string) {
374   return cy.getElementByDataTestsId(dataTestId).click({force: true})
375 }
376
377 function removeVNFWithVFModules(dataTestId: string) {
378   return nodeAction(dataTestId, 'Remove')
379     .getTagElementContainsText('button', 'Remove VNF').click()
380 }
381
382 function assertThatBodyFromDeployRequestEqualsToTemplateFromBackEnd(deviationFromExpected: { path: PropertyPath, value: any }[] = []) {
383   assertThatBodyFromDeployRequestEqualsToTemplateFromBackEndInternal(templateWithVnfSetup.instanceTemplateFile, deviationFromExpected);
384 }
385
386 function assertThatBodyFromDeployRequestEqualsToTemplateFromBackEnd_network(deviationFromExpected: { path: PropertyPath, value: any }[] = []) {
387   assertThatBodyFromDeployRequestEqualsToTemplateFromBackEndInternal(templateWithNetworkSetup.instanceTemplateFile, deviationFromExpected);
388 }
389
390 function assertThatBodyFromDeployRequestEqualsToTemplateFromBackEndInternal(filePathOfExpected: string, deviationFromExpected: { path: PropertyPath; value: any }[]) {
391   cy.getDrawingBoardDeployBtn().click();
392   cy.wait('@expectedPostAsyncInstantiation').then(xhr => {
393     cy.readFile(filePathOfExpected).then((expectedResult) => {
394       convertRollbackOnFailureValueFromStringToBoolean(expectedResult);
395
396       let xhrBodyWithoutIsDirtyField = removeIsDirtyFieldFromXhrRequestBody(xhr);
397       setDeviationInExpected(expectedResult, deviationFromExpected);
398       cy.deepCompare(xhrBodyWithoutIsDirtyField, expectedResult);
399     });
400   });
401 }
402
403
404 function assertThatBodyFromDeployRequestEqualsToFile(deviationFromExpected: { path: PropertyPath, value: any }[] = []) {
405   cy.getDrawingBoardDeployBtn().click();
406   cy.wait('@expectedPostAsyncInstantiation').then(xhr => {
407
408     cy.readFile(templateWithVnfSetup.instanceTemplateSetWithoutModifyFile).then((expectedResult) => {
409       setDeviationInExpected(expectedResult, deviationFromExpected);
410       cy.deepCompare(xhr.request.body, expectedResult);
411     });
412
413   });
414 }
415
416 function bodyOf(xhr: Cypress.WaitXHR) {
417   return JSON.parse(JSON.stringify(xhr.request.body));
418 }
419
420 function setDeviationInExpected(expectedResult: any, deviations: { path: PropertyPath; value: any }[]) {
421   for (const deviation of deviations) {
422     _.set(expectedResult, deviation.path, deviation.value);
423   }
424 }
425
426 function findPathOfLatestVfModule(serviceInstanceElementFromRedux: any, vfModulesContainerPath: string[]) {
427   let latestVfModuleRandomlySelectedKey: string = _.last(_.keys(
428     _.get(serviceInstanceElementFromRedux, vfModulesContainerPath)
429   )) as string;
430
431   return [...vfModulesContainerPath, latestVfModuleRandomlySelectedKey];
432 }
433
434 //We use this function because the deployService() on drawing-board-header.component class
435 // changes rollbackOnFailure value from string type to boolean.
436 function convertRollbackOnFailureValueFromStringToBoolean(expectedResult: any) {
437   expectedResult.rollbackOnFailure = Boolean(expectedResult.rollbackOnFailure);
438 }
439
440 function removeIsDirtyFieldFromXhrRequestBody(xhr: any) {
441   let xhrTempBody = bodyOf(xhr);
442   delete xhrTempBody.isDirty;
443   return xhrTempBody;
444 }
445
446 function mockAsyncBulkResponse() {
447   cy.server().route({
448     url: Cypress.config('baseUrl') + '/asyncInstantiation/bulk',
449     method: 'POST',
450     status: 200,
451     response: "[]",
452   }).as("expectedPostAsyncInstantiation");
453 }