stack_and_upgrade_controller.js 31 KB

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