stack_and_upgrade_controller.js 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677
  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. App.MainAdminStackAndUpgradeController = Em.Controller.extend(App.LocalStorage, {
  20. name: 'mainAdminStackAndUpgradeController',
  21. /**
  22. * @type {boolean}
  23. */
  24. isLoaded: false,
  25. /**
  26. * @type {object}
  27. * @default null
  28. */
  29. upgradeData: null,
  30. /**
  31. * @type {number}
  32. * @default null
  33. */
  34. upgradeId: null,
  35. /**
  36. * @type {string}
  37. * @default null
  38. */
  39. upgradeVersion: null,
  40. /**
  41. * @type {boolean}
  42. * @default false
  43. */
  44. isDowngrade: false,
  45. /**
  46. * version that currently applied to server
  47. * should be plain object, because stored to localStorage
  48. * @type {object|null}
  49. */
  50. currentVersion: null,
  51. /**
  52. * versions to which cluster could be upgraded
  53. * @type {Array}
  54. */
  55. targetVersions: [],
  56. /**
  57. * properties that stored to localStorage to resume wizard progress
  58. */
  59. wizardStorageProperties: ['upgradeId', 'upgradeVersion', 'currentVersion', 'isDowngrade'],
  60. /**
  61. * path to the mock json
  62. * @type {String}
  63. */
  64. mockRepoUrl: '/data/stack_versions/repo_versions_all.json',
  65. /**
  66. * api to get RepoVersions
  67. * @type {String}
  68. */
  69. realRepoUrl: function () {
  70. return App.get('apiPrefix') + App.get('stackVersionURL') +
  71. '/repository_versions?fields=*,operating_systems/*,operating_systems/repositories/*';
  72. }.property('App.stackVersionURL'),
  73. /**
  74. * path to the mock json
  75. * @type {String}
  76. */
  77. mockStackUrl: '/data/stack_versions/stack_version_all.json',
  78. /**
  79. * api to get ClusterStackVersions with repository_versions (use to init data load)
  80. * @type {String}
  81. */
  82. realStackUrl: function () {
  83. return App.get('apiPrefix') + '/clusters/' + App.get('clusterName') +
  84. '/stack_versions?fields=*,repository_versions/*,repository_versions/operating_systems/repositories/*';
  85. }.property('App.clusterName'),
  86. /**
  87. * api to get ClusterStackVersions without repository_versions (use to update data)
  88. * @type {String}
  89. */
  90. realUpdateUrl: function () {
  91. return App.get('apiPrefix') + '/clusters/' + App.get('clusterName') + '/stack_versions?fields=ClusterStackVersions/*';
  92. }.property('App.clusterName'),
  93. init: function () {
  94. this.initDBProperties();
  95. },
  96. /**
  97. * restore data from localStorage
  98. */
  99. initDBProperties: function () {
  100. this.get('wizardStorageProperties').forEach(function (property) {
  101. if (this.getDBProperty(property)) {
  102. this.set(property, this.getDBProperty(property));
  103. }
  104. }, this);
  105. },
  106. /**
  107. * load all data:
  108. * - upgrade data
  109. * - stack versions
  110. * - repo versions
  111. */
  112. load: function () {
  113. var dfd = $.Deferred();
  114. var self = this;
  115. this.loadUpgradeData(true).done(function() {
  116. self.loadStackVersionsToModel(true).done(function () {
  117. self.loadRepoVersionsToModel().done(function() {
  118. var currentVersion = App.StackVersion.find().findProperty('state', 'CURRENT');
  119. if (currentVersion) {
  120. self.set('currentVersion', {
  121. repository_version: currentVersion.get('repositoryVersion.repositoryVersion'),
  122. repository_name: currentVersion.get('repositoryVersion.displayName')
  123. });
  124. }
  125. dfd.resolve();
  126. });
  127. });
  128. });
  129. return dfd.promise();
  130. },
  131. /**
  132. * load upgrade tasks by upgrade id
  133. * @return {$.Deferred}
  134. * @param {boolean} onlyState
  135. */
  136. loadUpgradeData: function (onlyState) {
  137. var upgradeId = this.get('upgradeId');
  138. var deferred = $.Deferred();
  139. if (Em.isNone(upgradeId)) {
  140. deferred.resolve();
  141. console.log('Upgrade in INIT state');
  142. } else {
  143. App.ajax.send({
  144. name: (onlyState) ? 'admin.upgrade.state' : 'admin.upgrade.data',
  145. sender: this,
  146. data: {
  147. id: upgradeId
  148. },
  149. success: 'loadUpgradeDataSuccessCallback'
  150. }).then(deferred.resolve);
  151. }
  152. return deferred.promise();
  153. },
  154. /**
  155. * parse and push upgrade tasks to controller
  156. * @param data
  157. */
  158. loadUpgradeDataSuccessCallback: function (data) {
  159. App.set('upgradeState', data.Upgrade.request_status);
  160. this.setDBProperty('upgradeState', data.Upgrade.request_status);
  161. if (data.upgrade_groups) {
  162. this.updateUpgradeData(data);
  163. }
  164. },
  165. /**
  166. * update data of Upgrade
  167. * @param {object} newData
  168. */
  169. updateUpgradeData: function (newData) {
  170. var oldData = this.get('upgradeData'),
  171. groupsMap = {},
  172. itemsMap = {},
  173. tasksMap = {};
  174. if (Em.isNone(oldData)) {
  175. this.initUpgradeData(newData);
  176. } else {
  177. //create entities maps
  178. newData.upgrade_groups.forEach(function (newGroup) {
  179. groupsMap[newGroup.UpgradeGroup.group_id] = newGroup.UpgradeGroup;
  180. newGroup.upgrade_items.forEach(function (item) {
  181. itemsMap[item.UpgradeItem.stage_id] = item.UpgradeItem;
  182. item.tasks.forEach(function (task) {
  183. tasksMap[task.Tasks.id] = task.Tasks;
  184. });
  185. })
  186. });
  187. //update existed entities with new data
  188. oldData.upgradeGroups.forEach(function (oldGroup) {
  189. oldGroup.set('status', groupsMap[oldGroup.get('group_id')].status);
  190. oldGroup.set('progress_percent', groupsMap[oldGroup.get('group_id')].progress_percent);
  191. oldGroup.upgradeItems.forEach(function (item) {
  192. item.set('status', itemsMap[item.get('stage_id')].status);
  193. item.set('progress_percent', itemsMap[item.get('stage_id')].progress_percent);
  194. item.tasks.forEach(function (task) {
  195. task.set('status', tasksMap[task.get('id')].status);
  196. });
  197. })
  198. });
  199. oldData.set('Upgrade', newData.Upgrade);
  200. }
  201. },
  202. /**
  203. * change structure of Upgrade
  204. * In order to maintain nested views in template object should have direct link to its properties, for example
  205. * item.UpgradeItem.<properties> -> item.<properties>
  206. * @param {object} newData
  207. */
  208. initUpgradeData: function (newData) {
  209. var upgradeGroups = [];
  210. //wrap all entities into App.upgradeEntity
  211. newData.upgrade_groups.forEach(function (newGroup) {
  212. var oldGroup = App.upgradeEntity.create({type: 'GROUP'}, newGroup.UpgradeGroup);
  213. var upgradeItems = [];
  214. newGroup.upgrade_items.forEach(function (item) {
  215. var oldItem = App.upgradeEntity.create({type: 'ITEM'}, item.UpgradeItem);
  216. var tasks = [];
  217. item.tasks.forEach(function (task) {
  218. tasks.pushObject(App.upgradeEntity.create({type: 'TASK'}, task.Tasks));
  219. });
  220. oldItem.set('tasks', tasks);
  221. upgradeItems.pushObject(oldItem);
  222. });
  223. upgradeItems.reverse();
  224. oldGroup.set('upgradeItems', upgradeItems);
  225. upgradeGroups.pushObject(oldGroup);
  226. });
  227. upgradeGroups.reverse();
  228. this.set('upgradeData', Em.Object.create({
  229. upgradeGroups: upgradeGroups,
  230. Upgrade: newData.Upgrade
  231. }));
  232. },
  233. /**
  234. * downgrade confirmation popup
  235. * @param {object} event
  236. */
  237. confirmDowngrade: function (event) {
  238. var self = this;
  239. var currentVersion = this.get('currentVersion');
  240. return App.showConfirmationPopup(
  241. function() {
  242. self.downgrade.call(self, currentVersion, event);
  243. },
  244. Em.I18n.t('admin.stackUpgrade.downgrade.body').format(currentVersion.repository_name),
  245. null,
  246. Em.I18n.t('admin.stackUpgrade.dialog.downgrade.header').format(currentVersion.repository_name),
  247. Em.I18n.t('admin.stackUpgrade.downgrade.proceed')
  248. );
  249. },
  250. /**
  251. * make call to start downgrade process
  252. * @param {object} currentVersion
  253. * @param {object} event
  254. */
  255. downgrade: function (currentVersion, event) {
  256. this.abortUpgrade();
  257. App.ajax.send({
  258. name: 'admin.downgrade.start',
  259. sender: this,
  260. data: {
  261. value: currentVersion.repository_version,
  262. label: currentVersion.repository_name,
  263. isDowngrade: true
  264. },
  265. success: 'upgradeSuccessCallback'
  266. });
  267. },
  268. /**
  269. * abort upgrade (in order to start Downgrade)
  270. */
  271. abortUpgrade: function () {
  272. return App.ajax.send({
  273. name: 'admin.upgrade.abort',
  274. sender: this,
  275. data: {
  276. upgradeId: this.get('upgradeId')
  277. }
  278. });
  279. },
  280. /**
  281. * make call to start upgrade process and show popup with current progress
  282. * @param {object} version
  283. */
  284. upgrade: function (version) {
  285. App.ajax.send({
  286. name: 'admin.upgrade.start',
  287. sender: this,
  288. data: version,
  289. success: 'upgradeSuccessCallback'
  290. });
  291. this.setDBProperty('currentVersion', this.get('currentVersion'));
  292. },
  293. /**
  294. * success callback of <code>upgrade()</code>
  295. * @param {object} data
  296. */
  297. upgradeSuccessCallback: function (data, opt, params) {
  298. this.set('upgradeData', null);
  299. this.set('upgradeId', data.resources[0].Upgrade.request_id);
  300. this.set('upgradeVersion', params.label);
  301. this.set('isDowngrade', !!params.isDowngrade);
  302. this.setDBProperty('upgradeVersion', params.label);
  303. this.setDBProperty('upgradeId', data.resources[0].Upgrade.request_id);
  304. this.setDBProperty('upgradeState', 'PENDING');
  305. this.setDBProperty('isDowngrade', !!params.isDowngrade);
  306. App.set('upgradeState', 'PENDING');
  307. App.clusterStatus.setClusterStatus({
  308. wizardControllerName: this.get('name'),
  309. localdb: App.db.data
  310. });
  311. this.load();
  312. this.openUpgradeDialog();
  313. },
  314. /**
  315. * upgrade confirmation popup
  316. * @param {object} version
  317. * @return App.ModalPopup
  318. */
  319. confirmUpgrade: function (version) {
  320. var self = this;
  321. return App.showConfirmationPopup(
  322. function () {
  323. self.runPreUpgradeCheck.call(self, version);
  324. },
  325. Em.I18n.t('admin.stackUpgrade.upgrade.confirm.body').format(version.get('displayName')),
  326. null,
  327. Em.I18n.t('admin.stackUpgrade.dialog.header').format(version.get('displayName'))
  328. );
  329. },
  330. /**
  331. * send request for pre upgrade check
  332. * @param version
  333. */
  334. runPreUpgradeCheck: function(version) {
  335. var params = {
  336. value: version.get('repositoryVersion'),
  337. label: version.get('displayName')
  338. };
  339. if (App.get('supports.preUpgradeCheck')) {
  340. App.ajax.send({
  341. name: "admin.rolling_upgrade.pre_upgrade_check",
  342. sender: this,
  343. data: params,
  344. success: "runPreUpgradeCheckSuccess"
  345. });
  346. } else {
  347. this.upgrade(params);
  348. }
  349. },
  350. /**
  351. * success callback of <code>runPreUpgradeCheckSuccess()</code>
  352. * if there are some fails - it shows popup else run upgrade
  353. * @param data {object}
  354. * @param opt {object}
  355. * @param params {object}
  356. * @returns {App.ModalPopup|undefined}
  357. */
  358. runPreUpgradeCheckSuccess: function (data, opt, params) {
  359. if (data.items.someProperty('UpgradeChecks.status', "FAIL")) {
  360. var header = Em.I18n.t('popup.clusterCheck.Upgrade.header').format(params.label);
  361. var title = Em.I18n.t('popup.clusterCheck.Upgrade.title');
  362. var alert = Em.I18n.t('popup.clusterCheck.Upgrade.alert');
  363. App.showClusterCheckPopup(data, header, title, alert);
  364. } else {
  365. this.upgrade(params);
  366. }
  367. },
  368. /**
  369. * confirmation popup before install repository version
  370. */
  371. installRepoVersionConfirmation: function (repo) {
  372. var self = this;
  373. return App.showConfirmationPopup(function () {
  374. self.installRepoVersion(repo);
  375. },
  376. Em.I18n.t('admin.stackVersions.version.install.confirm').format(repo.get('displayName'))
  377. );
  378. },
  379. /**
  380. * sends request to install repoVersion to the cluster
  381. * and create clusterStackVersion resourse
  382. * @param {Em.Object} repo
  383. * @return {$.ajax}
  384. * @method installRepoVersion
  385. */
  386. installRepoVersion: function (repo) {
  387. var data = {
  388. ClusterStackVersions: {
  389. stack: repo.get('stackVersionType'),
  390. version: repo.get('stackVersionNumber'),
  391. repository_version: repo.get('repositoryVersion')
  392. },
  393. id: repo.get('id')
  394. };
  395. return App.ajax.send({
  396. name: 'admin.stack_version.install.repo_version',
  397. sender: this,
  398. data: data,
  399. success: 'installRepoVersionSuccess'
  400. });
  401. },
  402. /**
  403. * transform repo data into json for
  404. * saving changes to repository version
  405. * @param {Em.Object} repo
  406. * @returns {{operating_systems: Array}}
  407. */
  408. prepareRepoForSaving: function(repo) {
  409. var repoVersion = { "operating_systems": [] };
  410. repo.get('operatingSystems').forEach(function (os, k) {
  411. repoVersion.operating_systems.push({
  412. "OperatingSystems": {
  413. "os_type": os.get("osType")
  414. },
  415. "repositories": []
  416. });
  417. os.get('repositories').forEach(function (repository) {
  418. repoVersion.operating_systems[k].repositories.push({
  419. "Repositories": {
  420. "base_url": repository.get('baseUrl'),
  421. "repo_id": repository.get('repoId'),
  422. "repo_name": repository.get('repoName')
  423. }
  424. });
  425. });
  426. });
  427. return repoVersion;
  428. },
  429. /**
  430. * perform validation if <code>skip<code> is false and run save if
  431. * validation successfull or run save without validation is <code>skip<code> is true
  432. * @param {Em.Object} repo
  433. * @param {boolean} skip
  434. * @returns {$.Deferred}
  435. */
  436. saveRepoOS: function (repo, skip) {
  437. var self = this;
  438. var deferred = $.Deferred();
  439. this.validateRepoVersions(repo, skip).done(function(data) {
  440. if (data.length > 0) {
  441. deferred.resolve(data);
  442. } else {
  443. var repoVersion = self.prepareRepoForSaving(repo);
  444. App.ajax.send({
  445. name: 'admin.stack_versions.edit.repo',
  446. sender: this,
  447. data: {
  448. stackName: App.get('currentStackName'),
  449. stackVersion: App.get('currentStackVersionNumber'),
  450. repoVersionId: repo.get('repoVersionId'),
  451. repoVersion: repoVersion
  452. }
  453. }).success(function() {
  454. deferred.resolve([]);
  455. });
  456. }
  457. });
  458. return deferred.promise();
  459. },
  460. /**
  461. * send request for validation for each repository
  462. * @param {Em.Object} repo
  463. * @param {boolean} skip
  464. * @returns {*}
  465. */
  466. validateRepoVersions: function(repo, skip) {
  467. var deferred = $.Deferred(),
  468. totalCalls = 0,
  469. invalidUrls = [];
  470. if (skip) {
  471. deferred.resolve(invalidUrls);
  472. } else {
  473. repo.get('operatingSystems').forEach(function (os) {
  474. if (os.get('isSelected')) {
  475. os.get('repositories').forEach(function (repo) {
  476. totalCalls++;
  477. App.ajax.send({
  478. name: 'admin.stack_versions.validate.repo',
  479. sender: this,
  480. data: {
  481. repo: repo,
  482. repoId: repo.get('repoId'),
  483. baseUrl: repo.get('baseUrl'),
  484. osType: os.get('osType'),
  485. stackName: App.get('currentStackName'),
  486. stackVersion: App.get('currentStackVersionNumber')
  487. }
  488. })
  489. .success(function () {
  490. totalCalls--;
  491. if (totalCalls === 0) deferred.resolve(invalidUrls);
  492. })
  493. .error(function () {
  494. repo.set('hasError', true);
  495. invalidUrls.push(repo);
  496. totalCalls--;
  497. if (totalCalls === 0) deferred.resolve(invalidUrls);
  498. });
  499. });
  500. } else {
  501. return deferred.resolve(invalidUrls);
  502. }
  503. });
  504. }
  505. return deferred.promise();
  506. },
  507. /**
  508. * success callback for <code>installRepoVersion()<code>
  509. * saves request id to the db
  510. * @param data
  511. * @param opt
  512. * @param params
  513. * @method installStackVersionSuccess
  514. */
  515. installRepoVersionSuccess: function (data, opt, params) {
  516. App.db.set('repoVersionInstall', 'id', [data.Requests.id]);
  517. App.clusterStatus.setClusterStatus({
  518. wizardControllerName: this.get('name'),
  519. localdb: App.db.data
  520. });
  521. App.RepositoryVersion.find(params.id).set('defaultStatus', 'INSTALLING');
  522. },
  523. /**
  524. * opens a popup with installations state per host
  525. * @param {Em.Object} version
  526. * @method showProgressPopup
  527. */
  528. showProgressPopup: function(version) {
  529. var popupTitle = Em.I18n.t('admin.stackVersions.details.install.hosts.popup.title').format(version.get('displayName'));
  530. var requestIds = App.get('testMode') ? [1] : App.db.get('repoVersionInstall', 'id');
  531. var hostProgressPopupController = App.router.get('highAvailabilityProgressPopupController');
  532. hostProgressPopupController.initPopup(popupTitle, requestIds, this);
  533. },
  534. /**
  535. * reset upgradeState to INIT when upgrade is COMPLETED
  536. * and clean auxiliary data
  537. */
  538. finish: function () {
  539. if (App.get('upgradeState') === 'COMPLETED') {
  540. this.setDBProperty('upgradeId', undefined);
  541. this.setDBProperty('upgradeState', 'INIT');
  542. this.setDBProperty('upgradeVersion', undefined);
  543. this.setDBProperty('currentVersion', undefined);
  544. this.setDBProperty('isDowngrade', undefined);
  545. App.clusterStatus.setClusterStatus({
  546. localdb: App.db.data
  547. });
  548. App.set('upgradeState', 'INIT');
  549. }
  550. }.observes('App.upgradeState'),
  551. /**
  552. * show dialog with tasks of upgrade
  553. * @return {App.ModalPopup}
  554. */
  555. openUpgradeDialog: function () {
  556. App.router.transitionTo('admin.stackUpgrade');
  557. },
  558. /**
  559. * returns url to get data for repoVersion or clusterStackVersion
  560. * @param {Boolean} stack true if load clusterStackVersion
  561. * @param {Boolean} fullLoad true if load all data
  562. * @returns {String}
  563. * @method getUrl
  564. */
  565. getUrl: function(stack, fullLoad) {
  566. if (App.get('testMode')) {
  567. return stack ? this.get('mockStackUrl') : this.get('mockRepoUrl')
  568. } else {
  569. if (fullLoad) {
  570. return stack ? this.get('realStackUrl') : this.get('realRepoUrl');
  571. } else {
  572. return this.get('realUpdateUrl');
  573. }
  574. }
  575. },
  576. /**
  577. * get stack versions from server and push it to model
  578. * @return {*}
  579. * @method loadStackVersionsToModel
  580. */
  581. loadStackVersionsToModel: function (fullLoad) {
  582. var dfd = $.Deferred();
  583. App.HttpClient.get(this.getUrl(true, fullLoad), App.stackVersionMapper, {
  584. complete: function () {
  585. dfd.resolve();
  586. }
  587. });
  588. return dfd.promise();
  589. },
  590. /**
  591. * get repo versions from server and push it to model
  592. * @return {*}
  593. * @params {Boolean} isUpdate - if true loads part of data that need to be updated
  594. * @method loadRepoVersionsToModel()
  595. */
  596. loadRepoVersionsToModel: function () {
  597. var dfd = $.Deferred();
  598. App.HttpClient.get(this.getUrl(false, true), App.repoVersionMapper, {
  599. complete: function () {
  600. dfd.resolve();
  601. }
  602. });
  603. return dfd.promise();
  604. },
  605. /**
  606. * set status to Upgrade item
  607. * @param item
  608. * @param status
  609. */
  610. setUpgradeItemStatus: function(item, status) {
  611. return App.ajax.send({
  612. name: 'admin.upgrade.upgradeItem.setState',
  613. sender: this,
  614. data: {
  615. upgradeId: item.get('request_id'),
  616. itemId: item.get('stage_id'),
  617. groupId: item.get('group_id'),
  618. status: status
  619. }
  620. }).done(function () {
  621. item.set('status', status);
  622. });
  623. }
  624. });