Fixing the issue which prevents from clicking the [+] button on templates
[vid.git] / vid-webpack-master / src / app / drawingBoard / service-planning / objectsToTree / models / vfModule / vfModule.model.info.ts
1 import {DynamicInputsService} from "../../dynamicInputs.service";
2 import {ILevelNodeInfo} from "../basic.model.info";
3 import * as _ from "lodash";
4 import {VNFModel} from "../../../../../shared/models/vnfModel";
5 import {SharedTreeService} from "../../shared.tree.service";
6 import {VfModuleTreeNode} from "../../../../../shared/models/vfModuleTreeNode";
7 import {VfModuleInstance} from "../../../../../shared/models/vfModuleInstance";
8 import {VfModule} from "../../../../../shared/models/vfModule";
9 import {NgRedux} from "@angular-redux/store";
10 import {ITreeNode} from "angular-tree-component/dist/defs/api";
11 import {
12   GenericFormPopupComponent,
13   PopupType
14 } from "../../../../../shared/components/genericFormPopup/generic-form-popup.component";
15 import {DialogService} from "ng2-bootstrap-modal";
16 import {VfModulePopupService} from "../../../../../shared/components/genericFormPopup/genericFormServices/vfModule/vfModule.popup.service";
17 import {AppState} from "../../../../../shared/store/reducers";
18 import {MessageBoxData} from "../../../../../shared/components/messageBox/messageBox.data";
19 import {MessageBoxService} from "../../../../../shared/components/messageBox/messageBox.service";
20 import {AvailableNodeIcons} from "../../../available-models-tree/available-models-tree.service";
21 import {IframeService} from "../../../../../shared/utils/iframe.service";
22 import {
23   deleteActionVfModuleInstance,
24   deleteVFModuleField,
25   removeVfModuleInstance,
26   undoDeleteVfModuleInstance,
27   undoUgradeVFModule,
28   updateVFModulePosition,
29   upgradeVFModule
30 } from "../../../../../shared/storeUtil/utils/vfModule/vfModule.actions";
31 import {ComponentInfoService} from "../../../component-info/component-info.service";
32 import {ComponentInfoType} from "../../../component-info/component-info-model";
33 import {ModelInformationItem} from "../../../../../shared/components/model-information/model-information.component";
34 import {VfModuleUpgradePopupService} from "../../../../../shared/components/genericFormPopup/genericFormServices/vfModuleUpgrade/vfModule.upgrade.popuop.service";
35 import {FeatureFlagsService, Features} from "../../../../../shared/services/featureFlag/feature-flags.service";
36 import {Utils} from "../../../../../shared/utils/utils";
37
38 export class VFModuleModelInfo implements ILevelNodeInfo {
39   constructor(private _dynamicInputsService: DynamicInputsService,
40               private _sharedTreeService: SharedTreeService,
41               private _dialogService: DialogService,
42               private _vfModulePopupService: VfModulePopupService,
43               private _vfModuleUpgradePopupService: VfModuleUpgradePopupService,
44               private _iframeService: IframeService,
45               private _featureFlagsService: FeatureFlagsService,
46               private _store: NgRedux<AppState>,
47               private _componentInfoService: ComponentInfoService) {
48   }
49
50   name: string = 'vfModules';
51   type: string = 'Module';
52   typeName: string = 'M';
53   componentInfoType = ComponentInfoType.VFMODULE;
54
55   /***********************************************************
56    * return if user should provide instance name or not.
57    *        get info from parent (VNF)
58    * @param currentModel - current Model object
59    * @param parentModel - current parent Model object
60    ************************************************************/
61   isEcompGeneratedNaming(currentModel, parentModel): boolean {
62     const ecompGeneratedNaming = !_.isNil(parentModel.properties) ? parentModel.properties.ecomp_generated_naming : undefined;
63     return ecompGeneratedNaming === "true";
64   }
65
66   /***********************************************************
67    * return model dynamic inputs
68    * @param currentModel - current Model object
69    ************************************************************/
70   updateDynamicInputsDataFromModel = (currentModel): any => {
71     let displayInputs = _.isNil(currentModel) ? [] : currentModel.inputs;
72     return _.isEmpty(displayInputs) ? [] : this._dynamicInputsService.getArbitraryInputs(displayInputs);
73   };
74
75   /***********************************************************
76    * return vfModule model
77    * @param vfModuleModelId - current Model id
78    * @param instance
79    * @param serviceHierarchy - serviceHierarchy
80    ************************************************************/
81   getModel = (vfModuleModelId: string, instance, serviceHierarchy): Partial<VfModule> => {
82     if (!_.isNil(serviceHierarchy)) {
83       if (!_.isNil(serviceHierarchy[this.name]) && !_.isNil(serviceHierarchy[this.name][vfModuleModelId])) {
84         return new VfModule(serviceHierarchy[this.name][vfModuleModelId], this._featureFlagsService.getAllFlags());
85       }
86     }
87     return {};
88   };
89
90   createNode(instance: VfModuleInstance, currentModel: VfModule, parentModel: VNFModel, modelName: string, index: number): VfModuleTreeNode {
91     let dynamicModelName = Object.keys(instance)[index];
92     instance = instance[Object.keys(instance)[index]];
93     const isEcompGeneratedNaming: boolean = this.isEcompGeneratedNaming(currentModel, parentModel);
94
95     const dynamicInputs = this.updateDynamicInputsDataFromModel(currentModel);
96     let newVfModule: VfModuleTreeNode = new VfModuleTreeNode(instance, currentModel, modelName, dynamicInputs, isEcompGeneratedNaming, dynamicModelName);
97
98     newVfModule.missingData = this._sharedTreeService.hasMissingData(instance, newVfModule.dynamicInputs, isEcompGeneratedNaming, []);
99     newVfModule.typeName = this.typeName;
100     newVfModule.menuActions = this.getMenuAction(<any>newVfModule, currentModel.uuid);
101     newVfModule.isFailed = _.isNil(instance.isFailed) ? false : instance.isFailed;
102     newVfModule.statusMessage = !_.isNil(instance.statusMessage) ? instance.statusMessage : "";
103
104     newVfModule = this._sharedTreeService.addingStatusProperty(newVfModule);
105     return newVfModule;
106   }
107
108   createInstanceTreeNode(instance: VfModuleInstance, currentModel: VfModule, parentModel: VNFModel, modelName: string): VfModuleTreeNode | VfModuleTreeNode[] {
109     let numberOfChilds = Object.keys(instance).length;
110     if (numberOfChilds > 1) {
111       let result: VfModuleTreeNode[] = [];
112       for (let i = 0; i < numberOfChilds; i++) {
113         result.push(this.createNode(instance, currentModel, parentModel, modelName, i));
114       }
115       return result;
116     } else {
117       return this.createNode(instance, currentModel, parentModel, modelName, 0);
118     }
119   }
120
121   /***********************************************************
122    * return next level object (null because is last level)
123    ************************************************************/
124   getNextLevelObject(): any {
125     return null;
126   }
127
128   getTooltip = (): string => 'VFmodule';
129
130   getType = (): string => 'VFmodule';
131
132   /***********************************************************
133    * return if instance has missing data
134    * @param instance - vnf instance
135    * @param dynamicInputs
136    * @param isEcompGeneratedNaming
137    ************************************************************/
138   hasMissingData(instance, dynamicInputs: any, isEcompGeneratedNaming: boolean): boolean {
139     return this._sharedTreeService.hasMissingData(instance, dynamicInputs, isEcompGeneratedNaming, []);
140   }
141
142
143   /***********************************************************
144    * return if instance has missing data
145    * @param node - VFModule node
146    * @param serviceModelId - current service id
147    ************************************************************/
148   onClickAdd(node: ITreeNode, serviceModelId: string): void {
149     const vnfStoreKey = this._sharedTreeService.getSelectedVNF() || this.getDefaultVNF(node.parent, serviceModelId);
150     if (vnfStoreKey) {
151       this._dialogService.addDialog(GenericFormPopupComponent, {
152         type: PopupType.VF_MODULE,
153         uuidData: <any>{
154           serviceId: serviceModelId,
155           modelName: node.data.name,
156           vFModuleStoreKey: null,
157           vnfStoreKey: vnfStoreKey,
158           modelId: node.data.modelId,
159           type: node.data.type,
160           popupService: this._vfModulePopupService
161         },
162         node: node,
163         isUpdateMode: false
164       });
165     } else {
166       let messageBoxData: MessageBoxData = new MessageBoxData(
167         "Select a parent",  // modal title
168         "There are multiple instances on the right side that can contain this vf-module Please select the VNF instance, to add this vf-module to, on the right side and then click the + sign",
169         <any>"warning",
170         <any>"md",
171         [
172           {text: "Close", size: "medium", closeModal: true}
173         ]);
174       MessageBoxService.openModal.next(messageBoxData);
175     }
176   }
177
178   getDefaultVNF(node: ITreeNode, serviceModelId: string): string {
179     let keys = _.keys(_.pickBy(this._store.getState().service.serviceInstance[serviceModelId].vnfs, vnf => {
180       return (this._sharedTreeService.modelUniqueId(vnf) === node.data.modelUniqueId);
181     }));
182     return keys.length === 1 ? this._store.getState().service.serviceInstance[serviceModelId].vnfs[keys[0]].vnfStoreKey : null;
183   }
184
185   /***********************************************************
186    * return number of existing instances (in all VNF's)
187    * @param node - VfModule node
188    * @param serviceModelId - current service id
189    ************************************************************/
190   getNodeCount(node: ITreeNode, serviceModelId: string): number {
191     const vnfs = this._store.getState().service.serviceInstance[serviceModelId].vnfs;
192     let count: number = 0;
193     if (!_.isNil(this._store.getState().service.serviceInstance) && !_.isNil(this._store.getState().service.serviceInstance[serviceModelId])) {
194       const selectedVNF: string = this._sharedTreeService.getSelectedVNF();
195       if (selectedVNF) {
196         count += this.countNumberOfVFModule(vnfs[selectedVNF], node);
197       }else {
198         for (let vnfKey in vnfs) {
199           count += this.countNumberOfVFModule(vnfs[vnfKey], node);
200         }
201       }
202       return count;
203     }
204     return count;
205   }
206
207
208   countNumberOfVFModule(vnf, node): number {
209     let count = 0;
210     for (let vfModuleKey in vnf['vfModules']) {
211       for (let vfModule in vnf['vfModules'][vfModuleKey]) {
212         if (this._sharedTreeService.modelUniqueId(vnf['vfModules'][vfModuleKey][vfModule]) === node.data.modelUniqueId) {
213           const vfModuleObj = vnf['vfModules'][vfModuleKey][vfModule];
214           if (!(!_.isNil(vfModuleObj) && !_.isNil(vfModuleObj.action) && vfModuleObj.action.split('_').pop() === 'Delete')) count++;
215         }
216       }
217     }
218     return count;
219   }
220
221   getCountVFModuleOfSelectedVNF(node: ITreeNode, vnfStoreKey: string, serviceModelId: string): number {
222     let count: number = 0;
223     if (!_.isNil(this._store.getState().service.serviceInstance) && !_.isNil(this._store.getState().service.serviceInstance[serviceModelId])) {
224       const vnf = this._store.getState().service.serviceInstance[serviceModelId].vnfs[vnfStoreKey];
225       count += this.countNumberOfVFModule(vnf, node);
226       return count;
227     }
228     return count;
229   }
230
231
232   /***********************************************************
233    * should show node icon
234    * @param node - current ITrees node
235    * @param serviceModelId - service id
236    ************************************************************/
237   showNodeIcons(node: ITreeNode, serviceModelId: string): AvailableNodeIcons {
238     const selectedVNF: string = this._sharedTreeService.getSelectedVNF();
239     if (selectedVNF) {
240       return this.showVFModuleOnSelectedVNF(node, selectedVNF, serviceModelId);
241     } else {
242       const optionalSelected = this.getOptionalVNFs(serviceModelId, node.parent.data.modelUniqueId);
243       if (optionalSelected.length === 1) {
244         return this.showVFModuleOnSelectedVNF(node, optionalSelected[0].vnfStoreKey, serviceModelId);
245       } else {
246         return new AvailableNodeIcons(false, false);
247       }
248     }
249   }
250
251
252   showVFModuleOnSelectedVNF(node: ITreeNode, selectedVNF: string, serviceModelId: string): AvailableNodeIcons {
253     if (!_.isNil(this._store.getState().service.serviceInstance[serviceModelId].vnfs[selectedVNF])
254         && node.parent.data.modelUniqueId === this._sharedTreeService.modelUniqueId(this._store.getState().service.serviceInstance[serviceModelId].vnfs[selectedVNF])) {
255       const existingVFModules = this.getCountVFModuleOfSelectedVNF(node, selectedVNF, serviceModelId);
256       const reachedLimit = this.isVFModuleReachedLimit(node, this._store.getState().service.serviceHierarchy, serviceModelId, existingVFModules);
257       const showAddIcon = this._sharedTreeService.shouldShowAddIcon() && !reachedLimit;
258       return new AvailableNodeIcons(showAddIcon, reachedLimit);
259     }
260     return new AvailableNodeIcons(false, false);
261
262   }
263
264   getOptionalVNFs(serviceUUID: string, vnfModelUniqueId: string): any[] {
265     let result = [];
266     if (!_.isNil(this._store.getState().service.serviceInstance) && !_.isNil(this._store.getState().service.serviceInstance[serviceUUID])) {
267       const serviceVNFsInstances = this._store.getState().service.serviceInstance[serviceUUID].vnfs;
268       for (let vnfKey in serviceVNFsInstances) {
269         if (this._sharedTreeService.modelUniqueId(serviceVNFsInstances[vnfKey]) === vnfModelUniqueId) {
270           serviceVNFsInstances[vnfKey].vnfStoreKey = vnfKey;
271           result.push(serviceVNFsInstances[vnfKey]);
272         }
273       }
274     }
275     return result;
276   }
277
278
279   /************************************************
280    return number of instances with action Delete
281    @type: vnfs networks, vngGroups (only vfModule)
282    @node : node model from the left tree
283    ************************************************/
284   getExistingInstancesWithDeleteMode(node, serviceModelId: string, selectedVNF: string): number {
285     const existingInstances = this._store.getState().service.serviceInstance[serviceModelId].vnfs[selectedVNF].vfModules;
286     let counter = 0;
287     const modelUuid = node.data.uuid;
288
289     if (!_.isNil(existingInstances)) {
290       for (let instanceKey in existingInstances) {
291         let size = Object.keys(existingInstances[instanceKey]).length;
292         for (let i = 0; i < size; i++) {
293           let vfModuleDynamicKey = Object.keys(existingInstances[instanceKey])[i];
294           if (!_.isNil(existingInstances[instanceKey][vfModuleDynamicKey].action)) {
295             if (existingInstances[instanceKey][vfModuleDynamicKey]['uuid'] === modelUuid && existingInstances[instanceKey][vfModuleDynamicKey].action.split('_').pop() === 'Delete') {
296               counter++;
297             }
298           }
299         }
300
301       }
302     }
303     return counter;
304   }
305
306   isVFModuleReachedLimit(node: any, serviceHierarchy: any, serviceModelId: string, currentNodeCount: number): boolean {
307     const flags = this._featureFlagsService.getAllFlags();
308     let vnfModules = serviceHierarchy[serviceModelId].vfModules;
309     const maxInstances = vnfModules[node.data.name]
310       ? Utils.getMaxVfModule(vnfModules[node.data.name].properties, flags)
311       : null;
312     if (_.isNil(maxInstances)) {
313       return false;
314     }
315
316     return !(maxInstances > currentNodeCount);
317   }
318
319   getMenuAction(node: ITreeNode, serviceModelId: string): { [methodName: string]: { method: Function, visible: Function, enable: Function } } {
320     return {
321       edit: {
322         method: (node, serviceModelId) => {
323           this._iframeService.addClassOpenModal('content');
324           this._dialogService.addDialog(GenericFormPopupComponent, {
325             type: PopupType.VF_MODULE,
326             uuidData: <any>{
327               serviceId: serviceModelId,
328               modelName: node.data.modelName,
329               vFModuleStoreKey: node.data.dynamicModelName,
330               vnfStoreKey: node.parent.data.vnfStoreKey,
331               modelId: node.data.modelId,
332               type: node.data.type,
333               popupService: this._vfModulePopupService
334             },
335             node: node,
336             isUpdateMode: true
337           });
338         },
339         visible: (node) => this._sharedTreeService.shouldShowRemoveAndEdit(node),
340         enable: (node) => this._sharedTreeService.shouldShowRemoveAndEdit(node),
341       },
342       showAuditInfo: {
343         method: (node, serviceModelId) => {
344           let instance = this._store.getState().service.serviceInstance[serviceModelId].vnfs[node.parent.data.vnfStoreKey].vfModules[node.data.modelName][node.data.modelVersionId];
345           this._sharedTreeService.openAuditInfoModal(node, serviceModelId, instance, 'VFMODULE', this);
346         },
347         visible: (node) => this._sharedTreeService.shouldShowAuditInfo(node),
348         enable: (node) => this._sharedTreeService.shouldShowAuditInfo(node)
349       },
350       remove: {
351         method: (node, serviceModelId) => this._store.dispatch(removeVfModuleInstance(node.data.modelName, serviceModelId, node.parent.data.modelName, node.parent.data.vnfStoreKey, node.data.dynamicModelName)),
352         visible: (node) => this._sharedTreeService.shouldShowRemoveAndEdit(node),
353         enable: (node) => this._sharedTreeService.shouldShowRemoveAndEdit(node),
354       },
355       delete: {
356         method: (node, serviceModelId) => {
357           this._store.dispatch(deleteActionVfModuleInstance(node.data.dynamicModelName, node.parent.data.vnfStoreKey, serviceModelId))
358         },
359         visible: (node) => this._sharedTreeService.shouldShowDelete(node),
360         enable: (node) => this._sharedTreeService.shouldShowDelete(node)
361       },
362       undoDelete: {
363         method: (node, serviceModelId) => {
364           this._store.dispatch(undoDeleteVfModuleInstance(node.data.dynamicModelName, node.parent.data.vnfStoreKey, serviceModelId));
365           this._store.dispatch(deleteVFModuleField(node.data.modelName,  node.parent.data.vnfStoreKey, node.data.servicedId ,node.data.dynamicModelName, 'retainAssignments'));
366         },
367         visible: (node) => this._sharedTreeService.shouldShowUndoDelete(node),
368         enable: (node, serviceModelId) => this._sharedTreeService.shouldShowUndoDelete(node) && this._sharedTreeService.shouldShowDelete(node.parent) && !this._sharedTreeService.isServiceOnDeleteMode(serviceModelId)
369       },
370       upgrade: {
371         method: (node, serviceModelId) => {
372           this.upgradeVFM(serviceModelId, node);
373         },
374         visible: (node,serviceModelId) => {
375           return this._sharedTreeService.shouldShowUpgrade(node, serviceModelId);
376         },
377         enable:  (node, serviceModelId) => {
378           return this._sharedTreeService.shouldShowUpgrade(node, serviceModelId);
379         }
380       },
381       undoUpgrade: {
382         method: (node, serviceModelId) => {
383           this._sharedTreeService.undoUpgradeBottomUp(node, serviceModelId);
384           this._store.dispatch(undoUgradeVFModule(node.data.modelName, node.parent.data.vnfStoreKey, serviceModelId, node.data.dynamicModelName));
385         },
386         visible: (node) => {
387           return this._sharedTreeService.shouldShowUndoUpgrade(node);
388         },
389         enable: (node) => {
390           return this._sharedTreeService.shouldShowUndoUpgrade(node);
391         }
392       },
393     };
394   }
395
396   private upgradeVFM(serviceModelId, node) {
397     if (FeatureFlagsService.getFlagState(Features.FLAG_2002_VFM_UPGRADE_ADDITIONAL_OPTIONS, this._store)) {
398       this._iframeService.addClassOpenModal('content');
399       this._dialogService.addDialog(GenericFormPopupComponent, {
400         type: PopupType.VF_MODULE_UPGRADE,
401         uuidData: <any>{
402           serviceId: serviceModelId,
403           modelName: node.data.modelName,
404           vFModuleStoreKey: node.data.dynamicModelName,
405           vnfStoreKey: node.parent.data.vnfStoreKey,
406           modelId: node.data.modelId,
407           type: node.data.type,
408           popupService: this._vfModuleUpgradePopupService,
409           vfModule : _.cloneDeep(node)
410         },
411         node,
412         isUpdateMode: true
413       });
414     } else {
415       this._sharedTreeService.upgradeBottomUp(node, serviceModelId);
416       this._store.dispatch(upgradeVFModule(node.data.modelName,  node.parent.data.vnfStoreKey, serviceModelId ,node.data.dynamicModelName));
417     }
418   }
419
420   updatePosition(that, node, instanceId, parentStoreKey): void {
421     that.store.dispatch(updateVFModulePosition(node, instanceId, parentStoreKey));
422   }
423
424
425   getNodePosition(instance, deepDynamicName): number {
426     return (!_.isNil(instance) && !_.isNil(instance[deepDynamicName])) ? instance[deepDynamicName].position : null;
427   }
428
429   getInfo(model:Partial<VfModule>, instance): ModelInformationItem[] {
430     const modelInformation = !_.isEmpty(model) ? [
431       ModelInformationItem.createInstance("Base module", model.baseModule),
432       ModelInformationItem.createInstance("Min instances", !_.isNull(model.min) ? String(model.min) : null),
433       this._sharedTreeService.createMaximumToInstantiateModelInformationItem(model),
434       ModelInformationItem.createInstance("Initial instances count", !_.isNull(model.initial) ? String(model.initial) : null)
435     ] : [];
436
437     const instanceInfo = [];
438     const result = [modelInformation, instanceInfo];
439     return _.uniq(_.flatten(result));
440   }
441 }