manage_config_groups_controller.js 20 KB

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