2 * Copyright © 2016-2017 European Support Limited
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
17 import React from 'react';
19 import {render} from 'react-dom';
20 import ReactDOM from 'react-dom';
22 import isEqual from 'lodash/isEqual.js';
24 import lodashUnionBy from 'lodash/unionBy.js';
26 import i18n from 'nfvo-utils/i18n/i18n.js';
27 import Application from 'sdc-app/Application.jsx';
28 import store from 'sdc-app/AppStore.js';
29 import Configuration from 'sdc-app/config/Configuration.js';
30 import ScreensHelper from 'sdc-app/common/helpers/ScreensHelper.js';
33 import {onboardingMethod as onboardingMethodTypes, onboardingOriginTypes} from 'sdc-app/onboarding/softwareProduct/SoftwareProductConstants.js';
35 import {itemTypes} from './versionsPage/VersionsPageConstants.js';
37 import {AppContainer} from 'react-hot-loader';
38 import HeatSetupActionHelper from './softwareProduct/attachments/setup/HeatSetupActionHelper.js';
40 import {actionTypes, enums, screenTypes} from './OnboardingConstants.js';
41 import OnboardingActionHelper from './OnboardingActionHelper.js';
42 import Onboarding from './Onboarding.js';
44 export default class OnboardingPunchOut {
46 render({options: {data, apiRoot, apiHeaders}, onEvent}, element) {
47 if (!this.unsubscribeFromStore) {
48 this.unsubscribeFromStore = store.subscribe(() => this.handleStoreChange());
51 if (!this.isConfigSet) {
52 Configuration.setCatalogApiRoot(apiRoot);
53 Configuration.setCatalogApiHeaders(apiHeaders);
54 this.isConfigSet = true;
57 this.onEvent = (...args) => onEvent(...args);
58 this.handleData(data);
70 module.hot.accept('sdc-app/onboarding/Onboarding.js', () => {
71 const NextOnboarding = require('sdc-app/onboarding/Onboarding.js').default;
88 ReactDOM.unmountComponentAtNode(element);
89 this.rendered = false;
90 this.unsubscribeFromStore();
91 this.unsubscribeFromStore = null;
95 let {breadcrumbs: {selectedKeys = []} = {}} = data;
96 let dispatch = action => store.dispatch(action);
97 let {currentScreen, users: {usersList}, softwareProductList, finalizedSoftwareProductList, licenseModelList, finalizedLicenseModelList,
98 softwareProduct: {softwareProductEditor: {data: vspData = {}},
99 softwareProductComponents = {}, softwareProductQuestionnaire = {}}} = store.getState();
100 const wholeSoftwareProductList = [...softwareProductList, ...finalizedSoftwareProductList];
101 const wholeLicenseModelList = [...licenseModelList, ...finalizedLicenseModelList];
103 let {props: {version, isReadOnlyMode}, screen} = currentScreen;
104 let {componentEditor: {data: componentData = {}, qdata: componentQData = {}}} = softwareProductComponents;
105 if (this.programmaticBreadcrumbsUpdate) {
106 this.prevSelectedKeys = selectedKeys;
107 this.programmaticBreadcrumbsUpdate = false;
110 if (!isEqual(selectedKeys, this.prevSelectedKeys)) {
111 this.breadcrumbsPrefixSelected = isEqual(selectedKeys, this.prevSelectedKeys && this.prevSelectedKeys.slice(0, selectedKeys.length));
113 const [, screenType, prevVspId, , prevComponentId] = this.prevSelectedKeys || [];
114 let preNavigate = Promise.resolve();
115 if(screenType === enums.BREADCRUMS.SOFTWARE_PRODUCT && screen !== 'VERSIONS_PAGE' && !isReadOnlyMode) {
116 let dataToSave = prevVspId ? prevComponentId ? {componentData, qdata: componentQData} : {softwareProduct: vspData, qdata: softwareProductQuestionnaire.qdata} : {};
117 preNavigate = OnboardingActionHelper.autoSaveBeforeNavigate(dispatch, {
118 softwareProductId: prevVspId,
120 vspComponentId: prevComponentId,
125 let {currentScreen: {props: {softwareProductId}}, softwareProduct: {softwareProductAttachments: {heatSetup, heatSetupCache}}} = store.getState();
126 let heatSetupPopupPromise = currentScreen.screen === enums.SCREEN.SOFTWARE_PRODUCT_ATTACHMENTS ?
127 HeatSetupActionHelper.heatSetupLeaveConfirmation(dispatch, {softwareProductId, heatSetup, heatSetupCache}) :
129 Promise.all([preNavigate, heatSetupPopupPromise]).then(() => {
130 this.prevSelectedKeys = selectedKeys;
131 if (selectedKeys.length === 0) {
132 ScreensHelper.loadScreen(dispatch, {screen: enums.SCREEN.ONBOARDING_CATALOG});
134 } else if (selectedKeys.length === 1 || selectedKeys[1] === enums.BREADCRUMS.LICENSE_MODEL) {
135 let [licenseModelId, , licenseModelScreen] = selectedKeys;
136 let licenseModel = wholeLicenseModelList.find(vlm => vlm.id === licenseModelId);
137 ScreensHelper.loadScreen(dispatch, {screen: licenseModelScreen, screenType: screenTypes.LICENSE_MODEL,
138 props: {licenseModelId, version, licenseModel, usersList}});
140 } else if (selectedKeys.length <= 4 && selectedKeys[1] === enums.BREADCRUMS.SOFTWARE_PRODUCT) {
141 let [licenseModelId, , softwareProductId, softwareProductScreen] = selectedKeys;
142 let softwareProduct = softwareProductId ?
143 wholeSoftwareProductList.find(({id}) => id === softwareProductId) :
144 wholeSoftwareProductList.find(({vendorId}) => vendorId === licenseModelId);
145 if (!softwareProductId) {
146 softwareProductId = softwareProduct.id;
148 if (softwareProductScreen === enums.SCREEN.SOFTWARE_PRODUCT_ATTACHMENTS) {
149 softwareProduct = vspData;
150 //check current vsp fields to determine which file has uploaded
151 if(vspData.onboardingOrigin === onboardingOriginTypes.ZIP || vspData.candidateOnboardingOrigin === onboardingOriginTypes.ZIP) {
152 softwareProductScreen = enums.SCREEN.SOFTWARE_PRODUCT_ATTACHMENTS_SETUP;
154 else if(vspData.onboardingOrigin === onboardingOriginTypes.CSAR) {
155 softwareProductScreen = enums.SCREEN.SOFTWARE_PRODUCT_ATTACHMENTS_VALIDATION;
159 ScreensHelper.loadScreen(dispatch, {screen: softwareProductScreen, screenType: screenTypes.SOFTWARE_PRODUCT,
160 props: {softwareProductId, softwareProduct, version, usersList}});
162 } else if (selectedKeys.length === 5 && selectedKeys[1] === enums.BREADCRUMS.SOFTWARE_PRODUCT && selectedKeys[3] === enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENTS) {
163 let [licenseModelId, , softwareProductId, , componentId] = selectedKeys;
164 let softwareProduct = softwareProductId ?
165 wholeSoftwareProductList.find(({id}) => id === softwareProductId) :
166 wholeSoftwareProductList.find(({vendorId}) => vendorId === licenseModelId);
167 if (!softwareProductId) {
168 softwareProductId = softwareProduct.id;
170 ScreensHelper.loadScreen(dispatch, {screen: enums.SCREEN.SOFTWARE_PRODUCT_COMPONENTS, screenType: screenTypes.SOFTWARE_PRODUCT,
171 props: {softwareProductId, softwareProduct, componentId, version, usersList}});
173 } else if (selectedKeys.length === 6 && selectedKeys[1] === enums.BREADCRUMS.SOFTWARE_PRODUCT && selectedKeys[3] === enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENTS) {
174 let [licenseModelId, , softwareProductId, , componentId, componentScreen] = selectedKeys;
175 let softwareProduct = softwareProductId ?
176 wholeSoftwareProductList.find(({id}) => id === softwareProductId) :
177 wholeSoftwareProductList.find(({vendorId}) => vendorId === licenseModelId);
178 if (!softwareProductId) {
179 softwareProductId = softwareProduct.id;
181 ScreensHelper.loadScreen(dispatch, {screen: componentScreen, screenType: screenTypes.SOFTWARE_PRODUCT,
182 props: {softwareProductId, softwareProduct, componentId, version, usersList}});
185 console.error('Unknown breadcrumbs path: ', selectedKeys);
189 type: actionTypes.SET_CURRENT_SCREEN,
192 forceBreadCrumbsUpdate: true
199 handleStoreChange() {
200 let {currentScreen, licenseModelList, finalizedLicenseModelList, softwareProductList, finalizedSoftwareProductList, versionsPage:
201 {versionsList: {itemType, itemId}},
202 softwareProduct: {softwareProductEditor: {data: currentSoftwareProduct = {onboardingMethod: ''}},
203 softwareProductComponents: {componentsList}}} = store.getState();
204 const wholeSoftwareProductList = lodashUnionBy(softwareProductList, finalizedSoftwareProductList, 'id');
205 const wholeLicenseModelList = lodashUnionBy(licenseModelList, finalizedLicenseModelList, 'id');
206 let breadcrumbsData = {itemType, itemId, currentScreen, wholeLicenseModelList, wholeSoftwareProductList, currentSoftwareProduct, componentsList};
208 if (currentScreen.forceBreadCrumbsUpdate || !isEqual(breadcrumbsData, this.prevBreadcrumbsData) || this.breadcrumbsPrefixSelected) {
209 this.prevBreadcrumbsData = breadcrumbsData;
210 this.breadcrumbsPrefixSelected = false;
211 this.programmaticBreadcrumbsUpdate = true;
212 let breadcrumbs = this.buildBreadcrumbs(breadcrumbsData);
213 this.onEvent('breadcrumbsupdated', breadcrumbs);
215 type: actionTypes.SET_CURRENT_SCREEN,
218 forceBreadCrumbsUpdate: false
224 buildBreadcrumbs({currentScreen: {screen, props}, itemType, itemId, currentSoftwareProduct,
225 wholeLicenseModelList, wholeSoftwareProductList, componentsList}) {
226 let {onboardingMethod, onboardingOrigin, candidateOnboardingOrigin} = currentSoftwareProduct;
227 let screenToBreadcrumb;
229 case enums.SCREEN.ONBOARDING_CATALOG:
232 case enums.SCREEN.VERSIONS_PAGE:
233 let firstMenuItems = itemType === itemTypes.LICENSE_MODEL ? [
236 menuItems: wholeLicenseModelList.map(({id, name}) => ({
242 selectedKey: props.additionalProps.licenseModelId || currentSoftwareProduct.vendorId,
243 menuItems: wholeLicenseModelList.map(({id, name}) => ({
249 selectedKey: enums.BREADCRUMS.SOFTWARE_PRODUCT,
251 key: enums.BREADCRUMS.LICENSE_MODEL,
252 displayText: i18n('License Model')
254 key: enums.BREADCRUMS.SOFTWARE_PRODUCT,
255 displayText: i18n('Software Products')
260 menuItems: wholeSoftwareProductList
261 .filter(({id, vendorId}) => vendorId === currentSoftwareProduct.vendorId || id === itemId)
262 .map(({id, name}) => ({
271 selectedKey: enums.BREADCRUMS.VERSIONS_PAGE,
272 menuItems: [{key: enums.BREADCRUMS.VERSIONS_PAGE, displayText: i18n('Versions Page')}]
276 case enums.SCREEN.LICENSE_AGREEMENTS:
277 case enums.SCREEN.FEATURE_GROUPS:
278 case enums.SCREEN.ENTITLEMENT_POOLS:
279 case enums.SCREEN.LICENSE_KEY_GROUPS:
280 case enums.SCREEN.LICENSE_MODEL_OVERVIEW:
281 case enums.SCREEN.ACTIVITY_LOG:
282 screenToBreadcrumb = {
283 [enums.SCREEN.LICENSE_AGREEMENTS]: enums.BREADCRUMS.LICENSE_AGREEMENTS,
284 [enums.SCREEN.FEATURE_GROUPS]: enums.BREADCRUMS.FEATURE_GROUPS,
285 [enums.SCREEN.ENTITLEMENT_POOLS]: enums.BREADCRUMS.ENTITLEMENT_POOLS,
286 [enums.SCREEN.LICENSE_KEY_GROUPS]: enums.BREADCRUMS.LICENSE_KEY_GROUPS,
287 [enums.SCREEN.LICENSE_MODEL_OVERVIEW]: enums.BREADCRUMS.LICENSE_MODEL_OVERVIEW,
288 [enums.SCREEN.ACTIVITY_LOG]: enums.BREADCRUMS.ACTIVITY_LOG
292 selectedKey: props.licenseModelId,
293 menuItems: wholeLicenseModelList.map(({id, name}) => ({
299 selectedKey: enums.BREADCRUMS.LICENSE_MODEL,
301 key: enums.BREADCRUMS.LICENSE_MODEL,
302 displayText: i18n('License Model')
304 ...(wholeSoftwareProductList.findIndex(({vendorId}) => vendorId === props.licenseModelId) === -1 ? [] : [{
305 key: enums.BREADCRUMS.SOFTWARE_PRODUCT,
306 displayText: i18n('Software Products')
309 selectedKey: screenToBreadcrumb[screen],
311 key: enums.BREADCRUMS.LICENSE_MODEL_OVERVIEW,
312 displayText: i18n('Overview')
314 key: enums.BREADCRUMS.LICENSE_AGREEMENTS,
315 displayText: i18n('License Agreements')
317 key: enums.BREADCRUMS.FEATURE_GROUPS,
318 displayText: i18n('Feature Groups')
320 key: enums.BREADCRUMS.ENTITLEMENT_POOLS,
321 displayText: i18n('Entitlement Pools')
323 key: enums.BREADCRUMS.LICENSE_KEY_GROUPS,
324 displayText: i18n('License Key Groups')
326 key: enums.BREADCRUMS.ACTIVITY_LOG,
327 displayText: i18n('Activity Log')
332 case enums.SCREEN.SOFTWARE_PRODUCT_LANDING_PAGE:
333 case enums.SCREEN.SOFTWARE_PRODUCT_DETAILS:
334 case enums.SCREEN.SOFTWARE_PRODUCT_ATTACHMENTS:
335 case enums.SCREEN.SOFTWARE_PRODUCT_PROCESSES:
336 case enums.SCREEN.SOFTWARE_PRODUCT_DEPLOYMENT:
337 case enums.SCREEN.SOFTWARE_PRODUCT_NETWORKS:
338 case enums.SCREEN.SOFTWARE_PRODUCT_DEPENDENCIES:
339 case enums.SCREEN.SOFTWARE_PRODUCT_ACTIVITY_LOG:
340 case enums.SCREEN.SOFTWARE_PRODUCT_COMPONENTS:
342 case enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_PROCESSES:
343 case enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_COMPUTE:
344 case enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_STORAGE:
345 case enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_NETWORK:
346 case enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_GENERAL:
347 case enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_LOAD_BALANCING:
348 case enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_IMAGES:
349 case enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_MONITORING:
350 screenToBreadcrumb = {
351 [enums.SCREEN.SOFTWARE_PRODUCT_LANDING_PAGE]: enums.BREADCRUMS.SOFTWARE_PRODUCT_LANDING_PAGE,
352 [enums.SCREEN.SOFTWARE_PRODUCT_DETAILS]: enums.BREADCRUMS.SOFTWARE_PRODUCT_DETAILS,
353 [enums.SCREEN.SOFTWARE_PRODUCT_ATTACHMENTS]: enums.BREADCRUMS.SOFTWARE_PRODUCT_ATTACHMENTS,
354 [enums.SCREEN.SOFTWARE_PRODUCT_PROCESSES]: enums.BREADCRUMS.SOFTWARE_PRODUCT_PROCESSES,
355 [enums.SCREEN.SOFTWARE_PRODUCT_DEPLOYMENT]: enums.BREADCRUMS.SOFTWARE_PRODUCT_DEPLOYMENT,
356 [enums.SCREEN.SOFTWARE_PRODUCT_NETWORKS]: enums.BREADCRUMS.SOFTWARE_PRODUCT_NETWORKS,
357 [enums.SCREEN.SOFTWARE_PRODUCT_DEPENDENCIES]: enums.BREADCRUMS.SOFTWARE_PRODUCT_DEPENDENCIES,
358 [enums.SCREEN.SOFTWARE_PRODUCT_COMPONENTS]: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENTS,
359 [enums.SCREEN.SOFTWARE_PRODUCT_ACTIVITY_LOG]: enums.BREADCRUMS.SOFTWARE_PRODUCT_ACTIVITY_LOG
361 let componentScreenToBreadcrumb = {
362 [enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_PROCESSES]: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_PROCESSES,
363 [enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_COMPUTE]: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_COMPUTE,
364 [enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_STORAGE]: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_STORAGE,
365 [enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_NETWORK]: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_NETWORK,
366 [enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_GENERAL]: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_GENERAL,
367 [enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_LOAD_BALANCING]: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_LOAD_BALANCING,
368 [enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_IMAGES]: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_IMAGES,
369 [enums.SCREEN.SOFTWARE_PRODUCT_COMPONENT_MONITORING]: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_MONITORING
371 let licenseModelId = currentSoftwareProduct.vendorId;
372 let returnedBreadcrumb = [
374 selectedKey: licenseModelId,
375 menuItems: wholeLicenseModelList.map(({id, name}) => ({
381 selectedKey: enums.BREADCRUMS.SOFTWARE_PRODUCT,
383 key: enums.BREADCRUMS.LICENSE_MODEL,
384 displayText: i18n('License Model')
386 key: enums.BREADCRUMS.SOFTWARE_PRODUCT,
387 displayText: i18n('Software Products')
391 selectedKey: props.softwareProductId,
392 menuItems: wholeSoftwareProductList
393 .filter(({vendorId, id}) => vendorId === licenseModelId || id === props.softwareProductId)
394 .map(({id, name}) => ({
399 ...(/*screen === enums.SCREEN.SOFTWARE_PRODUCT_LANDING_PAGE ? [] :*/ [{
400 selectedKey: screenToBreadcrumb[screen] || enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENTS,
402 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_LANDING_PAGE,
403 displayText: i18n('Overview')
405 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_DETAILS,
406 displayText: i18n('General')
408 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_DEPLOYMENT,
409 displayText: i18n('Deployment Flavors')
411 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_PROCESSES,
412 displayText: i18n('Process Details')
414 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_NETWORKS,
415 displayText: i18n('Networks')
417 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_DEPENDENCIES,
418 displayText: i18n('Components Dependencies')
420 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_ATTACHMENTS,
421 displayText: i18n('Attachments')
423 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_ACTIVITY_LOG,
424 displayText: i18n('Activity Log')
426 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENTS,
427 displayText: i18n('Components')
430 case enums.BREADCRUMS.SOFTWARE_PRODUCT_ATTACHMENTS:
431 let isHeatData = onboardingOrigin !== onboardingOriginTypes.NONE || candidateOnboardingOrigin === onboardingOriginTypes.ZIP;
433 case enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENTS:
434 return (componentsList.length > 0);
435 case enums.BREADCRUMS.SOFTWARE_PRODUCT_DEPLOYMENT:
436 let isManualMode = onboardingMethod === onboardingMethodTypes.MANUAL;
438 case enums.BREADCRUMS.SOFTWARE_PRODUCT_DEPENDENCIES:
439 return (componentsList.length > 1);
446 if(props.componentId) {
447 returnedBreadcrumb = [
448 ...returnedBreadcrumb, {
449 selectedKey: props.componentId,
450 menuItems: componentsList
451 .map(({id, displayName}) => ({
453 displayText: displayName
457 selectedKey: componentScreenToBreadcrumb[screen],
459 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_GENERAL,
460 displayText: i18n('General')
462 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_COMPUTE,
463 displayText: i18n('Compute')
465 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_LOAD_BALANCING,
466 displayText: i18n('High Availability & Load Balancing')
468 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_NETWORK,
469 displayText: i18n('Networks')
471 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_STORAGE,
472 displayText: i18n('Storage')
474 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_IMAGES,
475 displayText: i18n('Images')
477 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_PROCESSES,
478 displayText: i18n('Process Details')
480 key: enums.BREADCRUMS.SOFTWARE_PRODUCT_COMPONENT_MONITORING,
481 displayText: i18n('Monitoring')
486 return returnedBreadcrumb;