3 * Copyright 2016 RIFT.IO Inc
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
9 * http://www.apache.org/licenses/LICENSE-2.0
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
18 import NetworkServiceActions
from './launchNetworkServiceActions.js';
19 import NetworkServiceSource
from './launchNetworkServiceSource.js';
20 import GUID
from 'utils/guid.js';
21 import AppHeaderActions
from 'widgets/header/headerActions.js';
22 import Alt
from '../alt';
23 import _cloneDeep
from 'lodash/cloneDeep';
24 import _find
from 'lodash/find';
25 import _merge
from 'lodash/merge';
27 class LaunchNetworkServiceStore
{
37 this.sshKeysList
= [];
39 this.sla_parameters
= [];
41 this.selectedNSD
= {};
42 this.dataCenters
= [];
43 this.isLoading
= false;
44 this.hasConfigureNSD
= false;
45 this.hasConfigureVNFD
= false;
46 this['input-parameters'] = [];
47 this.displayPlacementGroups
= true;
48 this.bindActions(NetworkServiceActions
);
49 this.nsdConfiguration
= {
53 /*Collection of vnf state containting cloud account and datacenter info.
57 this.configAgentAccounts
= [];
59 this.isPreviewing
= false;
60 this.isOpenMano
= false;
62 this.displayVIMAccounts
= false;
63 this.resourceOrchestrators
= [{
65 'ro-account-type': 'openmano'
67 this.selectedResourceOrchestrator
= this.resourceOrchestrators
[0];
68 this.dataCenterID
= null;
69 this.vnfDataCenters
= {};
70 this.registerAsync(NetworkServiceSource
);
71 this.exportPublicMethods({
72 getMockData
: getMockData
.bind(this),
73 getMockSLA
: getMockSLA
.bind(this),
74 saveNetworkServiceRecord
: this.saveNetworkServiceRecord
,
77 updateInputParam
: this.updateInputParam
,
78 resetView
: this.resetView
,
79 nameUpdated
: this.nameUpdated
,
81 descriptorSelected
: this.descriptorSelected
.bind(this),
82 deselectDescriptor
: this.deselectDescriptor
,
83 previewDescriptor
: this.previewDescriptor
,
85 ipProfileFn
: this.ipProfileFn
,
87 usersFn
: this.usersFn
,
94 console
.log('reseting state');
97 'input-parameter-xpath': null,
98 'ns-placement-groups': null,
99 'vnf-placement-groups':null
105 getCatalogSuccess(catalogs
) {
111 catalogs
.forEach(function(catalog
) {
112 switch (catalog
.type
) {
116 self
.descriptorSelected(catalog
.descriptors
[0])
118 console
.log('unable to select catalog')
129 nsd
[0].descriptors
.map(function(n
) {
133 nsd
, vnfd
, pnfd
, nsdDict
,
137 getConfigAgentSuccess(configAgentAccounts
) {
139 configAgentAccounts
: configAgentAccounts
142 getDataCentersSuccess(data
) {
143 let dataCenters
= data
;
146 dataCenters
: dataCenters
|| []
148 if (this.ro
&& this.ro
['account-type'] == 'openmano') {
149 newState
.dataCenterID
= dataCenters
[this.ro
.name
][0].uuid
151 this.setState(newState
)
153 getVDUSuccess(VNFD
) {
159 Alt
.actions
.global
.showScreenLoader
.defer();
163 console
.log('is Loading', this)
165 launchNSRSuccess(data
) {
166 console
.log('Launching Network Service')
167 let tokenizedHash
= window
.location
.hash
.split('/');
168 Alt
.actions
.global
.hideScreenLoader
.defer();
172 return window
.location
.hash
= 'launchpad/' + tokenizedHash
[2];
174 launchNSRError(data
) {
175 Alt
.actions
.global
.showNotification
.defer(data
);
176 Alt
.actions
.global
.hideScreenLoader
.defer();
181 getInstantiateSshKeySuccess
= (data
) => {
187 getResourceOrchestratorAccountsSuccess
= (data
) => {
189 Alt
.actions
.global
.hideScreenLoader
.defer();
191 ROAccounts
= ROAccounts
.concat(data
).filter(function(r
){ return r
.name
!= "rift" });
193 resourceOrchestrators
: ROAccounts
,
194 selectedResourceOrchestrator
: ROAccounts
[0],
195 dataCenterID
: ROAccounts
[0] && ROAccounts
[0].datacenters
&& ROAccounts
[0].datacenters
.datacenters
&& ROAccounts
[0].datacenters
.datacenters
[0] && ROAccounts
[0].datacenters
.datacenters
[0].name
,
196 dataCenterType
: ROAccounts
[0] && ROAccounts
[0].datacenters
&& ROAccounts
[0].datacenters
.datacenters
&& ROAccounts
[0].datacenters
.datacenters
[0] && ROAccounts
[0].datacenters
.datacenters
[0]['datacenter-type'],
197 displayVIMAccounts
: false
200 getResourceOrchestratorError
= (data
) => {
201 console
.log('getResourceOrchestrator Error: ', data
)
204 nameUpdated
= (e
) => {
209 deselectDescriptor
= () => {
210 console
.log('deselecting')
217 descriptorSelected
= (data
) => {
222 selectedNSDid
: NSD
.id
,
223 vld
: NSD
&& NSD
.vld
&& NSD
.vld
.map(function(v
) {
224 //Adding a type for UI state management
225 //This is deleted before launch
226 if(v
['ip-profile-ref']) {
227 v
.type
= 'ip-profile-ref';
229 if(v
['vim-network-name']) {
230 v
.type
= 'vim-network-name';
237 ipProfiles
: NSD
['ip-profiles']
239 newState
.selectedNSD
= data
;
240 newState
['input-parameters'] = [];
241 newState
['vnf-input-parameter'] = [];
242 if (NSD
['input-parameter-xpath']) {
243 let vnfParameters
= [];
244 newState
.hasConfigureNSD
= true;
245 NSD
['input-parameter-xpath'].map(function(p
) {
246 if (isVNFDInputParameter(p
)) {
247 newState
.hasConfigureVNFD
= true;
248 vnfParameters
.push(p
);
250 newState
.hasConfigureNSD
= true;
251 newState
['input-parameters'].push(p
);
254 newState
['vnf-input-parameter'] = NSD
['constituent-vnfd'].map(function(vnf
) {
256 'name': vnf
['vnf-name'],
257 'member-vnf-index-ref': vnf
['member-vnf-index'],
258 'vnfd-id-ref': vnf
['vnfd-id-ref'],
259 'input-parameter':_cloneDeep(vnfParameters
)
263 newState
.hasConfigureNSD
= false;
264 newState
['input-parameter'] = null;
266 if(NSD
['ns-placement-groups'] && NSD
['ns-placement-groups'].length
> 0 ) {
267 newState
['ns-placement-groups'] = NSD
['ns-placement-groups'];
269 if(NSD
['vnf-placement-groups'] && NSD
['vnf-placement-groups'].length
> 0 ) {
270 newState
['vnf-placement-groups'] = NSD
['vnf-placement-groups'];
272 NSD
["constituent-vnfd"] && NSD
["constituent-vnfd"].map((v
) => {
273 VNFIDs
.push(v
["vnfd-id-ref"]);
275 this.getInstance().getVDU(VNFIDs
);
276 this.setState(newState
);
278 function isVNFDInputParameter(p
){
279 return p
.xpath
.match('vnfd-catalog');
282 previewDescriptor
= (data
) => {
291 updateInputParam
= (i
, value
) => {
292 let ip
= this['input-parameters'];
295 'input-parameters': ip
298 updateVnfInputParam
= (i
, j
, value
) => {
299 let ip
= this['vnf-input-parameter'];
300 ip
[i
]['input-parameter'][j
].value
= value
;
302 'vnf-input-parameter': ip
308 updateSelectedRoAccount
: (resourceOrchestrator
) => {
309 let nsd
= self
.nsd
[0];
311 selectedResourceOrchestrator
: JSON
.parse(JSON
.parse(resourceOrchestrator
.target
.value
))
313 // if no datacenters and ro is not rift-ro
314 if (!(newState
.selectedResourceOrchestrator
.datacenters
&& newState
.selectedResourceOrchestrator
.datacenters
.datacenters
) ) {
315 Alt
.actions
.global
.showNotification
.defer("No data centers configured in resource orchestrator");
317 newState
.dataCenterID
= newState
.selectedResourceOrchestrator
.datacenters
.datacenters
[0].name
;
318 newState
.dataCenterType
= newState
.selectedResourceOrchestrator
.datacenters
.datacenters
[0]['datacenter-type'];
319 newState
.displayPlacementGroups
= newState
.selectedResourceOrchestrator
.name
== "rift" ? true : false;
321 self
.setState(newState
);
323 updateSelectedDataCenter
: (dataCenter
) => {
324 let dataCenterID
= JSON
.parse(JSON
.parse(dataCenter
.target
.value
));
325 let dataCenterType
= _find(self
.selectedResourceOrchestrator
.datacenters
.datacenters
, {name
: dataCenterID
})
332 placementGroupUpdate
: (i
, k
, event
) => {
333 let value
= event
.target
.value
;
334 let pg
= self
['ns-placement-groups'];
337 'ns-placement-groups': pg
340 hostAggregateUpdate
: (pgi
, hai
, k
, event
) => {
341 let value
= event
.target
.value
;
342 let pg
= self
['ns-placement-groups'];
343 let ha
= pg
[pgi
]['host-aggregate'][hai
];
346 'ns-placement-groups': pg
349 addHostAggregate
: (pgi
) => {
350 let pg
= self
['ns-placement-groups'];
351 let ha
= pg
[pgi
]['host-aggregate'];
354 'ns-placement-groups': pg
357 removeHostAggregate
: (pgi
, hai
) => {
358 let pg
= self
['ns-placement-groups'];
359 let ha
= pg
[pgi
]['host-aggregate'];
362 'ns-placement-groups': pg
365 getNSDByID
: (id
) => {
373 placementGroupUpdate
: (i
, k
, event
) => {
374 let value
= event
.target
.value
;
375 let pg
= self
['vnf-placement-groups'];
378 'vnf-placement-groups': pg
381 hostAggregateUpdate
: (pgi
, hai
, k
, event
) => {
382 let value
= event
.target
.value
;
383 let pg
= self
['vnf-placement-groups'];
384 let ha
= pg
[pgi
]['host-aggregate'][hai
];
387 'vnf-placement-groups': pg
390 addHostAggregate
: (pgi
) => {
391 let pg
= self
['vnf-placement-groups'];
392 let ha
= pg
[pgi
]['host-aggregate'];
395 'vnf-placement-groups': pg
398 removeHostAggregate
: (pgi
, hai
) => {
399 let pg
= self
['vnf-placement-groups'];
400 let ha
= pg
[pgi
]['host-aggregate'];
403 'vnf-placement-groups': pg
406 updateSelectedConfigAgent
: (id
) => {
408 let configAgentRef
= JSON
.parse(e
.target
.value
);
409 let vnfDC
= self
.vnfDataCenters
;
414 vnfDC
[id
]['config-agent-account'] = configAgentRef
;
416 if(vnfDC
[id
].hasOwnProperty('datacenter')) {
417 delete vnfCA
[id
]['config-agent-account'];
423 vnfDataCenters
: vnfDC
427 updateSelectedDataCenter
: (id
, dataCenter
) => {
428 let vnfDC
= self
.vnfDataCenters
;
429 let dc
= JSON
.parse(JSON
.parse(dataCenter
.target
.value
));
433 vnfDC
[id
]['member-vnf-index-ref'];
435 vnfDC
[id
].datacenter
= dc
;
441 vnfDataCenters
: vnfDC
451 let type
= e
.target
.value
;
453 if (vld
[i
].hasOwnProperty('type')) {
454 delete vld
[i
][vld
[i
].type
]
458 if(type
== 'ip-profile-ref') {
459 let IPProfile
= self
.ipProfiles
;
460 vld
[i
][type
] = IPProfile
[0] && IPProfile
[0].name
;
461 delete vld
[i
]['vim-network-name'];
462 delete vld
[i
]['ipv4-nat-pool-name'];
464 delete vld
[i
]['dns-server'];
465 vld
[i
]['ipv4-nat-pool-name'] = self
.dataCenterID
468 delete vld
[i
]['ip-profile-ref'];
469 delete vld
[i
]['vim-network-name'];
470 delete vld
[i
]['ipv4-nat-pool-name'];
472 self
.setState({vld
:vld
});
475 updateValue
: (i
, type
) => {
477 // Select Option returns JSON values.
478 let value
= e
.target
.nodeName
== "SELECT" ? JSON
.parse(e
.target
.value
) : e
.target
.value
;
480 vld
[i
][type
] = value
;
481 self
.setState({vld
:vld
});
486 ipProfileFn
= () => {
489 updateProfile
: (i
, key
) => {
491 // Select Option returns JSON values.
492 let value
= e
.target
.nodeName
== "SELECT" ? JSON
.parse(e
.target
.value
) : e
.target
.value
;
493 self
.ipProfiles
[i
]['ip-profile-params'][key
] = value
;
496 // Don't send this key
497 delete self
.ipProfiles
[i
]['ip-profile-params'][key
];
500 self
.setState({ipProfiles
:self
.ipProfiles
});
503 updateVersion
: (i
) => {
505 // Select Option returns JSON values.
506 let value
= e
.target
.value
;
507 self
.ipProfiles
[i
]['ip-profile-params']['ip-version'] = value
;
508 self
.setState({ipProfiles
:self
.ipProfiles
});
511 updateDNS
: (i
, dnsIndex
) => {
513 // Select Option returns JSON values.
514 let value
= e
.target
.nodeName
== "SELECT" ? JSON
.parse(e
.target
.value
) : e
.target
.value
;
515 self
.ipProfiles
[i
]['ip-profile-params']['dns-server'][dnsIndex
] = value
;
516 self
.setState({ipProfiles
:self
.ipProfiles
});
519 updateDHCP
: (i
, property
) => {
521 let value
= e
.target
.value
;
522 //If value is meant to be boolean, convert it
523 if(value
== "true" || value
== "false") {
524 value
= JSON
.parse(value
);
526 if(!self
.ipProfiles
[i
]['ip-profile-params'].hasOwnProperty('dhcp-params')) {
527 self
.ipProfiles
[i
]['ip-profile-params']['dhcp-params'] = {
533 //Removing DCHP property on disable to allow instantiation
535 self
.ipProfiles
[i
]['ip-profile-params']['dhcp-params'] = {
539 self
.ipProfiles
[i
]['ip-profile-params']['dhcp-params'][property
] = value
;
541 self
.setState({ipProfiles
:self
.ipProfiles
});
552 if(self
.ipProfiles
[i
]['ip-profile-params']['dns-server']) {
553 self
.ipProfiles
[i
]['ip-profile-params']['dns-server'].unshift({})
555 self
.ipProfiles
[i
]['ip-profile-params']['dns-server'] = [{}];
558 self
.setState({ipProfiles
:self
.ipProfiles
});
561 removeDNS
: (i
, k
) => {
564 self
.ipProfiles
[i
]['ip-profile-params']['dns-server'].splice(k
, 1);
565 if(self
.ipProfiles
[i
]['ip-profile-params']['dns-server'].length
== 0) {
566 delete self
.ipProfiles
[i
]['ip-profile-params']['dns-server'];
568 self
.setState({ipProfiles
:self
.ipProfiles
});
571 updateDNS
: (i
, k
) => {
574 let value
= e
.target
.value
;
575 self
.ipProfiles
[i
]['ip-profile-params']['dns-server'][k
].address
= value
;
576 self
.setState({ipProfiles
:self
.ipProfiles
});
584 updateNewKeyRefSelection
: (e
) => {
586 newRefSelection
: e
.target
.value
589 updateKeyRef
: (refIndex
, remove
) => {
592 let sshKeysRef
= self
.sshKeysRef
;
594 // if(!e.target.value) {
595 // return Alt.actions.global.showError.defer('Please select a key pair');
597 if(!isNaN(refIndex
)){
598 sshKeysRef
.splice(refIndex
, 1);
599 sshKeysRef
.push(e
.target
.value
);
601 sshKeysRef
.push(e
.target
.value
);
605 sshKeysRef
.splice(refIndex
, 1);
608 sshKeysRef
: sshKeysRef
,
609 newRefSelection
: null
618 add: function(sshKeysList
) {
623 'ssh-authorized-key': [sshKeysList
[0].name
]
625 let usersList
= self
.usersList
;
626 usersList
.push(newUser
);
632 remove: function(i
) {
634 self
.usersList
.splice(i
, 1);
636 usersList
: self
.usersList
640 update: function(i
, key
) {
642 let value
= e
.target
.value
;
643 self
.usersList
[i
][key
] = value
;
645 usersList
: self
.usersList
649 updateSSHkeyRef: function(i
, j
, remove
){
651 let usersList
= _cloneDeep(self
.usersList
)
652 let keys
= usersList
[i
]['ssh-authorized-key'];
654 let keyRef
= JSON
.parse(e
.target
.value
).name
;
662 usersList
[i
]['ssh-authorized-key'] = keys
;
670 saveNetworkServiceRecord(name
, launch
) {
672 //input-parameter: [{uuid: < some_unique_name>, xpath: <same as you got from nsd>, value: <user_entered_value>}]
674 'input-parameter-xpath':[{
682 // Create a filtered NSD payload from the decorated one as RW.REST cannot handle extra parameters now
684 nsdPayload
= _cloneDeep(_find(this.state
.nsd
[0].descriptors
, {id
: this.state
.selectedNSDid
}));
686 if (nsdPayload
!= {}) {
687 nsdPayload
['meta'] && delete nsdPayload
['meta'];
688 nsdPayload
['constituent-vnfd'] && nsdPayload
['constituent-vnfd'].map((constituentVnfd
) => {
689 constituentVnfd
['vnf-name'] && delete constituentVnfd
['vnf-name'];
690 constituentVnfd
['name'] && delete constituentVnfd
['name'];
692 nsdPayload
['placement-groups'] && nsdPayload
['placement-groups'].map((placementGroup
) => {
693 placementGroup
['member-vnfd'] && placementGroup
['member-vnfd'].map((memberVnfd
) => {
694 memberVnfd
['name'] && delete memberVnfd
['name'];
697 nsdPayload
['ns-placement-groups'] && delete nsdPayload
['ns-placement-groups'];
698 nsdPayload
['vnf-placement-groups'] && delete nsdPayload
['vnf-placement-groups'];
699 nsdPayload
.vld
= this.state
.vld
;
700 nsdPayload
.vld
&& nsdPayload
.vld
.map(function(v
){
704 nsdPayload
['input-parameter-xpath'] && nsdPayload
['input-parameter-xpath'].map((x
) => delete x
.value
);
710 "description": "a description for " + guuid
,
711 "admin-status": launch
? "ENABLED" : "DISABLED",
715 if(!this.state
.selectedResourceOrchestrator
) {
716 Alt
.actions
.global
.showNotification
.defer("No Resource Orchestrator selected");
719 if (this.state
.selectedResourceOrchestrator
.name
!= "rift") {
720 payload
["resource-orchestrator"] = this.state
.selectedResourceOrchestrator
.name
;
723 if(!this.state
.dataCenterID
) {
724 Alt
.actions
.global
.showNotification
.defer("No Data Center selected");
727 payload
["datacenter"] = this.state
.dataCenterID
;
729 //Clean Input Parameters
730 if (this.state
.hasConfigureNSD
) {
731 let ips
= _cloneDeep(this.state
['input-parameters']);
733 let ipsToSend
= ips
.filter(function(ip
) {
734 if (ip
.value
&& ip
.value
!= "") {
737 delete ip
['default-value'];
743 if (ipsToSend
.length
> 0) {
744 payload
['input-parameter'] = ipsToSend
;
747 //Clean VNF Input Parameters
748 if (this.state
.hasConfigureVNFD
) {
749 let vnf
= _cloneDeep(this.state
['vnf-input-parameter']);
750 vnf
= vnf
.filter(function(v
){
752 v
['input-parameter'] = v
['input-parameter'].filter(
754 if(i
.value
&& i
.value
!= "") {
756 delete i
['default-value'];
761 if (v
['input-parameter'].length
) {
766 if (vnf
.length
> 0) {
767 payload
['vnf-input-parameter'] = vnf
;
770 let VnfDataCenters
= this.state
.vnfDataCenters
;
771 if (Object
.keys(VnfDataCenters
).length
) {
772 payload
['vnf-datacenter-map'] = Object
.keys(VnfDataCenters
).map(function(k
) {
774 'member-vnf-index-ref' : k
,
775 datacenter
: VnfDataCenters
[k
].datacenter
779 // These placement groups need to be refactored. Too much boilerplate.
780 if (this.state
.displayPlacementGroups
) {
781 nsPg
= this.state
['ns-placement-groups'];
782 vnfPg
= this.state
['vnf-placement-groups'];
783 if(nsPg
&& (nsPg
.length
> 0)) {
784 payload
['nsd-placement-group-maps'] = nsPg
.map(function(n
, i
) {
785 if(n
['availability-zone'] || n
['server-group'] || (n
['host-aggregate'].length
> 0)) {
787 'cloud-type': self
.state
.dataCenterType
789 if(n
['host-aggregate'].length
> 0) {
790 obj
['host-aggregate'] = n
['host-aggregate'].map(function(h
, j
) {
792 'metadata-key': h
.key
,
793 'metadata-value': h
.value
797 if(n
['availability-zone'] && (n
['availability-zone'] != '')) {
798 obj
['availability-zone'] = {name
: n
['availability-zone']};
800 if(n
['server-group'] && (n
['server-group'] != '')) {
801 obj
['server-group'] = {name
: n
['server-group']};
803 obj
['placement-group-ref'] = n
.name
;
806 }).filter(function(o
){
814 if(vnfPg
&& (vnfPg
.length
> 0)) {
815 let vnfDataCenterDictionary
= {};
816 payload
['vnf-datacenter-map'] && payload
['vnf-datacenter-map'].map(function(d
) {
817 vnfDataCenterDictionary
[d
['member-vnf-index-ref']] = d
.datacenter
819 payload
['vnfd-placement-group-maps'] = vnfPg
.map(function(n
, i
) {
820 if(n
['availability-zone'] || n
['server-group'] || (n
['host-aggregate'].length
> 0)) {
821 let DC
= vnfDataCenterDictionary
[n
['member-vnf-index']];
823 'cloud-type': DC
? _find(self
.state
.selectedResourceOrchestrator
.datacenters
.datacenters
, {name
: DC
})['datacenter-type'] : self
.state
.dataCenterType
825 if(n
['host-aggregate'].length
> 0) {
826 obj
['host-aggregate'] = n
['host-aggregate'].map(function(h
, j
) {
828 'metadata-key': h
.key
,
829 'metadata-value': h
.value
833 if(n
['server-group'] && (n
['server-group'] != '')) {
834 obj
['server-group'] = {name
: n
['server-group']};
836 if(n
['availability-zone'] && (n
['availability-zone'] != '')) {
837 obj
['availability-zone'] = {name
: n
['availability-zone']};
839 obj
['placement-group-ref'] = n
.name
;
840 obj
['vnfd-id-ref'] = n
['vnfd-id-ref'];
843 }).filter(function(o
){
854 payload
['ssh-authorized-key'] = this.state
.sshKeysRef
.map(function(k
) {
855 return {'key-pair-ref': JSON
.parse(k
).name
};
858 payload
['user'] = addKeyPairRefToUsers(this.state
.usersList
);
867 function addKeyPairRefToUsers(list
) {
868 return list
.map(function(u
) {
871 'user-info': u
['user-info'],
872 'ssh-authorized-key': u
['ssh-authorized-key'].map(function(k
) {
881 function getMockSLA(id
) {
882 console
.log('Getting mock SLA Data for id: ' + id
);
884 sla_parameters
: slaData
888 function getMockData() {
889 console
.log('Getting mock Descriptor Data');
896 export default LaunchNetworkServiceStore
;