stack_and_upgrade_controller.js 11 KB

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