stack_and_upgrade_controller.js 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366
  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 {Object|null}
  24. */
  25. serviceToInstall: null,
  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. * version that currently applied to server
  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'],
  55. /**
  56. * restore data from localStorage
  57. */
  58. init: function () {
  59. this.get('wizardStorageProperties').forEach(function (property) {
  60. if (this.getDBProperty(property)) {
  61. this.set(property, this.getDBProperty(property));
  62. }
  63. }, this);
  64. },
  65. /**
  66. * @type {Array}
  67. */
  68. services: function() {
  69. return App.StackService.find().map(function(s) {
  70. s.set('isInstalled', App.Service.find().someProperty('serviceName', s.get('serviceName')));
  71. return s;
  72. });
  73. }.property('App.router.clusterController.isLoaded'),
  74. /**
  75. * launch Add Service wizard
  76. * @param event
  77. */
  78. goToAddService: function (event) {
  79. this.set('serviceToInstall', event.context);
  80. App.get('router').transitionTo('main.serviceAdd');
  81. },
  82. /**
  83. * call to fetch cluster stack versions
  84. * @return {$.ajax}
  85. */
  86. loadVersionsInfo: function () {
  87. return App.ajax.send({
  88. name: 'admin.stack_versions.all',
  89. sender: this,
  90. data: {},
  91. success: 'loadVersionsInfoSuccessCallback'
  92. });
  93. },
  94. /**
  95. * parse stack versions and
  96. * set <code>currentVersion</code>
  97. * set <code>targetVersions</code>
  98. * @param data
  99. */
  100. loadVersionsInfoSuccessCallback: function (data) {
  101. var versions = this.parseVersionsData(data);
  102. var current = versions.findProperty('state', 'CURRENT');
  103. var targetVersions = versions.without(current).filter(function (version) {
  104. //Only higher versions that have already been installed to all the hosts are shown
  105. return (version.state === 'INSTALLED' &&
  106. stringUtils.compareVersions(version.repository_version, current.repository_version) === 1);
  107. });
  108. this.set('currentVersion', current);
  109. this.set('targetVersions', targetVersions);
  110. },
  111. /**
  112. * parse ClusterStackVersions data to form common structure
  113. * @param {object} data
  114. * @return {Array}
  115. */
  116. parseVersionsData: function (data) {
  117. return data.items.map(function (item) {
  118. item.ClusterStackVersions.repository_name = item.repository_versions[0].RepositoryVersions.display_name;
  119. item.ClusterStackVersions.repository_id = item.repository_versions[0].RepositoryVersions.id;
  120. item.ClusterStackVersions.repository_version = item.repository_versions[0].RepositoryVersions.repository_version;
  121. return item.ClusterStackVersions;
  122. });
  123. },
  124. /**
  125. * load upgrade tasks by upgrade id
  126. * @return {$.Deferred}
  127. * @param {boolean} onlyState
  128. */
  129. loadUpgradeData: function (onlyState) {
  130. var upgradeId = this.get('upgradeId');
  131. var deferred = $.Deferred();
  132. if (Em.isNone(upgradeId)) {
  133. deferred.resolve();
  134. console.log('Upgrade in INIT state');
  135. } else {
  136. App.ajax.send({
  137. name: (onlyState) ? 'admin.upgrade.state' : 'admin.upgrade.data',
  138. sender: this,
  139. data: {
  140. id: upgradeId
  141. },
  142. success: 'loadUpgradeDataSuccessCallback'
  143. }).then(deferred.resolve);
  144. }
  145. return deferred.promise();
  146. },
  147. /**
  148. * parse and push upgrade tasks to controller
  149. * @param data
  150. */
  151. loadUpgradeDataSuccessCallback: function (data) {
  152. App.set('upgradeState', data.Upgrade.request_status);
  153. if (data.upgrade_groups) {
  154. this.updateUpgradeData(data);
  155. }
  156. },
  157. /**
  158. * update data of Upgrade
  159. * @param {object} newData
  160. */
  161. updateUpgradeData: function (newData) {
  162. var oldData = this.get('upgradeData'),
  163. groupsMap = {},
  164. itemsMap = {},
  165. tasksMap = {};
  166. if (Em.isNone(oldData)) {
  167. this.initUpgradeData(newData);
  168. } else {
  169. //create entities maps
  170. newData.upgrade_groups.forEach(function (newGroup) {
  171. groupsMap[newGroup.UpgradeGroup.group_id] = newGroup.UpgradeGroup;
  172. newGroup.upgrade_items.forEach(function (item) {
  173. itemsMap[item.UpgradeItem.stage_id] = item.UpgradeItem;
  174. item.tasks.forEach(function (task) {
  175. tasksMap[task.Tasks.id] = task.Tasks;
  176. });
  177. })
  178. });
  179. //update existed entities with new data
  180. oldData.upgradeGroups.forEach(function (oldGroup) {
  181. oldGroup.set('status', groupsMap[oldGroup.get('group_id')].status);
  182. oldGroup.set('progress_percent', groupsMap[oldGroup.get('group_id')].progress_percent);
  183. oldGroup.upgradeItems.forEach(function (item) {
  184. item.set('status', itemsMap[item.get('stage_id')].status);
  185. item.set('progress_percent', itemsMap[item.get('stage_id')].progress_percent);
  186. item.tasks.forEach(function (task) {
  187. task.set('status', tasksMap[task.get('id')].status);
  188. });
  189. })
  190. });
  191. oldData.set('Upgrade', newData.Upgrade);
  192. }
  193. },
  194. /**
  195. * change structure of Upgrade
  196. * In order to maintain nested views in template object should have direct link to its properties, for example
  197. * item.UpgradeItem.<properties> -> item.<properties>
  198. * @param {object} newData
  199. */
  200. initUpgradeData: function (newData) {
  201. var upgradeGroups = [];
  202. //wrap all entities into App.upgradeEntity
  203. newData.upgrade_groups.forEach(function (newGroup) {
  204. var oldGroup = App.upgradeEntity.create({type: 'GROUP'}, newGroup.UpgradeGroup);
  205. var upgradeItems = [];
  206. newGroup.upgrade_items.forEach(function (item) {
  207. var oldItem = App.upgradeEntity.create({type: 'ITEM'}, item.UpgradeItem);
  208. var tasks = [];
  209. item.tasks.forEach(function (task) {
  210. tasks.pushObject(App.upgradeEntity.create({type: 'TASK'}, task.Tasks));
  211. });
  212. oldItem.set('tasks', tasks);
  213. upgradeItems.pushObject(oldItem);
  214. });
  215. oldGroup.set('upgradeItems', upgradeItems);
  216. upgradeGroups.pushObject(oldGroup);
  217. });
  218. this.set('upgradeData', Em.Object.create({
  219. upgradeGroups: upgradeGroups,
  220. Upgrade: newData.Upgrade
  221. }));
  222. },
  223. /**
  224. * make call to start downgrade process
  225. */
  226. downgrade: function () {
  227. //TODO start downgrade
  228. },
  229. /**
  230. * make call to start upgrade process and show popup with current progress
  231. * @param {object} version
  232. */
  233. upgrade: function (version) {
  234. App.ajax.send({
  235. name: 'admin.upgrade.start',
  236. sender: this,
  237. data: {
  238. version: version.value
  239. },
  240. success: 'upgradeSuccessCallback'
  241. });
  242. this.set('upgradeVersion', version.label);
  243. this.setDBProperty('upgradeVersion', version.label);
  244. },
  245. /**
  246. * success callback of <code>upgrade()</code>
  247. * @param {object} data
  248. */
  249. upgradeSuccessCallback: function (data) {
  250. this.set('upgradeId', data.resources[0].Upgrade.request_id);
  251. this.setDBProperty('upgradeId', data.resources[0].Upgrade.request_id);
  252. App.clusterStatus.setClusterStatus({
  253. clusterName: App.get('clusterName'),
  254. clusterState: 'DEFAULT',
  255. localdb: App.db.data
  256. });
  257. this.openUpgradeDialog();
  258. },
  259. /**
  260. * send request for pre upgrade check
  261. * @param version
  262. */
  263. runPreUpgradeCheck: function(version) {
  264. if (App.get('supports.preUpgradeCheck')) {
  265. App.ajax.send({
  266. name: "admin.rolling_upgrade.pre_upgrade_check",
  267. sender: this,
  268. data: version,
  269. success: "runPreUpgradeCheckSuccess"
  270. });
  271. } else {
  272. this.upgrade(version);
  273. }
  274. },
  275. /**
  276. * success callback of <code>runPreUpgradeCheckSuccess()</code>
  277. * if there are some fails - it shows popup else run upgrade
  278. * @param data {object}
  279. * @param opt {object}
  280. * @param params {object}
  281. * @returns {App.ModalPopup|void}
  282. */
  283. runPreUpgradeCheckSuccess: function(data, opt, params) {
  284. if (data.items.someProperty('UpgradeChecks.status',"FAIL")) {
  285. return App.ModalPopup.show({
  286. header: Em.I18n.t('admin.stackUpgrade.preupgradeCheck.header').format(params.label),
  287. primary: Em.I18n.t('common.dismiss'),
  288. secondary: false,
  289. bodyClass: Em.View.extend({
  290. templateName: require('templates/main/admin/stack_upgrade/pre_upgrade_check_dialog'),
  291. checks: data.items.filterProperty('UpgradeChecks.status',"FAIL")
  292. })
  293. })
  294. } else {
  295. this.upgrade(params);
  296. }
  297. },
  298. /**
  299. * make call to resume upgrade process and show popup with current progress
  300. */
  301. resumeUpgrade: function () {
  302. //TODO resume upgrade
  303. this.openUpgradeDialog();
  304. },
  305. /**
  306. * make call to finish upgrade process
  307. */
  308. finalize: function () {
  309. //TODO execute finalize
  310. this.finish();
  311. },
  312. /**
  313. * finish upgrade wizard
  314. * clean auxiliary data
  315. */
  316. finish: function () {
  317. this.set('upgradeId', null);
  318. this.setDBProperty('upgradeId', undefined);
  319. App.set('upgradeState', 'INIT');
  320. this.set('upgradeVersion', null);
  321. this.setDBProperty('upgradeVersion', undefined);
  322. App.clusterStatus.setClusterStatus({
  323. clusterName: App.get('clusterName'),
  324. clusterState: 'DEFAULT',
  325. localdb: App.db.data
  326. });
  327. },
  328. /**
  329. * show dialog with tasks of upgrade
  330. * @return {App.ModalPopup}
  331. */
  332. openUpgradeDialog: function () {
  333. App.router.transitionTo('admin.stackUpgrade');
  334. }
  335. });