stack_and_upgrade_controller.js 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940
  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 stringUtils = require('utils/string_utils');
  20. App.MainAdminStackAndUpgradeController = Em.Controller.extend(App.LocalStorage, {
  21. name: 'mainAdminStackAndUpgradeController',
  22. /**
  23. * @type {boolean}
  24. */
  25. isLoaded: false,
  26. /**
  27. * @type {object}
  28. * @default null
  29. */
  30. upgradeData: null,
  31. /**
  32. * @type {number}
  33. * @default null
  34. */
  35. upgradeId: null,
  36. /**
  37. * @type {string}
  38. * @default null
  39. */
  40. upgradeVersion: null,
  41. /**
  42. * @type {boolean}
  43. * @default false
  44. */
  45. isDowngrade: false,
  46. /**
  47. * version that currently applied to server
  48. * should be plain object, because stored to localStorage
  49. * @type {object|null}
  50. */
  51. currentVersion: null,
  52. /**
  53. * versions to which cluster could be upgraded
  54. * @type {Array}
  55. */
  56. targetVersions: [],
  57. /**
  58. * @type {boolean} true if some request that should disable actions is in progress
  59. */
  60. requestInProgress: false,
  61. /**
  62. * @type {boolean} true while no updated upgrade info is loaded after retry
  63. */
  64. isRetryPending: false,
  65. /**
  66. * properties that stored to localStorage to resume wizard progress
  67. */
  68. wizardStorageProperties: ['upgradeId', 'upgradeVersion', 'currentVersion', 'isDowngrade'],
  69. /**
  70. * mutable properties of Upgrade Task
  71. * @type {Array}
  72. */
  73. taskDetailsProperties: ['status', 'stdout', 'stderr', 'error_log', 'host_name', 'output_log'],
  74. isLoadUpgradeDataPending: false,
  75. /**
  76. * path to the mock json
  77. * @type {String}
  78. */
  79. mockRepoUrl: '/data/stack_versions/repo_versions_all.json',
  80. /**
  81. * api to get RepoVersions
  82. * @type {String}
  83. */
  84. realRepoUrl: function () {
  85. return App.get('apiPrefix') + App.get('stackVersionURL') +
  86. '/compatible_repository_versions?fields=*,operating_systems/*,operating_systems/repositories/*';
  87. }.property('App.stackVersionURL'),
  88. /**
  89. * path to the mock json
  90. * @type {String}
  91. */
  92. mockStackUrl: '/data/stack_versions/stack_version_all.json',
  93. /**
  94. * api to get ClusterStackVersions with repository_versions (use to init data load)
  95. * @type {String}
  96. */
  97. realStackUrl: function () {
  98. return App.get('apiPrefix') + '/clusters/' + App.get('clusterName') +
  99. '/stack_versions?fields=*,repository_versions/*,repository_versions/operating_systems/repositories/*';
  100. }.property('App.clusterName'),
  101. /**
  102. * api to get ClusterStackVersions without repository_versions (use to update data)
  103. * @type {String}
  104. */
  105. realUpdateUrl: function () {
  106. return App.get('apiPrefix') + '/clusters/' + App.get('clusterName') + '/stack_versions?fields=ClusterStackVersions/*';
  107. }.property('App.clusterName'),
  108. /**
  109. * Determines if list of services with checks that failed and were skipped by user during the upgrade is loaded
  110. * @type {boolean}
  111. */
  112. areSkippedServiceChecksLoaded: false,
  113. /**
  114. * List of services with checks that failed and were skipped by user during the upgrade
  115. * @type {array}
  116. */
  117. skippedServiceChecks: [],
  118. /**
  119. * status of tasks/items/groups which should be grayed out and disabled
  120. * @type {Array}
  121. */
  122. nonActiveStates: ['PENDING', 'ABORTED'],
  123. init: function () {
  124. this.initDBProperties();
  125. },
  126. /**
  127. * restore data from localStorage
  128. */
  129. initDBProperties: function () {
  130. this.get('wizardStorageProperties').forEach(function (property) {
  131. if (this.getDBProperty(property)) {
  132. this.set(property, this.getDBProperty(property));
  133. }
  134. }, this);
  135. },
  136. /**
  137. * load all data:
  138. * - upgrade data
  139. * - stack versions
  140. * - repo versions
  141. */
  142. load: function () {
  143. var dfd = $.Deferred();
  144. var self = this;
  145. this.loadUpgradeData(true).done(function() {
  146. self.loadStackVersionsToModel(true).done(function () {
  147. self.loadRepoVersionsToModel().done(function() {
  148. var currentVersion = App.StackVersion.find().findProperty('state', 'CURRENT');
  149. if (currentVersion) {
  150. self.set('currentVersion', {
  151. repository_version: currentVersion.get('repositoryVersion.repositoryVersion'),
  152. repository_name: currentVersion.get('repositoryVersion.displayName')
  153. });
  154. }
  155. dfd.resolve();
  156. });
  157. });
  158. });
  159. return dfd.promise();
  160. },
  161. /**
  162. * load upgrade tasks by upgrade id
  163. * @return {$.Deferred}
  164. * @param {boolean} onlyState
  165. */
  166. loadUpgradeData: function (onlyState) {
  167. var upgradeId = this.get('upgradeId'),
  168. deferred = $.Deferred(),
  169. self = this;
  170. if (Em.isNone(upgradeId)) {
  171. deferred.resolve();
  172. console.log('Upgrade in INIT state');
  173. } else {
  174. this.set('isLoadUpgradeDataPending', true);
  175. App.ajax.send({
  176. name: (onlyState) ? 'admin.upgrade.state' : 'admin.upgrade.data',
  177. sender: this,
  178. data: {
  179. id: upgradeId
  180. },
  181. success: 'loadUpgradeDataSuccessCallback'
  182. }).then(deferred.resolve).complete(function () {
  183. self.set('isLoadUpgradeDataPending', false);
  184. });
  185. }
  186. return deferred.promise();
  187. },
  188. /**
  189. * parse and push upgrade tasks to controller
  190. * @param data
  191. */
  192. loadUpgradeDataSuccessCallback: function (data) {
  193. if (Em.isNone(data)) return;
  194. App.set('upgradeState', data.Upgrade.request_status);
  195. this.setDBProperty('upgradeState', data.Upgrade.request_status);
  196. if (data.upgrade_groups) {
  197. this.updateUpgradeData(data);
  198. }
  199. if (this.get('isRetryPending') && data.Upgrade.request_status != 'ABORTED') {
  200. this.setProperties({
  201. requestInProgress: false,
  202. isRetryPending: false
  203. });
  204. }
  205. },
  206. /**
  207. * update data of Upgrade
  208. * @param {object} newData
  209. */
  210. updateUpgradeData: function (newData) {
  211. var oldData = this.get('upgradeData'),
  212. nonActiveStates = this.get('nonActiveStates'),
  213. groupsMap = {},
  214. itemsMap = {};
  215. if (Em.isNone(oldData) || (newData.upgrade_groups.length !== oldData.upgradeGroups.length)) {
  216. this.initUpgradeData(newData);
  217. } else {
  218. //create entities maps
  219. newData.upgrade_groups.forEach(function (newGroup) {
  220. groupsMap[newGroup.UpgradeGroup.group_id] = newGroup.UpgradeGroup;
  221. newGroup.upgrade_items.forEach(function (item) {
  222. itemsMap[item.UpgradeItem.stage_id] = item.UpgradeItem;
  223. })
  224. });
  225. //update existed entities with new data
  226. oldData.upgradeGroups.forEach(function (oldGroup) {
  227. oldGroup.set('status', groupsMap[oldGroup.get('group_id')].status);
  228. oldGroup.set('progress_percent', groupsMap[oldGroup.get('group_id')].progress_percent);
  229. oldGroup.set('completed_task_count', groupsMap[oldGroup.get('group_id')].completed_task_count);
  230. oldGroup.upgradeItems.forEach(function (item) {
  231. item.set('status', itemsMap[item.get('stage_id')].status);
  232. item.set('progress_percent', itemsMap[item.get('stage_id')].progress_percent);
  233. });
  234. var hasExpandableItems = oldGroup.upgradeItems.some(function (item) {
  235. return !nonActiveStates.contains(item.get('status'));
  236. });
  237. oldGroup.set('hasExpandableItems', hasExpandableItems);
  238. });
  239. oldData.set('Upgrade', newData.Upgrade);
  240. }
  241. },
  242. /**
  243. * change structure of Upgrade
  244. * In order to maintain nested views in template object should have direct link to its properties, for example
  245. * item.UpgradeItem.<properties> -> item.<properties>
  246. * @param {object} newData
  247. */
  248. initUpgradeData: function (newData) {
  249. var upgradeGroups = [],
  250. nonActiveStates = this.get('nonActiveStates');
  251. //wrap all entities into App.upgradeEntity
  252. newData.upgrade_groups.forEach(function (newGroup) {
  253. var hasExpandableItems = newGroup.upgrade_items.some(function (item) {
  254. return !nonActiveStates.contains(item.UpgradeItem.status);
  255. }),
  256. oldGroup = App.upgradeEntity.create({type: 'GROUP', hasExpandableItems: hasExpandableItems}, newGroup.UpgradeGroup),
  257. upgradeItems = [];
  258. newGroup.upgrade_items.forEach(function (item) {
  259. var oldItem = App.upgradeEntity.create({type: 'ITEM'}, item.UpgradeItem);
  260. oldItem.set('tasks', []);
  261. upgradeItems.pushObject(oldItem);
  262. });
  263. upgradeItems.reverse();
  264. oldGroup.set('upgradeItems', upgradeItems);
  265. upgradeGroups.pushObject(oldGroup);
  266. });
  267. upgradeGroups.reverse();
  268. this.set('upgradeData', Em.Object.create({
  269. upgradeGroups: upgradeGroups,
  270. Upgrade: newData.Upgrade
  271. }));
  272. },
  273. /**
  274. * request Upgrade Item and its tasks from server
  275. * @return {$.ajax}
  276. */
  277. getUpgradeItem: function (item) {
  278. return App.ajax.send({
  279. name: 'admin.upgrade.upgrade_item',
  280. sender: this,
  281. data: {
  282. upgradeId: item.get('request_id'),
  283. groupId: item.get('group_id'),
  284. stageId: item.get('stage_id')
  285. },
  286. success: 'getUpgradeItemSuccessCallback'
  287. });
  288. },
  289. /**
  290. * success callback of <code>getTasks</code>
  291. * @param {object} data
  292. */
  293. getUpgradeItemSuccessCallback: function (data) {
  294. this.get('upgradeData.upgradeGroups').forEach(function (group) {
  295. if (group.get('group_id') === data.UpgradeItem.group_id) {
  296. group.get('upgradeItems').forEach(function (item) {
  297. if (item.get('stage_id') === data.UpgradeItem.stage_id) {
  298. if (item.get('tasks.length')) {
  299. item.set('isTasksLoaded', true);
  300. data.tasks.forEach(function (task) {
  301. var currentTask = item.get('tasks').findProperty('id', task.Tasks.id);
  302. this.get('taskDetailsProperties').forEach(function (property) {
  303. currentTask.set(property, task.Tasks[property]);
  304. }, this);
  305. }, this);
  306. } else {
  307. var tasks = [];
  308. data.tasks.forEach(function (task) {
  309. tasks.pushObject(App.upgradeEntity.create({type: 'TASK'}, task.Tasks));
  310. });
  311. item.set('tasks', tasks);
  312. }
  313. item.set('isTasksLoaded', true);
  314. }
  315. }, this);
  316. }
  317. }, this);
  318. },
  319. /**
  320. * downgrade confirmation popup
  321. * @param {object} event
  322. */
  323. confirmDowngrade: function (event) {
  324. var self = this;
  325. var currentVersion = this.get('currentVersion');
  326. return App.showConfirmationPopup(
  327. function() {
  328. self.downgrade.call(self, currentVersion, event);
  329. },
  330. Em.I18n.t('admin.stackUpgrade.downgrade.body').format(currentVersion.repository_name),
  331. null,
  332. Em.I18n.t('admin.stackUpgrade.dialog.downgrade.header').format(currentVersion.repository_name),
  333. Em.I18n.t('admin.stackUpgrade.downgrade.proceed')
  334. );
  335. },
  336. /**
  337. * make call to start downgrade process
  338. * @param {object} currentVersion
  339. * @param {object} event
  340. */
  341. downgrade: function (currentVersion, event) {
  342. this.set('requestInProgress', true);
  343. this.abortUpgrade();
  344. App.ajax.send({
  345. name: 'admin.downgrade.start',
  346. sender: this,
  347. data: {
  348. from: App.RepositoryVersion.find().findProperty('displayName', this.get('upgradeVersion')).get('repositoryVersion'),
  349. value: currentVersion.repository_version,
  350. label: currentVersion.repository_name,
  351. isDowngrade: true
  352. },
  353. success: 'upgradeSuccessCallback',
  354. callback: function() {
  355. this.sender.set('requestInProgress', false);
  356. }
  357. });
  358. },
  359. /**
  360. * abort upgrade (in order to start Downgrade)
  361. */
  362. abortUpgrade: function () {
  363. return App.ajax.send({
  364. name: 'admin.upgrade.abort',
  365. sender: this,
  366. data: {
  367. upgradeId: this.get('upgradeId')
  368. }
  369. });
  370. },
  371. retryUpgrade: function () {
  372. this.setProperties({
  373. requestInProgress: true,
  374. isRetryPending: true
  375. });
  376. return App.ajax.send({
  377. name: 'admin.upgrade.retry',
  378. sender: this,
  379. data: {
  380. upgradeId: this.get('upgradeId')
  381. }
  382. });
  383. },
  384. /**
  385. * make call to start upgrade process and show popup with current progress
  386. * @param {object} version
  387. */
  388. upgrade: function (version) {
  389. this.set('requestInProgress', true);
  390. App.ajax.send({
  391. name: 'admin.upgrade.start',
  392. sender: this,
  393. data: version,
  394. success: 'upgradeSuccessCallback',
  395. callback: function() {
  396. this.sender.set('requestInProgress', false);
  397. }
  398. });
  399. this.setDBProperty('currentVersion', this.get('currentVersion'));
  400. },
  401. /**
  402. * success callback of <code>upgrade()</code>
  403. * @param {object} data
  404. */
  405. upgradeSuccessCallback: function (data, opt, params) {
  406. this.set('upgradeData', null);
  407. this.set('upgradeId', data.resources[0].Upgrade.request_id);
  408. this.set('upgradeVersion', params.label);
  409. this.set('isDowngrade', !!params.isDowngrade);
  410. this.setDBProperty('upgradeVersion', params.label);
  411. this.setDBProperty('upgradeId', data.resources[0].Upgrade.request_id);
  412. this.setDBProperty('upgradeState', 'PENDING');
  413. this.setDBProperty('isDowngrade', !!params.isDowngrade);
  414. App.set('upgradeState', 'PENDING');
  415. App.clusterStatus.setClusterStatus({
  416. wizardControllerName: this.get('name'),
  417. localdb: App.db.data
  418. });
  419. this.load();
  420. this.openUpgradeDialog();
  421. },
  422. /**
  423. * upgrade confirmation popup
  424. * @param {object} version
  425. * @return App.ModalPopup
  426. */
  427. confirmUpgrade: function (version) {
  428. var self = this;
  429. return App.showConfirmationPopup(
  430. function () {
  431. self.runPreUpgradeCheck.call(self, version);
  432. },
  433. Em.I18n.t('admin.stackUpgrade.upgrade.confirm.body').format(version.get('displayName')),
  434. null,
  435. Em.I18n.t('admin.stackUpgrade.dialog.header').format(version.get('displayName'))
  436. );
  437. },
  438. /**
  439. * send request for pre upgrade check
  440. * @param version
  441. */
  442. runPreUpgradeCheck: function(version) {
  443. var params = {
  444. value: version.get('repositoryVersion'),
  445. label: version.get('displayName')
  446. };
  447. if (App.get('supports.preUpgradeCheck')) {
  448. this.set('requestInProgress', true);
  449. App.ajax.send({
  450. name: "admin.rolling_upgrade.pre_upgrade_check",
  451. sender: this,
  452. data: params,
  453. success: "runPreUpgradeCheckSuccess",
  454. error: "runPreUpgradeCheckError"
  455. });
  456. } else {
  457. this.upgrade(params);
  458. }
  459. },
  460. /**
  461. * success callback of <code>runPreUpgradeCheckSuccess()</code>
  462. * if there are some fails - it shows popup else run upgrade
  463. * @param data {object}
  464. * @param opt {object}
  465. * @param params {object}
  466. * @returns {App.ModalPopup|undefined}
  467. */
  468. runPreUpgradeCheckSuccess: function (data, opt, params) {
  469. var self = this;
  470. if (data.items.someProperty('UpgradeChecks.status', 'FAIL') || data.items.someProperty('UpgradeChecks.status', 'WARNING')) {
  471. this.set('requestInProgress', false);
  472. var header = Em.I18n.t('popup.clusterCheck.Upgrade.header').format(params.label),
  473. failTitle = Em.I18n.t('popup.clusterCheck.Upgrade.fail.title'),
  474. failAlert = new Em.Handlebars.SafeString(Em.I18n.t('popup.clusterCheck.Upgrade.fail.alert')),
  475. warningTitle = Em.I18n.t('popup.clusterCheck.Upgrade.warning.title'),
  476. warningAlert = new Em.Handlebars.SafeString(Em.I18n.t('popup.clusterCheck.Upgrade.warning.alert')),
  477. configsMergeWarning = data.items.findProperty('UpgradeChecks.id', "CONFIG_MERGE"),
  478. configs = [];
  479. if (configsMergeWarning && Em.get(configsMergeWarning, 'UpgradeChecks.status') === 'WARNING') {
  480. data.items = data.items.rejectProperty('UpgradeChecks.id', 'CONFIG_MERGE');
  481. var configsMergeCheckData = Em.get(configsMergeWarning, 'UpgradeChecks.failed_detail');
  482. if (configsMergeCheckData) {
  483. configs = configsMergeCheckData.map(function (item) {
  484. var isDeprecated = Em.isNone(item.new_stack_value),
  485. willBeRemoved = Em.isNone(item.result_value);
  486. return {
  487. type: item.type,
  488. name: item.property,
  489. currentValue: item.current,
  490. recommendedValue: isDeprecated ? Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.deprecated') : item.new_stack_value,
  491. isDeprecated: isDeprecated,
  492. resultingValue: willBeRemoved ? Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.willBeRemoved') : item.result_value,
  493. willBeRemoved: willBeRemoved
  494. };
  495. });
  496. }
  497. }
  498. App.showClusterCheckPopup(data, header, failTitle, failAlert, warningTitle, warningAlert, function () {
  499. self.upgrade(params);
  500. }, configs, params.label);
  501. } else {
  502. this.upgrade(params);
  503. }
  504. },
  505. runPreUpgradeCheckError: function() {
  506. this.set('requestInProgress', false);
  507. },
  508. confirmRetryUpgrade: function (version) {
  509. var self = this;
  510. return App.showConfirmationPopup(
  511. function () {
  512. self.retryUpgrade();
  513. },
  514. Em.I18n.t('admin.stackUpgrade.upgrade.retry.confirm.body').format(version.get('displayName')),
  515. null,
  516. Em.I18n.t('admin.stackUpgrade.dialog.header').format(version.get('displayName'))
  517. );
  518. },
  519. confirmRetryDowngrade: function () {
  520. var self = this,
  521. currentVersion = this.get('currentVersion');
  522. return App.showConfirmationPopup(
  523. function() {
  524. self.retryUpgrade();
  525. },
  526. Em.I18n.t('admin.stackUpgrade.downgrade.retry.body').format(currentVersion.repository_name),
  527. null,
  528. Em.I18n.t('admin.stackUpgrade.dialog.downgrade.header').format(currentVersion.repository_name),
  529. Em.I18n.t('admin.stackUpgrade.downgrade.proceed')
  530. );
  531. },
  532. /**
  533. * confirmation popup before install repository version
  534. */
  535. installRepoVersionConfirmation: function (repo) {
  536. var self = this;
  537. return App.showConfirmationPopup(function () {
  538. self.installRepoVersion(repo);
  539. },
  540. Em.I18n.t('admin.stackVersions.version.install.confirm').format(repo.get('displayName'))
  541. );
  542. },
  543. /**
  544. * sends request to install repoVersion to the cluster
  545. * and create clusterStackVersion resourse
  546. * @param {Em.Object} repo
  547. * @return {$.ajax}
  548. * @method installRepoVersion
  549. */
  550. installRepoVersion: function (repo) {
  551. this.set('requestInProgress', true);
  552. var data = {
  553. ClusterStackVersions: {
  554. stack: repo.get('stackVersionType'),
  555. version: repo.get('stackVersionNumber'),
  556. repository_version: repo.get('repositoryVersion')
  557. },
  558. id: repo.get('id')
  559. };
  560. return App.ajax.send({
  561. name: 'admin.stack_version.install.repo_version',
  562. sender: this,
  563. data: data,
  564. success: 'installRepoVersionSuccess',
  565. callback: function() {
  566. this.sender.set('requestInProgress', false);
  567. }
  568. });
  569. },
  570. /**
  571. * transform repo data into json for
  572. * saving changes to repository version
  573. * @param {Em.Object} repo
  574. * @returns {{operating_systems: Array}}
  575. */
  576. prepareRepoForSaving: function(repo) {
  577. var repoVersion = { "operating_systems": [] };
  578. repo.get('operatingSystems').forEach(function (os, k) {
  579. repoVersion.operating_systems.push({
  580. "OperatingSystems": {
  581. "os_type": os.get("osType")
  582. },
  583. "repositories": []
  584. });
  585. os.get('repositories').forEach(function (repository) {
  586. repoVersion.operating_systems[k].repositories.push({
  587. "Repositories": {
  588. "base_url": repository.get('baseUrl'),
  589. "repo_id": repository.get('repoId'),
  590. "repo_name": repository.get('repoName')
  591. }
  592. });
  593. });
  594. });
  595. return repoVersion;
  596. },
  597. /**
  598. * Return stack version for the repo object
  599. * @param {Em.Object} repo
  600. * */
  601. getStackVersionNumber: function(repo){
  602. var stackVersionNumber = repo.get('stackVersion');
  603. if(null == stackVersionNumber)
  604. stackVersionNumber = App.get('currentStackVersion');
  605. return stackVersionNumber;
  606. },
  607. /**
  608. * perform validation if <code>skip<code> is false and run save if
  609. * validation successfull or run save without validation is <code>skip<code> is true
  610. * @param {Em.Object} repo
  611. * @param {boolean} skip
  612. * @returns {$.Deferred}
  613. */
  614. saveRepoOS: function (repo, skip) {
  615. var self = this;
  616. var deferred = $.Deferred();
  617. this.validateRepoVersions(repo, skip).done(function(data) {
  618. if (data.length > 0) {
  619. deferred.resolve(data);
  620. } else {
  621. var repoVersion = self.prepareRepoForSaving(repo);
  622. var stackVersionNumber = self.getStackVersionNumber(repo);
  623. console.log("Repository stack version:"+stackVersionNumber);
  624. App.ajax.send({
  625. name: 'admin.stack_versions.edit.repo',
  626. sender: this,
  627. data: {
  628. stackName: App.get('currentStackName'),
  629. stackVersion: stackVersionNumber,
  630. repoVersionId: repo.get('repoVersionId'),
  631. repoVersion: repoVersion
  632. }
  633. }).success(function() {
  634. deferred.resolve([]);
  635. });
  636. }
  637. });
  638. return deferred.promise();
  639. },
  640. /**
  641. * send request for validation for each repository
  642. * @param {Em.Object} repo
  643. * @param {boolean} skip
  644. * @returns {*}
  645. */
  646. validateRepoVersions: function(repo, skip) {
  647. var deferred = $.Deferred(),
  648. totalCalls = 0,
  649. invalidUrls = [];
  650. if (skip) {
  651. deferred.resolve(invalidUrls);
  652. } else {
  653. var stackVersionNumber = this.getStackVersionNumber(repo);
  654. repo.get('operatingSystems').forEach(function (os) {
  655. if (os.get('isSelected')) {
  656. os.get('repositories').forEach(function (repo) {
  657. totalCalls++;
  658. App.ajax.send({
  659. name: 'admin.stack_versions.validate.repo',
  660. sender: this,
  661. data: {
  662. repo: repo,
  663. repoId: repo.get('repoId'),
  664. baseUrl: repo.get('baseUrl'),
  665. osType: os.get('osType'),
  666. stackName: App.get('currentStackName'),
  667. stackVersion: stackVersionNumber
  668. }
  669. })
  670. .success(function () {
  671. totalCalls--;
  672. if (totalCalls === 0) deferred.resolve(invalidUrls);
  673. })
  674. .error(function () {
  675. repo.set('hasError', true);
  676. invalidUrls.push(repo);
  677. totalCalls--;
  678. if (totalCalls === 0) deferred.resolve(invalidUrls);
  679. });
  680. });
  681. } else {
  682. return deferred.resolve(invalidUrls);
  683. }
  684. });
  685. }
  686. return deferred.promise();
  687. },
  688. /**
  689. * success callback for <code>installRepoVersion()<code>
  690. * saves request id to the db
  691. * @param data
  692. * @param opt
  693. * @param params
  694. * @method installStackVersionSuccess
  695. */
  696. installRepoVersionSuccess: function (data, opt, params) {
  697. var version = App.RepositoryVersion.find(params.id);
  698. App.db.set('repoVersionInstall', 'id', [data.Requests.id]);
  699. App.clusterStatus.setClusterStatus({
  700. wizardControllerName: this.get('name'),
  701. localdb: App.db.data
  702. });
  703. version.set('defaultStatus', 'INSTALLING');
  704. if (version.get('stackVersion')) {
  705. version.set('stackVersion.state', 'INSTALLING');
  706. }
  707. },
  708. /**
  709. * opens a popup with installations state per host
  710. * @param {Em.Object} version
  711. * @method showProgressPopup
  712. */
  713. showProgressPopup: function(version) {
  714. var popupTitle = Em.I18n.t('admin.stackVersions.details.install.hosts.popup.title').format(version.get('displayName'));
  715. var requestIds = App.get('testMode') ? [1] : App.db.get('repoVersionInstall', 'id');
  716. var hostProgressPopupController = App.router.get('highAvailabilityProgressPopupController');
  717. hostProgressPopupController.initPopup(popupTitle, requestIds, this);
  718. },
  719. /**
  720. * reset upgradeState to INIT when upgrade is COMPLETED
  721. * and clean auxiliary data
  722. */
  723. finish: function () {
  724. if (App.get('upgradeState') === 'COMPLETED') {
  725. this.setDBProperty('upgradeId', undefined);
  726. this.setDBProperty('upgradeState', 'INIT');
  727. this.setDBProperty('upgradeVersion', undefined);
  728. this.setDBProperty('currentVersion', undefined);
  729. this.setDBProperty('isDowngrade', undefined);
  730. App.clusterStatus.setClusterStatus({
  731. localdb: App.db.data
  732. });
  733. App.set('upgradeState', 'INIT');
  734. }
  735. }.observes('App.upgradeState'),
  736. /**
  737. * show dialog with tasks of upgrade
  738. * @return {App.ModalPopup}
  739. */
  740. openUpgradeDialog: function () {
  741. App.router.transitionTo('admin.stackUpgrade');
  742. },
  743. /**
  744. * returns url to get data for repoVersion or clusterStackVersion
  745. * @param {Boolean} stack true if load clusterStackVersion
  746. * @param {Boolean} fullLoad true if load all data
  747. * @returns {String}
  748. * @method getUrl
  749. */
  750. getUrl: function(stack, fullLoad) {
  751. if (App.get('testMode')) {
  752. return stack ? this.get('mockStackUrl') : this.get('mockRepoUrl')
  753. } else {
  754. if (fullLoad) {
  755. return stack ? this.get('realStackUrl') : this.get('realRepoUrl');
  756. } else {
  757. return this.get('realUpdateUrl');
  758. }
  759. }
  760. },
  761. /**
  762. * get stack versions from server and push it to model
  763. * @return {*}
  764. * @method loadStackVersionsToModel
  765. */
  766. loadStackVersionsToModel: function (fullLoad) {
  767. var dfd = $.Deferred();
  768. App.HttpClient.get(this.getUrl(true, fullLoad), App.stackVersionMapper, {
  769. complete: function () {
  770. dfd.resolve();
  771. }
  772. });
  773. return dfd.promise();
  774. },
  775. /**
  776. * get repo versions from server and push it to model
  777. * @return {*}
  778. * @params {Boolean} isUpdate - if true loads part of data that need to be updated
  779. * @method loadRepoVersionsToModel()
  780. */
  781. loadRepoVersionsToModel: function () {
  782. var dfd = $.Deferred();
  783. App.HttpClient.get(this.getUrl(false, true), App.repoVersionMapper, {
  784. complete: function () {
  785. dfd.resolve();
  786. }
  787. });
  788. return dfd.promise();
  789. },
  790. /**
  791. * set status to Upgrade item
  792. * @param item
  793. * @param status
  794. */
  795. setUpgradeItemStatus: function(item, status) {
  796. this.set('requestInProgress', true);
  797. return App.ajax.send({
  798. name: 'admin.upgrade.upgradeItem.setState',
  799. sender: this,
  800. data: {
  801. upgradeId: item.get('request_id'),
  802. itemId: item.get('stage_id'),
  803. groupId: item.get('group_id'),
  804. status: status
  805. },
  806. callback: function() {
  807. this.sender.set('requestInProgress', false);
  808. }
  809. }).done(function () {
  810. item.set('status', status);
  811. });
  812. },
  813. currentVersionObserver: function () {
  814. var versionNumber = this.get('currentVersion.repository_version');
  815. var currentVersionObject = App.RepositoryVersion.find().findProperty('status', 'CURRENT');
  816. var versionName = currentVersionObject && currentVersionObject.get('stackVersionType');
  817. App.set('isStormMetricsSupported', versionName != 'HDP' || stringUtils.compareVersions(versionNumber, '2.2.2') > -1 || !versionNumber);
  818. }.observes('currentVersion.repository_version'),
  819. /**
  820. * get the installed repositories of HDP from server
  821. */
  822. loadRepositories: function () {
  823. if (App.router.get('clusterController.isLoaded')) {
  824. var nameVersionCombo = App.get('currentStackVersion');
  825. var stackName = nameVersionCombo.split('-')[0];
  826. var stackVersion = nameVersionCombo.split('-')[1];
  827. App.ajax.send({
  828. name: 'cluster.load_repositories',
  829. sender: this,
  830. data: {
  831. stackName: stackName,
  832. stackVersion: stackVersion
  833. },
  834. success: 'loadRepositoriesSuccessCallback',
  835. error: 'loadRepositoriesErrorCallback'
  836. });
  837. }
  838. }.observes('App.router.clusterController.isLoaded'),
  839. loadRepositoriesSuccessCallback: function (data) {
  840. var allRepos = [];
  841. data.items.forEach(function (os) {
  842. os.repositories.forEach(function (repository) {
  843. var osType = repository.Repositories.os_type;
  844. var repo = Em.Object.create({
  845. baseUrl: repository.Repositories.base_url,
  846. osType: osType,
  847. repoId: repository.Repositories.repo_id,
  848. repoName : repository.Repositories.repo_name,
  849. stackName : repository.Repositories.stack_name,
  850. stackVersion : repository.Repositories.stack_version,
  851. isFirst: false
  852. });
  853. var group = allRepos.findProperty('name', osType);
  854. if (!group) {
  855. group = {
  856. name: osType,
  857. repositories: []
  858. };
  859. repo.set('isFirst', true);
  860. allRepos.push(group);
  861. }
  862. group.repositories.push(repo);
  863. });
  864. }, this);
  865. allRepos.stackVersion = App.get('currentStackVersionNumber');
  866. this.set('allRepos', allRepos);
  867. },
  868. loadRepositoriesErrorCallback: function (request, ajaxOptions, error) {
  869. console.log('Error message is: ' + request.responseText);
  870. }
  871. });