/** * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var App = require('app'); var validator = require('utils/validator'); var hostsManagement = require('utils/hosts'); var numberUtils = require('utils/number_utils'); App.ManageConfigGroupsController = Em.Controller.extend(App.ConfigOverridable, { name: 'manageConfigGroupsController', /** * Determines if needed data is already loaded * Loading chain starts at loadHosts and is complete on the loadConfigGroups (if user on * the Installer) or on the _onLoadConfigGroupsSuccess (otherwise) * @type {boolean} */ isLoaded: false, /** * Determines if user currently is on the Cluster Installer * @type {boolean} */ isInstaller: false, /** * Determines if user currently is on the Add Service Wizard * @type {boolean} */ isAddService: false, /** * Current service name * @type {string} */ serviceName: null, /** * @type {App.ConfigGroup[]} */ configGroups: [], /** * @type {App.ConfigGroup[]} */ originalConfigGroups: [], /** * @type {App.ConfigGroup} */ selectedConfigGroup: null, /** * @type {string[]} */ selectedHosts: [], /** * List of all hosts in the cluster * @type {{ * id: string, * ip: string, * osType: string, * osArch: string, * hostName: string, * publicHostName: string, * cpu: number, * memory: number, * diskTotal: string, * diskFree: string, * disksMounted: number, * hostComponents: { * componentName: string, * displayName: string * }[] * }[]} */ clusterHosts: [], /** * List of available service components for serviceName * @type {{componentName: string, displayName: string, selected: boolean}[]} */ componentsForFilter: function () { return App.StackServiceComponent.find().filterProperty('serviceName', this.get('serviceName')).map(function (component) { return Em.Object.create({ componentName: component.get('componentName'), displayName: App.format.role(component.get('componentName')), selected: false }); }); }.property('serviceName'), /** * Determines when host may be deleted from config group * @type {boolean} */ isDeleteHostsDisabled: function () { var selectedConfigGroup = this.get('selectedConfigGroup'); if (selectedConfigGroup) { return selectedConfigGroup.isDefault || this.get('selectedHosts').length === 0; } return true; }.property('selectedConfigGroup', 'selectedConfigGroup.hosts.length', 'selectedHosts.length'), /** * Map with modified/deleted/created config groups * @type {{ * toClearHosts: App.ConfigGroup[], * toDelete: App.ConfigGroup[], * toSetHosts: App.ConfigGroup[], * toCreate: App.ConfigGroup[] * }} */ hostsModifiedConfigGroups: function () { if (!this.get('isLoaded')) { return false; } var groupsToClearHosts = []; var groupsToDelete = []; var groupsToSetHosts = []; var groupsToCreate = []; var groups = this.get('configGroups'); var originalGroups = this.get('originalConfigGroups'); // remove default group var originalGroupsCopy = originalGroups.without(originalGroups.findProperty('isDefault')); var originalGroupsIds = originalGroupsCopy.mapProperty('id'); groups.forEach(function (group) { if (!group.get('isDefault')) { var originalGroup = originalGroupsCopy.findProperty('id', group.get('id')); if (originalGroup) { if (!(JSON.stringify(group.get('hosts').slice().sort()) === JSON.stringify(originalGroup.get('hosts').sort()))) { groupsToClearHosts.push(group.set('id', originalGroup.get('id'))); if (group.get('hosts').length) { groupsToSetHosts.push(group.set('id', originalGroup.get('id'))); } // should update name or description } else if (group.get('description') !== originalGroup.get('description') || group.get('name') !== originalGroup.get('name') ) { groupsToSetHosts.push(group.set('id', originalGroup.get('id'))); } originalGroupsIds = originalGroupsIds.without(group.get('id')); } else { groupsToCreate.push(group); } } }); originalGroupsIds.forEach(function (id) { groupsToDelete.push(originalGroupsCopy.findProperty('id', id)); }, this); return { toClearHosts: groupsToClearHosts, toDelete: groupsToDelete, toSetHosts: groupsToSetHosts, toCreate: groupsToCreate, initialGroups: originalGroupsCopy }; }.property('selectedConfigGroup.hosts.@each', 'selectedConfigGroup.hosts.length', 'selectedConfigGroup.description', 'configGroups', 'isLoaded'), /** * Determines if some changes were done with config groups * @use hostsModifiedConfigGroups * @type {boolean} */ isHostsModified: function () { if (!this.get('isLoaded')) { return false; } var ignoreKeys = ['initialGroups']; var modifiedGroups = this.get('hostsModifiedConfigGroups'); return Em.keys(modifiedGroups).map(function (key) { return ignoreKeys.contains(key) ? 0 : Em.get(modifiedGroups[key], 'length'); }).reduce(Em.sum) > 0; }.property('hostsModifiedConfigGroups'), /** * Resort config groups according to order: * default group first, other - last * @method resortConfigGroup */ resortConfigGroup: function() { var configGroups = Em.copy(this.get('configGroups')); if(configGroups.length < 2) return; var defaultConfigGroup = configGroups.findProperty('isDefault'); configGroups.removeObject(defaultConfigGroup); var sorted = [defaultConfigGroup].concat(configGroups.sortProperty('name')); this.removeObserver('configGroups.@each.name', this, 'resortConfigGroup'); this.set('configGroups', sorted); this.addObserver('configGroups.@each.name', this, 'resortConfigGroup'); }.observes('configGroups.@each.name'), /** * Load hosts from server or * get them from installerController if user on the install wizard * get them from isAddServiceController if user on the add service wizard * @method loadHosts */ loadHosts: function() { this.set('isLoaded', false); if (this.get('isInstaller')) { var allHosts = this.get('isAddService') ? App.router.get('addServiceController').get('allHosts') : App.router.get('installerController').get('allHosts'); this.set('clusterHosts', allHosts); this.loadConfigGroups(this.get('serviceName')); } else { this.loadHostsFromServer(); } }, /** * Request all hosts directly from server * @method loadHostsFromServer * @return {$.ajax} */ loadHostsFromServer: function() { return App.ajax.send({ name: 'hosts.config_groups', sender: this, data: {}, success: '_loadHostsFromServerSuccessCallback', error: '_loadHostsFromServerErrorCallback' }); }, /** * Success-callback for loadHostsFromServer * Parse hosts response and wrap them into Ember.Object * @param {object} data * @method _loadHostsFromServerSuccessCallback * @private */ _loadHostsFromServerSuccessCallback: function (data) { var wrappedHosts = []; data.items.forEach(function (host) { var hostComponents = []; var diskInfo = host.Hosts.disk_info.filter(function(item) { return /^ext|^ntfs|^fat|^xfs/i.test(item.type); }); if (diskInfo.length) { diskInfo = diskInfo.reduce(function(a, b) { return { available: parseInt(a.available) + parseInt(b.available), size: parseInt(a.size) + parseInt(b.size) }; }); } host.host_components.forEach(function (hostComponent) { hostComponents.push(Em.Object.create({ componentName: hostComponent.HostRoles.component_name, displayName: App.format.role(hostComponent.HostRoles.component_name) })); }, this); wrappedHosts.pushObject(Em.Object.create({ id: host.Hosts.host_name, ip: host.Hosts.ip, osType: host.Hosts.os_type, osArch: host.Hosts.os_arch, hostName: host.Hosts.host_name, publicHostName: host.Hosts.public_host_name, cpu: host.Hosts.cpu_count, memory: host.Hosts.total_mem, diskTotal: numberUtils.bytesToSize(diskInfo.size, 0, undefined, 1024), diskFree: numberUtils.bytesToSize(diskInfo.available, 0, undefined, 1024), disksMounted: host.Hosts.disk_info.length, hostComponents: hostComponents } )); }, this); this.set('clusterHosts', wrappedHosts); this.loadConfigGroups(this.get('serviceName')); }, /** * Error-callback for loadHostsFromServer * @method _loadHostsFromServerErrorCallback * @private */ _loadHostsFromServerErrorCallback: function () { console.warn('ERROR: request to fetch all hosts failed'); this.set('clusterHosts', []); this.loadConfigGroups(this.get('serviceName')); }, /** * Load config groups from server if user is on the already installed cluster * If not - use loaded data form wizardStep7Controller * @param {string} serviceName * @method loadConfigGroups */ loadConfigGroups: function (serviceName) { if (this.get('isInstaller')) { this.set('serviceName', serviceName); var configGroups = this.copyConfigGroups(App.router.get('wizardStep7Controller.selectedService.configGroups')); var originalConfigGroups = this.copyConfigGroups(configGroups); this.setProperties({ configGroups: configGroups, originalConfigGroups: originalConfigGroups, isLoaded: true }); } else { this.set('serviceName', serviceName); App.ajax.send({ name: 'service.load_config_groups', data: { serviceName: serviceName }, sender: this, success: '_onLoadConfigGroupsSuccess' }); } }, /** * Success-callback for loadConfigGroups * @param {object} data * @private * @method _onLoadConfigGroupsSuccess */ _onLoadConfigGroupsSuccess: function (data) { var usedHosts = []; var unusedHosts = []; var serviceName = this.get('serviceName'); var serviceDisplayName = App.StackService.find().findProperty('serviceName', this.get('serviceName')).get('displayName'); var defaultConfigGroup = App.ConfigGroup.create({ name: serviceDisplayName + " Default", description: "Default cluster level " + this.get('serviceName') + " configuration", isDefault: true, parentConfigGroup: null, service: this.get('content'), configSiteTags: [], serviceName: serviceName }); if (data && data.items) { var groupToTypeToTagMap = {}; var configGroups = []; data.items.forEach(function (configGroup) { configGroup = configGroup.ConfigGroup; var hostNames = configGroup.hosts.mapProperty('host_name'); var newConfigGroup = App.ConfigGroup.create({ id: configGroup.id, name: configGroup.group_name, description: configGroup.description, isDefault: false, parentConfigGroup: defaultConfigGroup, service: App.Service.find().findProperty('serviceName', configGroup.tag), hosts: hostNames, configSiteTags: [], properties: [], apiResponse: configGroup }); usedHosts = usedHosts.concat(newConfigGroup.get('hosts')); configGroups.push(newConfigGroup); var newConfigGroupSiteTags = newConfigGroup.get('configSiteTags'); configGroup.desired_configs.forEach(function (config) { newConfigGroupSiteTags.push(App.ConfigSiteTag.create({ site: config.type, tag: config.tag })); if (!groupToTypeToTagMap[configGroup.group_name]) { groupToTypeToTagMap[configGroup.group_name] = {} } groupToTypeToTagMap[configGroup.group_name][config.type] = config.tag; }); }, this); unusedHosts = this.get('clusterHosts').mapProperty('hostName'); usedHosts.uniq().forEach(function (host) { unusedHosts = unusedHosts.without(host); }, this); defaultConfigGroup.set('childConfigGroups', configGroups); defaultConfigGroup.set('hosts', unusedHosts); var allGroups = [defaultConfigGroup].concat(configGroups); this.set('configGroups', allGroups); var originalGroups = this.copyConfigGroups(allGroups); this.set('originalConfigGroups', originalGroups); this.loadProperties(groupToTypeToTagMap); this.set('isLoaded', true); } }, /** * * @param {object} groupToTypeToTagMap * @method loadProperties */ loadProperties: function (groupToTypeToTagMap) { var typeTagToGroupMap = {}; var urlParams = []; for (var group in groupToTypeToTagMap) { var overrideTypeTags = groupToTypeToTagMap[group]; for (var type in overrideTypeTags) { var tag = overrideTypeTags[type]; typeTagToGroupMap[type + "///" + tag] = group; urlParams.push('(type=' + type + '&tag=' + tag + ')'); } } var params = urlParams.join('|'); if (urlParams.length) { App.ajax.send({ name: 'config.host_overrides', sender: this, data: { params: params, typeTagToGroupMap: typeTagToGroupMap }, success: '_onLoadPropertiesSuccess' }); } }, /** * Success-callback for loadProperties * @param {object} data * @param {object} opt * @param {object} params * @private * @method _onLoadPropertiesSuccess */ _onLoadPropertiesSuccess: function (data, opt, params) { data.items.forEach(function (configs) { var typeTagConfigs = []; var group = params.typeTagToGroupMap[configs.type + "///" + configs.tag]; for (var config in configs.properties) { typeTagConfigs.push(Em.Object.create({ name: config, value: configs.properties[config] })); } this.get('configGroups').findProperty('name', group).get('properties').pushObjects(typeTagConfigs); }, this); }, /** * Show popup with properties overridden in the selected config group * @method showProperties */ showProperties: function () { var properies = this.get('selectedConfigGroup.propertiesList').htmlSafe(); if (properies) { App.showAlertPopup(Em.I18n.t('services.service.config_groups_popup.properties'), properies); } }, /** * Show popup with hosts to add to the selected config group * @returns {boolean} * @method addHosts */ addHosts: function () { if (this.get('selectedConfigGroup.isAddHostsDisabled')) { return false; } var availableHosts = this.get('selectedConfigGroup.availableHosts'); var popupDescription = { header: Em.I18n.t('hosts.selectHostsDialog.title'), dialogMessage: Em.I18n.t('hosts.selectHostsDialog.message').format(this.get('selectedConfigGroup.displayName')) }; hostsManagement.launchHostsSelectionDialog(availableHosts, [], false, this.get('componentsForFilter'), this.addHostsCallback.bind(this), popupDescription); }, /** * add hosts callback * @param {string[]} selectedHosts * @method addHostsCallback */ addHostsCallback: function (selectedHosts) { if (selectedHosts) { var group = this.get('selectedConfigGroup'); selectedHosts.forEach(function (hostName) { group.get('hosts').pushObject(hostName); group.get('parentConfigGroup.hosts').removeObject(hostName); }); } }, /** * delete hosts from group * @method deleteHosts */ deleteHosts: function () { if (this.get('isDeleteHostsDisabled')) { return; } this.get('selectedHosts').slice().forEach(function (hostName) { this.get('selectedConfigGroup.parentConfigGroup.hosts').pushObject(hostName); this.get('selectedConfigGroup.hosts').removeObject(hostName); }, this); this.set('selectedHosts', []); }, /** * show popup for confirmation delete config group * @method confirmDelete */ confirmDelete: function () { var self = this; App.showConfirmationPopup(function() { self.deleteConfigGroup(); }); }, /** * delete selected config group (stored in the selectedConfigGroup) * then select default config group * @method deleteConfigGroup */ deleteConfigGroup: function () { var selectedConfigGroup = this.get('selectedConfigGroup'); if (this.get('isDeleteGroupDisabled')) { return; } //move hosts of group to default group (available hosts) this.set('selectedHosts', selectedConfigGroup.get('hosts')); this.deleteHosts(); this.get('configGroups').removeObject(selectedConfigGroup); this.set('selectedConfigGroup', this.get('configGroups').findProperty('isDefault')); }, /** * rename new config group (not allowed for default group) * @method renameConfigGroup */ renameConfigGroup: function () { if(this.get('selectedConfigGroup.isDefault')) { return; } var self = this; var renameGroupPopup = App.ModalPopup.show({ header: Em.I18n.t('services.service.config_groups.rename_config_group_popup.header'), bodyClass: Em.View.extend({ templateName: require('templates/main/service/new_config_group') }), configGroupName: self.get('selectedConfigGroup.name'), configGroupDesc: self.get('selectedConfigGroup.description'), warningMessage: null, isDescriptionDirty: false, validate: function () { var warningMessage = ''; var originalGroup = self.get('selectedConfigGroup'); var groupName = this.get('configGroupName').trim(); if (originalGroup.get('description') !== this.get('configGroupDesc') && !this.get('isDescriptionDirty')) { this.set('isDescriptionDirty', true); } if (originalGroup.get('name').trim() === groupName) { if (this.get('isDescriptionDirty')) { warningMessage = ''; } else { warningMessage = Em.I18n.t("config.group.selection.dialog.err.name.exists"); } } else { if (self.get('configGroups').mapProperty('name').contains(groupName)) { warningMessage = Em.I18n.t("config.group.selection.dialog.err.name.exists"); } else if (groupName && !validator.isValidConfigGroupName(groupName)) { warningMessage = Em.I18n.t("form.validator.configGroupName"); } } this.set('warningMessage', warningMessage); }.observes('configGroupName', 'configGroupDesc'), disablePrimary: function () { return !(this.get('configGroupName').trim().length > 0 && (this.get('warningMessage') !== null && !this.get('warningMessage'))); }.property('warningMessage', 'configGroupName', 'configGroupDesc'), onPrimary: function () { self.set('selectedConfigGroup.name', this.get('configGroupName')); self.set('selectedConfigGroup.description', this.get('configGroupDesc')); self.get('selectedConfigGroup.properties').forEach(function(property){ property.set('group', self.get('selectedConfigGroup')); }); this.hide(); } }); this.set('renameGroupPopup', renameGroupPopup); }, /** * add new config group (or copy existing) * @param {boolean} duplicated true - copy selectedConfigGroup, false - create a new one * @method addConfigGroup */ addConfigGroup: function (duplicated) { duplicated = (duplicated === true); var self = this; var addGroupPopup = App.ModalPopup.show({ header: Em.I18n.t('services.service.config_groups.add_config_group_popup.header'), bodyClass: Em.View.extend({ templateName: require('templates/main/service/new_config_group') }), configGroupName: duplicated ? self.get('selectedConfigGroup.name') + ' Copy' : "", configGroupDesc: duplicated ? self.get('selectedConfigGroup.description') + ' (Copy)' : "", warningMessage: '', didInsertElement: function(){ this.validate(); this.$('input').focus(); }, validate: function () { var warningMessage = ''; var groupName = this.get('configGroupName').trim(); if (self.get('configGroups').mapProperty('name').contains(groupName)) { warningMessage = Em.I18n.t("config.group.selection.dialog.err.name.exists"); } else if (groupName && !validator.isValidConfigGroupName(groupName)) { warningMessage = Em.I18n.t("form.validator.configGroupName"); } this.set('warningMessage', warningMessage); }.observes('configGroupName'), disablePrimary: function () { return !(this.get('configGroupName').trim().length > 0 && !this.get('warningMessage')); }.property('warningMessage', 'configGroupName'), onPrimary: function () { var defaultConfigGroup = self.get('configGroups').findProperty('isDefault'); var properties = []; var newConfigGroupData = App.ConfigGroup.create({ id: null, name: this.get('configGroupName').trim(), description: this.get('configGroupDesc'), isDefault: false, parentConfigGroup: defaultConfigGroup, service: Em.Object.create({id: self.get('serviceName')}), hosts: [], configSiteTags: [], properties: [] }); if (duplicated) { self.get('selectedConfigGroup.properties').forEach(function(property) { var property = App.ServiceConfigProperty.create($.extend(false, {}, property)); property.set('group', newConfigGroupData); properties.push(property); }); newConfigGroupData.set('properties', properties); } else { newConfigGroupData.set('properties', []); } self.get('configGroups').pushObject(newConfigGroupData); defaultConfigGroup.get('childConfigGroups').pushObject(newConfigGroupData); this.hide(); } }); this.set('addGroupPopup', addGroupPopup); }, /** * Duplicate existing config group * @method duplicateConfigGroup */ duplicateConfigGroup: function() { this.addConfigGroup(true); }, /** * copy config groups to manage popup to give user choice whether or not save changes * @param originGroups * @return {Array} * @method copyConfigGroups */ copyConfigGroups: function (originGroups) { var configGroups = []; var result = []; var defaultConfigGroup = App.ConfigGroup.create($.extend(true, {}, originGroups.findProperty('isDefault'))); originGroups.forEach(function (configGroup) { if (!configGroup.get('isDefault')) { var copiedGroup = App.ConfigGroup.create($.extend(true, {}, configGroup)); copiedGroup.set('parentConfigGroup', defaultConfigGroup); configGroups.pushObject(copiedGroup); } }); defaultConfigGroup.set('childConfigGroups', configGroups.slice()); configGroups.pushObject(defaultConfigGroup); configGroups.forEach(function (group) { var groupCopy = {}; for (var prop in group) { if (group.hasOwnProperty(prop)) { groupCopy[prop] = group[prop]; } } groupCopy.properties.forEach(function(property){ property.set('group', group); }); result.push(App.ConfigGroup.create(groupCopy)); }, this); return result; }, /** * Show popup with config groups * User may edit/create/delete them * @param {Em.Controller} controller * @param {App.Service} service * @returns {App.ModalPopup} * @method manageConfigurationGroups */ manageConfigurationGroups: function (controller, service) { var configsController = this; var serviceData = (controller && controller.get('selectedService')) || service; var serviceName = serviceData.get('serviceName'); var displayName = serviceData.get('displayName'); this.setProperties({ isInstaller: !!controller, serviceName: serviceName }); if (controller) { configsController.set('isAddService', controller.get('content.controllerName') == 'addServiceController'); } return App.ModalPopup.show({ header: Em.I18n.t('services.service.config_groups_popup.header').format(displayName), bodyClass: App.MainServiceManageConfigGroupView.extend({ serviceName: serviceName, displayName: displayName, controller: configsController }), classNames: ['sixty-percent-width-modal', 'manage-configuration-group-popup'], primary: Em.I18n.t('common.save'), subViewController: configsController, /** * handle onPrimary action particularly in wizard * @param {Em.Controller} controller * @param {object} modifiedConfigGroups */ onPrimaryWizard: function (controller, modifiedConfigGroups) { controller.set('selectedService.configGroups', configsController.get('configGroups')); controller.selectedServiceObserver(); if (controller.get('name') == "wizardStep7Controller") { if (controller.get('selectedService.selected') === false && modifiedConfigGroups.toDelete.length > 0) { controller.setGroupsToDelete(modifiedConfigGroups.toDelete); } configsController.persistConfigGroups(); this.updateConfigGroupOnServicePage(); } this.hide(); }, /** * run requests which delete config group and clear its hosts * @param {Function} finishFunction * @param {object} modifiedConfigGroups */ runClearCGQueue: function (finishFunction, modifiedConfigGroups) { var counter = 0; var dfd = $.Deferred(); var doneFunction = function (xhr, text, errorThrown) { counter--; if (counter === 0) dfd.resolve(); finishFunction(xhr, text, errorThrown); }; modifiedConfigGroups.toClearHosts.forEach(function (cg) { counter++; var initialGroupState = modifiedConfigGroups.initialGroups.findProperty('name', cg.get('name')); configsController.clearConfigurationGroupHosts(cg, initialGroupState, doneFunction, doneFunction); }, this); modifiedConfigGroups.toDelete.forEach(function (cg) { counter++; configsController.deleteConfigurationGroup(cg, doneFunction, doneFunction); }, this); if (counter === 0) dfd.resolve(); return dfd.promise(); }, /** * run requests which change properties of config group * @param {Function} finishFunction * @param {object} modifiedConfigGroups */ runModifyCGQueue: function (finishFunction, modifiedConfigGroups) { var counter = 0; var dfd = $.Deferred(); var doneFunction = function (xhr, text, errorThrown) { counter--; if (counter === 0) dfd.resolve(); finishFunction(xhr, text, errorThrown); }; modifiedConfigGroups.toSetHosts.forEach(function (cg) { counter++; configsController.updateConfigurationGroup(cg, doneFunction, doneFunction); }, this); if (counter === 0) dfd.resolve(); return dfd.promise(); }, /** * run requests which create new config group * @param {Function} finishFunction * @param {object} modifiedConfigGroups */ runCreateCGQueue: function (finishFunction, modifiedConfigGroups) { var counter = 0; var dfd = $.Deferred(); var doneFunction = function (xhr, text, errorThrown) { counter--; if (counter === 0) dfd.resolve(); finishFunction(xhr, text, errorThrown); }; modifiedConfigGroups.toCreate.forEach(function (cg) { counter++; configsController.postNewConfigurationGroup(cg, doneFunction); }, this); if (counter === 0) dfd.resolve(); return dfd.promise(); }, onPrimary: function () { var modifiedConfigGroups = configsController.get('hostsModifiedConfigGroups'); var errors = []; var self = this; var finishFunction = function (xhr, text, errorThrown) { if (xhr && errorThrown) { var error = xhr.status + "(" + errorThrown + ") "; try { var json = $.parseJSON(xhr.responseText); error += json.message; } catch (err) { } errors.push(error); } }; // Save modified config-groups if (controller) { //called only in Wizard return this.onPrimaryWizard(controller, modifiedConfigGroups); } this.runClearCGQueue(finishFunction, modifiedConfigGroups).done(function () { self.runModifyCGQueue(finishFunction, modifiedConfigGroups).done(function () { self.runCreateCGQueue(finishFunction, modifiedConfigGroups).done(function () { if (errors.length > 0) { self.get('subViewController').set('errorMessage', errors.join(". ")); } else { self.updateConfigGroupOnServicePage(); self.hide(); } }); }); }); }, updateConfigGroupOnServicePage: function () { var selectedConfigGroup = configsController.get('selectedConfigGroup'); var managedConfigGroups = configsController.get('configGroups'); if (!controller) { controller = App.router.get('mainServiceInfoConfigsController'); //controller.set('configGroups', managedConfigGroups); controller.loadConfigGroups([controller.get('content.serviceName')]); } else { controller.set('selectedService.configGroups', managedConfigGroups); } var selectEventObject = {}; //check whether selectedConfigGroup exists if (selectedConfigGroup && controller.get('configGroups').someProperty('name', selectedConfigGroup.get('name'))) { selectEventObject.context = selectedConfigGroup; } else { selectEventObject.context = managedConfigGroups.findProperty('isDefault', true); } controller.selectConfigGroup(selectEventObject); }, updateButtons: function () { var modified = this.get('subViewController.isHostsModified'); this.set('disablePrimary', !modified); }.observes('subViewController.isHostsModified'), didInsertElement: Em.K }); } });