manage_config_groups_controller.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558
  1. /**
  2. * Licensed to the Apache Software Foundation (ASF) under one
  3. * or more contributor license agreements. See the NOTICE file
  4. * distributed with this work for additional information
  5. * regarding copyright ownership. The ASF licenses this file
  6. * to you under the Apache License, Version 2.0 (the
  7. * "License"); you may not use this file except in compliance
  8. * with the License. You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. */
  18. var App = require('app');
  19. var hostsManagement = require('utils/hosts');
  20. var numberUtils = require('utils/number_utils');
  21. App.ManageConfigGroupsController = Em.Controller.extend({
  22. name: 'manageConfigGroupsController',
  23. isLoaded: false,
  24. isInstaller: false,
  25. isAddService: false,
  26. serviceName: null,
  27. configGroups: [],
  28. originalConfigGroups: [],
  29. selectedConfigGroup: null,
  30. selectedHosts: [],
  31. clusterHosts: [],
  32. resortConfigGroup: function() {
  33. var configGroups = Ember.copy(this.get('configGroups'));
  34. if(configGroups.length < 2){
  35. return;
  36. }
  37. var defaultConfigGroup = configGroups.findProperty('isDefault');
  38. configGroups.removeObject(defaultConfigGroup);
  39. var sorted = [defaultConfigGroup].concat(configGroups.sortProperty('name'));
  40. this.removeObserver('configGroups.@each.name', this, 'resortConfigGroup');
  41. this.set('configGroups', sorted);
  42. this.addObserver('configGroups.@each.name', this, 'resortConfigGroup');
  43. }.observes('configGroups.@each.name'),
  44. loadHosts: function() {
  45. this.set('isLoaded', false);
  46. if (this.get('isInstaller')) {
  47. var allHosts = this.get('isAddService') ? App.router.get('addServiceController').get('allHosts') : App.router.get('installerController').get('allHosts');
  48. this.set('clusterHosts', allHosts);
  49. this.loadConfigGroups(this.get('serviceName'));
  50. } else {
  51. this.loadHostsFromServer();
  52. }
  53. },
  54. /**
  55. * request all hosts directly from server
  56. */
  57. loadHostsFromServer: function() {
  58. App.ajax.send({
  59. name: 'hosts.config_groups',
  60. sender: this,
  61. data: {},
  62. success: 'loadHostsFromServerSuccessCallback',
  63. error: 'loadHostsFromServerErrorCallback'
  64. });
  65. },
  66. /**
  67. * parse hosts response and wrap them into Ember.Object
  68. * @param data
  69. */
  70. loadHostsFromServerSuccessCallback: function (data) {
  71. var wrappedHosts = [];
  72. data.items.forEach(function (host) {
  73. var hostComponents = [];
  74. var diskInfo = host.Hosts.disk_info.filter(function(item) {
  75. return /^ext|^ntfs|^fat|^xfs/i.test(item.type);
  76. });
  77. if (diskInfo.length) {
  78. diskInfo = diskInfo.reduce(function(a, b) {
  79. return {
  80. available: parseInt(a.available) + parseInt(b.available),
  81. size: parseInt(a.size) + parseInt(b.size)
  82. };
  83. });
  84. }
  85. host.host_components.forEach(function (hostComponent) {
  86. hostComponents.push(Em.Object.create({
  87. componentName: hostComponent.HostRoles.component_name,
  88. displayName: App.format.role(hostComponent.HostRoles.component_name)
  89. }));
  90. }, this);
  91. wrappedHosts.pushObject(Em.Object.create({
  92. id: host.Hosts.host_name,
  93. ip: host.Hosts.ip,
  94. osType: host.Hosts.os_type,
  95. osArch: host.Hosts.os_arch,
  96. hostName: host.Hosts.host_name,
  97. publicHostName: host.Hosts.public_host_name,
  98. cpu: host.Hosts.cpu_count,
  99. memory: host.Hosts.total_mem,
  100. diskTotal: numberUtils.bytesToSize(diskInfo.size, 0, undefined, 1024),
  101. diskFree: numberUtils.bytesToSize(diskInfo.available, 0, undefined, 1024),
  102. disksMounted: host.Hosts.disk_info.length,
  103. hostComponents: hostComponents
  104. }
  105. ));
  106. }, this);
  107. this.set('clusterHosts', wrappedHosts);
  108. this.loadConfigGroups(this.get('serviceName'));
  109. },
  110. loadHostsFromServerErrorCallback: function () {
  111. console.warn('ERROR: request to fetch all hosts failed');
  112. this.set('clusterHosts', []);
  113. this.loadConfigGroups(this.get('serviceName'));
  114. },
  115. loadConfigGroups: function (serviceName) {
  116. if (this.get('isInstaller')) {
  117. this.set('serviceName', serviceName);
  118. var configGroups = this.copyConfigGroups(App.router.get('wizardStep7Controller.selectedService.configGroups'));
  119. var originalConfigGroups = this.copyConfigGroups(configGroups);
  120. this.set('configGroups', configGroups);
  121. this.set('originalConfigGroups', originalConfigGroups);
  122. this.set('isLoaded', true);
  123. } else {
  124. this.set('serviceName', serviceName);
  125. App.ajax.send({
  126. name: 'service.load_config_groups',
  127. data: {
  128. serviceName: serviceName
  129. },
  130. sender: this,
  131. success: 'onLoadConfigGroupsSuccess',
  132. error: 'onLoadConfigGroupsError'
  133. });
  134. }
  135. },
  136. onLoadConfigGroupsSuccess: function (data) {
  137. var usedHosts = [];
  138. var unusedHosts = [];
  139. var serviceName = this.get('serviceName');
  140. var serviceDisplayName = this.get('displayName');
  141. var defaultConfigGroup = App.ConfigGroup.create({
  142. name: serviceDisplayName + " Default",
  143. description: "Default cluster level " + this.get('serviceName') + " configuration",
  144. isDefault: true,
  145. parentConfigGroup: null,
  146. service: this.get('content'),
  147. configSiteTags: [],
  148. serviceName: serviceName
  149. });
  150. if (data && data.items) {
  151. var groupToTypeToTagMap = {};
  152. var configGroups = [];
  153. data.items.forEach(function (configGroup) {
  154. configGroup = configGroup.ConfigGroup;
  155. var hostNames = configGroup.hosts.mapProperty('host_name');
  156. var newConfigGroup = App.ConfigGroup.create({
  157. id: configGroup.id,
  158. name: configGroup.group_name,
  159. description: configGroup.description,
  160. isDefault: false,
  161. parentConfigGroup: defaultConfigGroup,
  162. service: App.Service.find().findProperty('serviceName', configGroup.tag),
  163. hosts: hostNames,
  164. configSiteTags: [],
  165. properties: [],
  166. apiResponse: configGroup
  167. });
  168. usedHosts = usedHosts.concat(newConfigGroup.get('hosts'));
  169. configGroups.push(newConfigGroup);
  170. var newConfigGroupSiteTags = newConfigGroup.get('configSiteTags');
  171. configGroup.desired_configs.forEach(function (config) {
  172. newConfigGroupSiteTags.push(App.ConfigSiteTag.create({
  173. site: config.type,
  174. tag: config.tag
  175. }));
  176. if (!groupToTypeToTagMap[configGroup.group_name]) {
  177. groupToTypeToTagMap[configGroup.group_name] = {}
  178. }
  179. groupToTypeToTagMap[configGroup.group_name][config.type] = config.tag;
  180. });
  181. }, this);
  182. unusedHosts = this.get('clusterHosts').mapProperty('hostName');
  183. usedHosts.uniq().forEach(function (host) {
  184. unusedHosts = unusedHosts.without(host);
  185. }, this);
  186. defaultConfigGroup.set('childConfigGroups', configGroups);
  187. defaultConfigGroup.set('hosts', unusedHosts);
  188. var allGroups = [defaultConfigGroup].concat(configGroups);
  189. this.set('configGroups', allGroups);
  190. var originalGroups = this.copyConfigGroups(allGroups);
  191. this.set('originalConfigGroups', originalGroups);
  192. this.loadProperties(groupToTypeToTagMap);
  193. this.set('isLoaded', true);
  194. }
  195. },
  196. onLoadConfigGroupsError: function () {
  197. console.error('Unable to load config groups for service.');
  198. },
  199. loadProperties: function (groupToTypeToTagMap) {
  200. var typeTagToGroupMap = {};
  201. var urlParams = [];
  202. for (var group in groupToTypeToTagMap) {
  203. var overrideTypeTags = groupToTypeToTagMap[group];
  204. for (var type in overrideTypeTags) {
  205. var tag = overrideTypeTags[type];
  206. typeTagToGroupMap[type + "///" + tag] = group;
  207. urlParams.push('(type=' + type + '&tag=' + tag + ')');
  208. }
  209. }
  210. var params = urlParams.join('|');
  211. if (urlParams.length) {
  212. App.ajax.send({
  213. name: 'config.host_overrides',
  214. sender: this,
  215. data: {
  216. params: params,
  217. typeTagToGroupMap: typeTagToGroupMap
  218. },
  219. success: 'onLoadPropertiesSuccess'
  220. });
  221. }
  222. },
  223. onLoadPropertiesSuccess: function (data, opt, params) {
  224. data.items.forEach(function (configs) {
  225. var typeTagConfigs = [];
  226. App.config.loadedConfigurationsCache[configs.type + "_" + configs.tag] = configs.properties;
  227. var group = params.typeTagToGroupMap[configs.type + "///" + configs.tag];
  228. for (var config in configs.properties) {
  229. typeTagConfigs.push({
  230. name: config,
  231. value: configs.properties[config]
  232. });
  233. }
  234. this.get('configGroups').findProperty('name', group).get('properties').pushObjects(typeTagConfigs);
  235. }, this);
  236. },
  237. showProperties: function () {
  238. var properies = this.get('selectedConfigGroup.propertiesList').htmlSafe();
  239. if (properies) {
  240. App.showAlertPopup(Em.I18n.t('services.service.config_groups_popup.properties'), properies);
  241. }
  242. },
  243. addHosts: function () {
  244. if (this.get('selectedConfigGroup.isAddHostsDisabled')){
  245. return false;
  246. }
  247. var availableHosts = this.get('selectedConfigGroup.availableHosts');
  248. var popupDescription = {
  249. header: Em.I18n.t('hosts.selectHostsDialog.title'),
  250. dialogMessage: Em.I18n.t('hosts.selectHostsDialog.message').format(this.get('displayName'))
  251. };
  252. hostsManagement.launchHostsSelectionDialog(availableHosts, [], false, this.get('componentsForFilter'), this.addHostsCallback.bind(this), popupDescription);
  253. },
  254. /**
  255. * add hosts callback
  256. */
  257. addHostsCallback: function (selectedHosts) {
  258. var group = this.get('selectedConfigGroup');
  259. if (selectedHosts) {
  260. var defaultHosts = group.get('parentConfigGroup.hosts').slice();
  261. var configGroupHosts = group.get('hosts');
  262. selectedHosts.forEach(function (hostName) {
  263. configGroupHosts.pushObject(hostName);
  264. defaultHosts.removeObject(hostName);
  265. });
  266. group.set('parentConfigGroup.hosts', defaultHosts);
  267. }
  268. },
  269. /**
  270. * delete hosts from group
  271. */
  272. deleteHosts: function () {
  273. if (this.get('isDeleteHostsDisabled')) {
  274. return;
  275. }
  276. var groupHosts = this.get('selectedConfigGroup.hosts');
  277. var defaultGroupHosts = this.get('selectedConfigGroup.parentConfigGroup.hosts').slice();
  278. this.get('selectedHosts').slice().forEach(function (hostName) {
  279. defaultGroupHosts.pushObject(hostName);
  280. groupHosts.removeObject(hostName);
  281. });
  282. this.set('selectedConfigGroup.parentConfigGroup.hosts', defaultGroupHosts);
  283. this.set('selectedHosts', []);
  284. },
  285. isDeleteHostsDisabled: function () {
  286. var selectedConfigGroup = this.get('selectedConfigGroup');
  287. if (selectedConfigGroup) {
  288. return selectedConfigGroup.isDefault || this.get('selectedHosts').length === 0;
  289. }
  290. return true;
  291. }.property('selectedConfigGroup', 'selectedConfigGroup.hosts.length', 'selectedHosts.length'),
  292. /**
  293. * confirm delete config group
  294. */
  295. confirmDelete : function () {
  296. var self = this;
  297. App.showConfirmationPopup(function() {
  298. self.deleteConfigGroup();
  299. });
  300. },
  301. /**
  302. * add hosts to group
  303. * @return {Array}
  304. */
  305. componentsForFilter: function () {
  306. return App.StackServiceComponent.find().filterProperty('serviceName', this.get('serviceName')).map(function (component) {
  307. return Em.Object.create({
  308. displayName: component.get('displayName'),
  309. componentName: component.get('isClient') ? 'CLIENT' : component.get('componentName'),
  310. selected: false
  311. });
  312. });
  313. }.property('serviceName'),
  314. /**
  315. * delete selected config group
  316. */
  317. deleteConfigGroup: function () {
  318. var selectedConfigGroup = this.get('selectedConfigGroup');
  319. if (this.get('isDeleteGroupDisabled')) {
  320. return;
  321. }
  322. //move hosts of group to default group (available hosts)
  323. this.set('selectedHosts', selectedConfigGroup.get('hosts'));
  324. this.deleteHosts();
  325. this.get('configGroups').removeObject(selectedConfigGroup);
  326. this.set('selectedConfigGroup', this.get('configGroups').findProperty('isDefault'));
  327. },
  328. /**
  329. * rename new config group
  330. */
  331. renameConfigGroup: function () {
  332. if(this.get('selectedConfigGroup.isDefault')) {
  333. return;
  334. }
  335. var self = this;
  336. this.renameGroupPopup = App.ModalPopup.show({
  337. primary: Em.I18n.t('ok'),
  338. secondary: Em.I18n.t('common.cancel'),
  339. header: Em.I18n.t('services.service.config_groups.rename_config_group_popup.header'),
  340. bodyClass: Ember.View.extend({
  341. templateName: require('templates/main/service/new_config_group')
  342. }),
  343. configGroupName: self.get('selectedConfigGroup.name'),
  344. configGroupDesc: self.get('selectedConfigGroup.description'),
  345. warningMessage: '',
  346. isDescriptionDirty: false,
  347. validate: function () {
  348. var warningMessage = '';
  349. var originalGroup = self.get('selectedConfigGroup');
  350. if (originalGroup.get('description') !== this.get('configGroupDesc') && !this.get('isDescriptionDirty')) {
  351. this.set('isDescriptionDirty', true);
  352. }
  353. if (originalGroup.get('name').trim() === this.get('configGroupName').trim()) {
  354. if (this.get('isDescriptionDirty')) {
  355. warningMessage = '';
  356. } else {
  357. warningMessage = Em.I18n.t("config.group.selection.dialog.err.name.exists");
  358. }
  359. } else {
  360. if (self.get('configGroups').mapProperty('name').contains(this.get('configGroupName').trim())) {
  361. warningMessage = Em.I18n.t("config.group.selection.dialog.err.name.exists");
  362. }
  363. }
  364. this.set('warningMessage', warningMessage);
  365. }.observes('configGroupName', 'configGroupDesc'),
  366. disablePrimary: function () {
  367. return !(this.get('configGroupName').trim().length > 0 && !this.get('warningMessage'));
  368. }.property('warningMessage', 'configGroupName', 'configGroupDesc'),
  369. onPrimary: function () {
  370. self.set('selectedConfigGroup.name', this.get('configGroupName'));
  371. self.set('selectedConfigGroup.description', this.get('configGroupDesc'));
  372. self.get('selectedConfigGroup.properties').forEach(function(property){
  373. property.set('group', self.get('selectedConfigGroup'));
  374. });
  375. this.hide();
  376. }
  377. });
  378. this.get('renameGroupPopup').validate();
  379. },
  380. /**
  381. * add new config group
  382. */
  383. addConfigGroup: function (duplicated) {
  384. duplicated = (duplicated === true);
  385. var self = this;
  386. this.addGroupPopup = App.ModalPopup.show({
  387. primary: Em.I18n.t('ok'),
  388. secondary: Em.I18n.t('common.cancel'),
  389. header: Em.I18n.t('services.service.config_groups.add_config_group_popup.header'),
  390. bodyClass: Ember.View.extend({
  391. templateName: require('templates/main/service/new_config_group')
  392. }),
  393. configGroupName: duplicated ? self.get('selectedConfigGroup.name') + ' Copy' : "",
  394. configGroupDesc: duplicated ? self.get('selectedConfigGroup.description') + ' (Copy)' : "",
  395. warningMessage: '',
  396. didInsertElement: function(){
  397. this.validate();
  398. this.$('input').focus();
  399. },
  400. validate: function () {
  401. var warningMessage = '';
  402. if (self.get('configGroups').mapProperty('name').contains(this.get('configGroupName').trim())) {
  403. warningMessage = Em.I18n.t("config.group.selection.dialog.err.name.exists");
  404. }
  405. this.set('warningMessage', warningMessage);
  406. }.observes('configGroupName'),
  407. disablePrimary: function () {
  408. return !(this.get('configGroupName').trim().length > 0 && !this.get('warningMessage'));
  409. }.property('warningMessage', 'configGroupName'),
  410. onPrimary: function () {
  411. var defaultConfigGroup = self.get('configGroups').findProperty('isDefault');
  412. var properties = [];
  413. var newConfigGroupData = App.ConfigGroup.create({
  414. id: null,
  415. name: this.get('configGroupName').trim(),
  416. description: this.get('configGroupDesc'),
  417. isDefault: false,
  418. parentConfigGroup: defaultConfigGroup,
  419. service: Em.Object.create({id: self.get('serviceName')}),
  420. hosts: [],
  421. configSiteTags: [],
  422. properties: []
  423. });
  424. if (duplicated) {
  425. self.get('selectedConfigGroup.properties').forEach(function(property) {
  426. var property = App.ServiceConfigProperty.create($.extend(false, {}, property));
  427. property.set('group', newConfigGroupData);
  428. properties.push(property);
  429. });
  430. newConfigGroupData.set('properties', properties);
  431. } else {
  432. newConfigGroupData.set('properties', []);
  433. }
  434. self.get('configGroups').pushObject(newConfigGroupData);
  435. defaultConfigGroup.get('childConfigGroups').pushObject(newConfigGroupData);
  436. this.hide();
  437. }
  438. });
  439. },
  440. duplicateConfigGroup: function() {
  441. this.addConfigGroup(true);
  442. },
  443. hostsModifiedConfigGroups: function () {
  444. if (!this.get('isLoaded')) {
  445. return false;
  446. }
  447. var groupsToClearHosts = [];
  448. var groupsToDelete = [];
  449. var groupsToSetHosts = [];
  450. var groupsToCreate = [];
  451. var groups = this.get('configGroups');
  452. var originalGroups = this.get('originalConfigGroups');
  453. var originalGroupsNames = originalGroups.mapProperty('name').without(originalGroups.findProperty('isDefault').get('name'));
  454. groups.forEach(function (group) {
  455. if (!group.get('isDefault')) {
  456. var originalGroup = originalGroups.findProperty('name', group.get('name'));
  457. if (originalGroup) {
  458. if (!(JSON.stringify(group.get('hosts').slice().sort()) === JSON.stringify(originalGroup.get('hosts').sort()))) {
  459. groupsToClearHosts.push(group.set('id', originalGroup.get('id')));
  460. if (group.get('hosts').length) {
  461. groupsToSetHosts.push(group.set('id', originalGroup.get('id')));
  462. }
  463. } else if (group.get('description') !== originalGroup.get('description')) {
  464. groupsToSetHosts.push(group.set('id', originalGroup.get('id')));
  465. }
  466. originalGroupsNames = originalGroupsNames.without(group.get('name'));
  467. } else {
  468. groupsToCreate.push(group);
  469. }
  470. }
  471. });
  472. originalGroupsNames.forEach(function (groupName) {
  473. groupsToDelete.push(originalGroups.findProperty('name', groupName));
  474. }, this);
  475. return {
  476. toClearHosts: groupsToClearHosts,
  477. toDelete: groupsToDelete,
  478. toSetHosts: groupsToSetHosts,
  479. toCreate: groupsToCreate
  480. };
  481. }.property('selectedConfigGroup.hosts.@each', 'selectedConfigGroup.hosts.length', 'selectedConfigGroup.description', 'configGroups', 'isLoaded'),
  482. isHostsModified: function () {
  483. var modifiedGroups = this.get('hostsModifiedConfigGroups');
  484. if (!this.get('isLoaded')) {
  485. return false;
  486. }
  487. return !!(modifiedGroups.toClearHosts.length || modifiedGroups.toSetHosts.length || modifiedGroups.toCreate.length || modifiedGroups.toDelete.length);
  488. }.property('hostsModifiedConfigGroups'),
  489. /**
  490. * copy config groups to manage popup to give user choice whether or not save changes
  491. * @param originGroups
  492. * @return {Array}
  493. */
  494. copyConfigGroups: function (originGroups) {
  495. var configGroups = [];
  496. var result = [];
  497. var defaultConfigGroup = App.ConfigGroup.create($.extend(true, {}, originGroups.findProperty('isDefault')));
  498. originGroups.forEach(function (configGroup) {
  499. if (!configGroup.get('isDefault')) {
  500. var copiedGroup = App.ConfigGroup.create($.extend(true, {}, configGroup));
  501. copiedGroup.set('parentConfigGroup', defaultConfigGroup);
  502. configGroups.pushObject(copiedGroup);
  503. }
  504. });
  505. defaultConfigGroup.set('childConfigGroups', configGroups.slice());
  506. configGroups.pushObject(defaultConfigGroup);
  507. configGroups.forEach(function (group) {
  508. var groupCopy = {};
  509. for (var prop in group) {
  510. if (group.hasOwnProperty(prop)) {
  511. groupCopy[prop] = group[prop];
  512. }
  513. }
  514. groupCopy.properties.forEach(function(property){
  515. property.set('group', group);
  516. });
  517. result.push(App.ConfigGroup.create(groupCopy));
  518. }, this);
  519. return result;
  520. }
  521. });