stack_and_upgrade_controller.js 19 KB

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