stack_and_upgrade_controller.js 59 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798
  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 {string}
  43. * @default null
  44. */
  45. upgradeType: null,
  46. /**
  47. * @type {boolean}
  48. * @default true
  49. */
  50. downgradeAllowed: true,
  51. /**
  52. * @type {string}
  53. * @default null
  54. */
  55. upgradeTypeDisplayName: null,
  56. /**
  57. * @type {object}
  58. * @default null
  59. */
  60. failuresTolerance: null,
  61. /**
  62. * @type {boolean}
  63. * @default false
  64. */
  65. isDowngrade: false,
  66. /**
  67. * flag which indicate that upgrade suspended
  68. * @type {boolean}
  69. * @default false
  70. */
  71. isSuspended: false,
  72. /**
  73. * version that currently applied to server
  74. * should be plain object, because stored to localStorage
  75. * @type {object|null}
  76. */
  77. currentVersion: null,
  78. /**
  79. * versions to which cluster could be upgraded
  80. * @type {Array}
  81. */
  82. targetVersions: [],
  83. /**
  84. * @type {object}
  85. * @default null
  86. */
  87. slaveComponentStructuredInfo: null,
  88. /**
  89. * @type {Array}
  90. */
  91. serviceCheckFailuresServicenames: [],
  92. /**
  93. * methods through which cluster could be upgraded, "allowed" indicated if the method is allowed
  94. * by stack upgrade path
  95. * @type {Array}
  96. */
  97. upgradeMethods: [
  98. Em.Object.create({
  99. displayName: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.RU.title'),
  100. type: 'ROLLING',
  101. icon: "icon-dashboard",
  102. description: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.RU.description'),
  103. selected: false,
  104. allowed: true,
  105. isCheckComplete: false,
  106. isCheckRequestInProgress: false,
  107. precheckResultsMessage: '',
  108. precheckResultsTitle: '',
  109. action: ''
  110. }),
  111. Em.Object.create({
  112. displayName: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.EU.title'),
  113. type: 'NON_ROLLING',
  114. icon: "icon-bolt",
  115. description: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.EU.description'),
  116. selected: false,
  117. allowed: true,
  118. isCheckComplete: false,
  119. isCheckRequestInProgress: false,
  120. precheckResultsMessage: '',
  121. precheckResultsTitle: '',
  122. action: ''
  123. })
  124. ],
  125. runningCheckRequests: [],
  126. /**
  127. * @type {boolean} true if some request that should disable actions is in progress
  128. */
  129. requestInProgress: false,
  130. /**
  131. * @type {number} repo id, request for which is currently in progress
  132. */
  133. requestInProgressRepoId: null,
  134. /**
  135. * @type {boolean} true while no updated upgrade info is loaded after retry
  136. */
  137. isRetryPending: false,
  138. /**
  139. * properties that stored to localStorage to resume wizard progress
  140. */
  141. wizardStorageProperties: [
  142. 'upgradeId',
  143. 'upgradeVersion',
  144. 'currentVersion',
  145. 'upgradeTypeDisplayName',
  146. 'upgradeType',
  147. 'failuresTolerance',
  148. 'isDowngrade',
  149. 'downgradeAllowed',
  150. 'isSuspended'
  151. ],
  152. /**
  153. * mutable properties of Upgrade Task
  154. * @type {Array}
  155. */
  156. taskDetailsProperties: ['status', 'stdout', 'stderr', 'error_log', 'host_name', 'output_log'],
  157. /**
  158. * Context for Finalize item
  159. * @type {string}
  160. */
  161. finalizeContext: 'Confirm Finalize',
  162. /**
  163. * Context for Slave component failures manual item
  164. * @type {string}
  165. */
  166. slaveFailuresContext: "Check Component Versions",
  167. /**
  168. * Context for Service check (may include slave component) failures manual item
  169. * @type {string}
  170. */
  171. serviceCheckFailuresContext: "Verifying Skipped Failures",
  172. /**
  173. * Check if current item is Finalize
  174. * @type {boolean}
  175. */
  176. isFinalizeItem: false,
  177. isLoadUpgradeDataPending: false,
  178. /**
  179. * path to the mock json
  180. * @type {String}
  181. */
  182. mockRepoUrl: '/data/stack_versions/repo_versions_all.json',
  183. /**
  184. * api to get RepoVersions
  185. * @type {String}
  186. */
  187. realRepoUrl: function () {
  188. return App.get('apiPrefix') + App.get('stackVersionURL') +
  189. '/compatible_repository_versions?fields=*,operating_systems/*,operating_systems/repositories/*';
  190. }.property('App.stackVersionURL'),
  191. /**
  192. * path to the mock json
  193. * @type {String}
  194. */
  195. mockStackUrl: '/data/stack_versions/stack_version_all.json',
  196. /**
  197. * api to get ClusterStackVersions with repository_versions (use to init data load)
  198. * @type {String}
  199. */
  200. realStackUrl: function () {
  201. return App.get('apiPrefix') + '/clusters/' + App.get('clusterName') +
  202. '/stack_versions?fields=*,repository_versions/*,repository_versions/operating_systems/repositories/*';
  203. }.property('App.clusterName'),
  204. /**
  205. * api to get ClusterStackVersions without repository_versions (use to update data)
  206. * @type {String}
  207. */
  208. realUpdateUrl: function () {
  209. return App.get('apiPrefix') + '/clusters/' + App.get('clusterName') + '/stack_versions?fields=ClusterStackVersions/*';
  210. }.property('App.clusterName'),
  211. /**
  212. * Determines if list of services with checks that failed and were skipped by user during the upgrade is loaded
  213. * @type {boolean}
  214. */
  215. areSkippedServiceChecksLoaded: false,
  216. /**
  217. * List of services with checks that failed and were skipped by user during the upgrade
  218. * @type {array}
  219. */
  220. skippedServiceChecks: [],
  221. /**
  222. * status of tasks/items/groups which should be grayed out and disabled
  223. * @type {Array}
  224. */
  225. nonActiveStates: ['PENDING', 'ABORTED'],
  226. /**
  227. * status of Upgrade request
  228. * @type {string}
  229. */
  230. requestStatus: function () {
  231. if (this.get('upgradeData.Upgrade') && App.get('upgradeSuspended')) {
  232. return 'SUSPENDED';
  233. } else if (this.get('upgradeData.Upgrade')){
  234. return this.get('upgradeData.Upgrade.request_status');
  235. } else {
  236. return '';
  237. }
  238. }.property('upgradeData.Upgrade.request_status', 'App.upgradeSuspended'),
  239. init: function () {
  240. this.initDBProperties();
  241. },
  242. /**
  243. * restore data from localStorage
  244. */
  245. initDBProperties: function () {
  246. var props = this.getDBProperties(this.get('wizardStorageProperties'));
  247. Em.keys(props).forEach(function (k) {
  248. if (!Em.isNone(props[k])) {
  249. this.set(k, props[k]);
  250. }
  251. }, this);
  252. },
  253. /**
  254. * load all data:
  255. * - upgrade data
  256. * - stack versions
  257. * - repo versions
  258. */
  259. load: function () {
  260. var dfd = $.Deferred();
  261. var self = this;
  262. this.loadUpgradeData(true).done(function() {
  263. self.loadStackVersionsToModel(true).done(function () {
  264. self.loadRepoVersionsToModel().done(function() {
  265. var currentVersion = App.StackVersion.find().findProperty('state', 'CURRENT');
  266. if (currentVersion) {
  267. self.set('currentVersion', {
  268. repository_version: currentVersion.get('repositoryVersion.repositoryVersion'),
  269. repository_name: currentVersion.get('repositoryVersion.displayName')
  270. });
  271. }
  272. dfd.resolve();
  273. });
  274. });
  275. });
  276. return dfd.promise();
  277. },
  278. /**
  279. * load upgrade tasks by upgrade id
  280. * @return {$.Deferred}
  281. * @param {boolean} onlyState
  282. */
  283. loadUpgradeData: function (onlyState) {
  284. var upgradeId = this.get('upgradeId'),
  285. deferred = $.Deferred(),
  286. self = this;
  287. if (Em.isNone(upgradeId)) {
  288. deferred.resolve();
  289. } else {
  290. this.set('isLoadUpgradeDataPending', true);
  291. App.ajax.send({
  292. name: (onlyState) ? 'admin.upgrade.state' : 'admin.upgrade.data',
  293. sender: this,
  294. data: {
  295. id: upgradeId
  296. },
  297. success: 'loadUpgradeDataSuccessCallback'
  298. }).then(deferred.resolve).complete(function () {
  299. self.set('isLoadUpgradeDataPending', false);
  300. });
  301. }
  302. return deferred.promise();
  303. },
  304. /**
  305. * parse and push upgrade tasks to controller
  306. * @param data
  307. */
  308. loadUpgradeDataSuccessCallback: function (data) {
  309. if (Em.isNone(data)) return;
  310. App.set('upgradeState', data.Upgrade.request_status);
  311. this.setDBProperty('upgradeState', data.Upgrade.request_status);
  312. this.set('isSuspended', data.Upgrade.suspended);
  313. this.setDBProperty('isSuspended', data.Upgrade.suspended);
  314. if (data.upgrade_groups) {
  315. this.updateUpgradeData(data);
  316. }
  317. if (this.get('isRetryPending') && data.Upgrade.request_status !== 'ABORTED') {
  318. this.setProperties({
  319. requestInProgress: false,
  320. isRetryPending: false
  321. });
  322. }
  323. },
  324. /**
  325. * update data of Upgrade
  326. * @param {object} newData
  327. */
  328. updateUpgradeData: function (newData) {
  329. var oldData = this.get('upgradeData'),
  330. nonActiveStates = this.get('nonActiveStates'),
  331. groupsMap = {},
  332. itemsMap = {};
  333. if (Em.isNone(oldData) || (newData.upgrade_groups.length !== oldData.upgradeGroups.length)) {
  334. this.initUpgradeData(newData);
  335. } else {
  336. //create entities maps
  337. newData.upgrade_groups.forEach(function (newGroup) {
  338. groupsMap[newGroup.UpgradeGroup.group_id] = newGroup.UpgradeGroup;
  339. newGroup.upgrade_items.forEach(function (item) {
  340. itemsMap[item.UpgradeItem.stage_id] = item.UpgradeItem;
  341. })
  342. });
  343. //update existed entities with new data
  344. oldData.upgradeGroups.forEach(function (oldGroup) {
  345. oldGroup.set('status', groupsMap[oldGroup.get('group_id')].status);
  346. oldGroup.set('display_status', groupsMap[oldGroup.get('group_id')].display_status);
  347. oldGroup.set('progress_percent', groupsMap[oldGroup.get('group_id')].progress_percent);
  348. oldGroup.set('completed_task_count', groupsMap[oldGroup.get('group_id')].completed_task_count);
  349. oldGroup.upgradeItems.forEach(function (item) {
  350. item.set('status', itemsMap[item.get('stage_id')].status);
  351. item.set('display_status', itemsMap[item.get('stage_id')].display_status);
  352. item.set('progress_percent', itemsMap[item.get('stage_id')].progress_percent);
  353. });
  354. var hasExpandableItems = oldGroup.upgradeItems.some(function (item) {
  355. return !nonActiveStates.contains(item.get('status'));
  356. });
  357. oldGroup.set('hasExpandableItems', hasExpandableItems);
  358. });
  359. oldData.set('Upgrade', newData.Upgrade);
  360. }
  361. },
  362. /**
  363. * change structure of Upgrade
  364. * In order to maintain nested views in template object should have direct link to its properties, for example
  365. * item.UpgradeItem.<properties> -> item.<properties>
  366. * @param {object} newData
  367. */
  368. initUpgradeData: function (newData) {
  369. var upgradeGroups = [],
  370. nonActiveStates = this.get('nonActiveStates');
  371. //wrap all entities into App.upgradeEntity
  372. newData.upgrade_groups.forEach(function (newGroup) {
  373. var hasExpandableItems = newGroup.upgrade_items.some(function (item) {
  374. return !nonActiveStates.contains(item.UpgradeItem.status);
  375. }),
  376. oldGroup = App.upgradeEntity.create({type: 'GROUP', hasExpandableItems: hasExpandableItems}, newGroup.UpgradeGroup),
  377. upgradeItems = [];
  378. newGroup.upgrade_items.forEach(function (item) {
  379. var oldItem = App.upgradeEntity.create({type: 'ITEM'}, item.UpgradeItem);
  380. var status = oldItem.get('status');
  381. if ('HOLDING' == status){
  382. //manualItem
  383. var text = oldItem.get('text');
  384. try {
  385. var messageArray = $.parseJSON(text)
  386. var messages = [];
  387. for(var i = 0; i < messageArray.length; i ++){
  388. var aMessageObj = messageArray[i];
  389. messages.push(aMessageObj.message);
  390. }
  391. oldItem.set('messages', messages)
  392. oldItem.set('text', messages.join(' '))
  393. } catch (err){}
  394. }
  395. oldItem.set('tasks', []);
  396. upgradeItems.pushObject(oldItem);
  397. });
  398. upgradeItems.reverse();
  399. oldGroup.set('upgradeItems', upgradeItems);
  400. upgradeGroups.pushObject(oldGroup);
  401. });
  402. upgradeGroups.reverse();
  403. this.set('upgradeData', Em.Object.create({
  404. upgradeGroups: upgradeGroups,
  405. Upgrade: newData.Upgrade
  406. }));
  407. this.set('downgradeAllowed', newData.Upgrade.downgrade_allowed);
  408. this.setDBProperty('downgradeAllowed', newData.Upgrade.downgrade_allowed);
  409. },
  410. /**
  411. * request Upgrade Item and its tasks from server
  412. * @param {Em.Object} item
  413. * @param {Function} customCallback
  414. * @return {$.ajax}
  415. */
  416. getUpgradeItem: function (item, customCallback) {
  417. return App.ajax.send({
  418. name: 'admin.upgrade.upgrade_item',
  419. sender: this,
  420. data: {
  421. upgradeId: item.get('request_id'),
  422. groupId: item.get('group_id'),
  423. stageId: item.get('stage_id')
  424. },
  425. success: customCallback || 'getUpgradeItemSuccessCallback'
  426. });
  427. },
  428. /**
  429. * success callback of <code>getTasks</code>
  430. * @param {object} data
  431. */
  432. getUpgradeItemSuccessCallback: function (data) {
  433. this.get('upgradeData.upgradeGroups').forEach(function (group) {
  434. if (group.get('group_id') === data.UpgradeItem.group_id) {
  435. group.get('upgradeItems').forEach(function (item) {
  436. if (item.get('stage_id') === data.UpgradeItem.stage_id) {
  437. if (item.get('tasks.length')) {
  438. item.set('isTasksLoaded', true);
  439. data.tasks.forEach(function (task) {
  440. var currentTask = item.get('tasks').findProperty('id', task.Tasks.id);
  441. this.get('taskDetailsProperties').forEach(function (property) {
  442. currentTask.set(property, task.Tasks[property]);
  443. }, this);
  444. }, this);
  445. } else {
  446. var tasks = [];
  447. data.tasks.forEach(function (task) {
  448. tasks.pushObject(App.upgradeEntity.create({type: 'TASK'}, task.Tasks));
  449. });
  450. item.set('tasks', tasks);
  451. }
  452. item.set('isTasksLoaded', true);
  453. }
  454. }, this);
  455. }
  456. }, this);
  457. },
  458. /**
  459. * Failures info may includes service_check and host_component failures. These two types should be displayed separately.
  460. */
  461. getServiceCheckItemSuccessCallback: function(data) {
  462. var task = data.tasks[0];
  463. var info = {
  464. hosts: [],
  465. host_detail: {}
  466. };
  467. if (task && task.Tasks && task.Tasks.structured_out && task.Tasks.structured_out.failures) {
  468. this.set('serviceCheckFailuresServicenames', task.Tasks.structured_out.failures.service_check || []);
  469. if (task.Tasks.structured_out.failures.host_component) {
  470. for (var hostname in task.Tasks.structured_out.failures.host_component){
  471. info.hosts.push(hostname);
  472. }
  473. info.host_detail = task.Tasks.structured_out.failures.host_component;
  474. }
  475. this.set('slaveComponentStructuredInfo', info);
  476. }
  477. },
  478. getSlaveComponentItemSuccessCallback: function(data) {
  479. var info = data.tasks[0];
  480. if (info && info.Tasks && info.Tasks.structured_out) {
  481. this.set('slaveComponentStructuredInfo', info.Tasks.structured_out);
  482. }
  483. },
  484. /**
  485. * downgrade confirmation popup
  486. * @param {object} event
  487. */
  488. confirmDowngrade: function (event) {
  489. var self = this;
  490. var currentVersion = this.get('currentVersion');
  491. return App.showConfirmationPopup(
  492. function() {
  493. self.downgrade.call(self, currentVersion, event);
  494. },
  495. Em.I18n.t('admin.stackUpgrade.downgrade.body').format(currentVersion.repository_name),
  496. null,
  497. Em.I18n.t('admin.stackUpgrade.dialog.downgrade.header').format(currentVersion.repository_name),
  498. Em.I18n.t('admin.stackUpgrade.downgrade.proceed')
  499. );
  500. },
  501. /**
  502. * make call to start downgrade process
  503. * @param {object} currentVersion
  504. * @param {object} event
  505. */
  506. downgrade: function (currentVersion, event) {
  507. this.set('requestInProgress', true);
  508. this.abortUpgrade();
  509. App.ajax.send({
  510. name: 'admin.downgrade.start',
  511. sender: this,
  512. data: {
  513. from: App.RepositoryVersion.find().findProperty('displayName', this.get('upgradeVersion')).get('repositoryVersion'),
  514. value: currentVersion.repository_version,
  515. label: currentVersion.repository_name,
  516. isDowngrade: true,
  517. upgradeType: this.get('upgradeType')
  518. },
  519. success: 'upgradeSuccessCallback',
  520. callback: function() {
  521. this.sender.set('requestInProgress', false);
  522. }
  523. });
  524. },
  525. /**
  526. * abort upgrade (in order to start Downgrade)
  527. */
  528. abortUpgrade: function () {
  529. var errorCallback = this.get('isDowngrade') ? 'abortDowngradeErrorCallback' : 'abortUpgradeErrorCallback';
  530. return App.ajax.send({
  531. name: 'admin.upgrade.abort',
  532. sender: this,
  533. data: {
  534. upgradeId: this.get('upgradeId'),
  535. isDowngrade: this.get('isDowngrade')
  536. },
  537. error: errorCallback
  538. });
  539. },
  540. /**
  541. * suspend upgrade (in order to restart it later)
  542. */
  543. abortUpgradeWithSuspend: function () {
  544. var errorCallback = this.get('isDowngrade') ? 'abortDowngradeErrorCallback' : 'abortUpgradeErrorCallback';
  545. return App.ajax.send({
  546. name: 'admin.upgrade.suspend',
  547. sender: this,
  548. data: {
  549. upgradeId: this.get('upgradeId'),
  550. isDowngrade: this.get('isDowngrade')
  551. },
  552. error: errorCallback
  553. });
  554. },
  555. /**
  556. * error callback of <code>abortUpgrade()</code>
  557. * @param {object} data
  558. */
  559. abortUpgradeErrorCallback: function (data) {
  560. var header = Em.I18n.t('admin.stackUpgrade.state.paused.fail.header');
  561. var body = Em.I18n.t('admin.stackUpgrade.state.paused.fail.body');
  562. if(data && data.responseText){
  563. try {
  564. var json = $.parseJSON(data.responseText);
  565. body = body + ' ' + json.message;
  566. } catch (err) {}
  567. }
  568. App.showAlertPopup(header, body);
  569. },
  570. /**
  571. * error callback of <code>abortDowngrade()</code>
  572. * @param {object} data
  573. */
  574. abortDowngradeErrorCallback: function (data) {
  575. var header = Em.I18n.t('admin.stackDowngrade.state.paused.fail.header');
  576. var body = Em.I18n.t('admin.stackDowngrade.state.paused.fail.body');
  577. if(data && data.responseText){
  578. try {
  579. var json = $.parseJSON(data.responseText);
  580. body = body + ' ' + json.message;
  581. } catch (err) {}
  582. }
  583. App.showAlertPopup(header, body);
  584. },
  585. retryUpgrade: function () {
  586. this.setProperties({
  587. requestInProgress: true,
  588. isRetryPending: true
  589. });
  590. return App.ajax.send({
  591. name: 'admin.upgrade.retry',
  592. sender: this,
  593. data: {
  594. upgradeId: this.get('upgradeId')
  595. }
  596. });
  597. },
  598. /**
  599. * make call to start upgrade process and show popup with current progress
  600. * @param {object} version
  601. */
  602. upgrade: function (version) {
  603. var self = this;
  604. this.set('requestInProgress', true);
  605. App.ajax.send({
  606. name: 'admin.upgrade.start',
  607. sender: this,
  608. data: version,
  609. success: 'upgradeSuccessCallback',
  610. error: 'upgradeErrorCallback',
  611. callback: function() {
  612. this.sender.set('requestInProgress', false);
  613. }
  614. });
  615. this.setDBProperty('currentVersion', this.get('currentVersion'));
  616. // Show a "preparing the upgrade..." dialog in case the api call returns too slow
  617. if (App.router.get('currentState.name') != 'stackUpgrade') {
  618. self.showPreparingUpgradeIndicator();
  619. }
  620. },
  621. /**
  622. * Should progress bar be displayed when preparing upgrade,
  623. * should show after Upgrade Options window and before Upgrade Wizard
  624. * @method showPreparingUpgradeIndicator
  625. */
  626. showPreparingUpgradeIndicator: function () {
  627. return App.ModalPopup.show({
  628. header: '',
  629. showFooter: false,
  630. showCloseButton: false,
  631. bodyClass: Em.View.extend({
  632. templateName: require('templates/wizard/step8/step8_log_popup'),
  633. controllerBinding: 'App.router.mainAdminStackAndUpgradeController',
  634. /**
  635. * Css-property for progress-bar
  636. * @type {string}
  637. */
  638. barWidth: 'width: 100%;',
  639. progressBarClass: 'progress progress-striped active log_popup',
  640. /**
  641. * Popup-message
  642. * @type {string}
  643. */
  644. message: Em.I18n.t('admin.stackUpgrade.dialog.prepareUpgrade.header'),
  645. /**
  646. * Hide popup when upgrade wizard is open
  647. * @method autoHide
  648. */
  649. autoHide: function () {
  650. if (!this.get('controller.requestInProgress')) {
  651. this.get('parentView').hide();
  652. }
  653. }.observes('controller.requestInProgress')
  654. })
  655. });
  656. },
  657. /**
  658. * error callback of <code>upgrade()</code>
  659. * @param {object} data
  660. */
  661. upgradeErrorCallback: function (data) {
  662. var header = Em.I18n.t('admin.stackVersions.upgrade.start.fail.title');
  663. var body = "";
  664. if(data && data.responseText){
  665. try {
  666. var json = $.parseJSON(data.responseText);
  667. body = json.message;
  668. } catch (err) {}
  669. }
  670. App.showAlertPopup(header, body);
  671. },
  672. /**
  673. * success callback of <code>upgrade()</code>
  674. * @param {object} data
  675. */
  676. upgradeSuccessCallback: function (data, opt, params) {
  677. this.set('upgradeData', null);
  678. this.set('upgradeId', data.resources[0].Upgrade.request_id);
  679. this.set('upgradeVersion', params.label);
  680. this.set('isDowngrade', !!params.isDowngrade);
  681. var upgradeMethod = this.get('upgradeMethods').findProperty('type', params.type);
  682. var upgradeTypeDisplayName = null;
  683. var upgradeType = null;
  684. if (upgradeMethod) {
  685. upgradeTypeDisplayName = upgradeMethod.get('displayName');
  686. upgradeType = upgradeMethod.get('type');
  687. }
  688. this.set('upgradeType', upgradeType);
  689. this.set('upgradeTypeDisplayName', upgradeTypeDisplayName);
  690. this.set('failuresTolerance', Em.Object.create({
  691. skipComponentFailures: params.skipComponentFailures == 'true',
  692. skipSCFailures: params.skipSCFailures == 'true'
  693. }));
  694. this.setDBProperties({
  695. upgradeVersion: params.label,
  696. upgradeId: data.resources[0].Upgrade.request_id,
  697. upgradeState: 'PENDING',
  698. isDowngrade: !!params.isDowngrade,
  699. upgradeType: upgradeType,
  700. upgradeTypeDisplayName: upgradeTypeDisplayName,
  701. failuresTolerance: Em.Object.create({
  702. skipComponentFailures: params.skipComponentFailures == 'true',
  703. skipSCFailures: params.skipSCFailures == 'true'
  704. })
  705. });
  706. App.set('upgradeState', 'PENDING');
  707. App.clusterStatus.setClusterStatus({
  708. wizardControllerName: this.get('name'),
  709. localdb: App.db.data
  710. });
  711. this.load();
  712. this.openUpgradeDialog();
  713. },
  714. /**
  715. * success callback of updating upgrade options including failures tolerance. etc
  716. * @param {object} data
  717. */
  718. updateOptionsSuccessCallback: function (data, opt, params) {
  719. this.set('failuresTolerance', Em.Object.create({
  720. skipComponentFailures: params.skipComponentFailures == 'true',
  721. skipSCFailures: params.skipSCFailures == 'true'
  722. }));
  723. },
  724. /**
  725. * run upgrade checks and add results to each method object and set selected method
  726. * @param {Em.Object} version
  727. */
  728. runUpgradeMethodChecks: function(version) {
  729. this.get('upgradeMethods').forEach(function (method) {
  730. if (method.get('allowed')) {
  731. this.runPreUpgradeCheckOnly({
  732. value: version.get('repositoryVersion'),
  733. label: version.get('displayName'),
  734. type: method.get('type')
  735. });
  736. } else {
  737. //if method not supported in current stack version, mark as check completed
  738. method.setProperties({
  739. isCheckComplete: false,
  740. isCheckRequestInProgress: false,
  741. action: ''
  742. });
  743. }
  744. }, this);
  745. },
  746. getConfigsWarnings: function (configsMergeWarning) {
  747. var configs = [];
  748. if (configsMergeWarning && Em.get(configsMergeWarning, 'UpgradeChecks.status') === 'WARNING') {
  749. var configsMergeCheckData = Em.get(configsMergeWarning, 'UpgradeChecks.failed_detail');
  750. if (configsMergeCheckData && Em.isArray(configsMergeCheckData)) {
  751. configs = configsMergeCheckData.map(function (item) {
  752. var isDeprecated = Em.isNone(item.new_stack_value),
  753. willBeRemoved = Em.isNone(item.result_value);
  754. return {
  755. type: item.type,
  756. name: item.property,
  757. currentValue: item.current,
  758. recommendedValue: isDeprecated ? Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.deprecated') : item.new_stack_value,
  759. isDeprecated: isDeprecated,
  760. resultingValue: willBeRemoved ? Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.willBeRemoved') : item.result_value,
  761. willBeRemoved: willBeRemoved
  762. };
  763. });
  764. }
  765. }
  766. return configs;
  767. },
  768. /**
  769. * Open upgrade options window: upgrade type and failures tolerance
  770. * @param {boolean} isInUpgradeWizard
  771. * @param {object} version
  772. * @return App.ModalPopup
  773. */
  774. upgradeOptions: function (isInUpgradeWizard, version) {
  775. var self = this,
  776. upgradeMethods = this.get('upgradeMethods'),
  777. runningCheckRequests = this.get('runningCheckRequests');
  778. if (!isInUpgradeWizard) {
  779. upgradeMethods.setEach('isCheckRequestInProgress', true);
  780. upgradeMethods.setEach('selected', false);
  781. var request = this.getSupportedUpgradeTypes(Ember.Object.create({
  782. stackName: App.get('currentStackVersion').split('-')[0],
  783. stackVersion: App.get('currentStackVersion').split('-')[1],
  784. toVersion: version.get('repositoryVersion')
  785. })).done(function () {
  786. if (App.get('router.currentState.name') === 'versions' && App.get('router.currentState.parentState.name') === 'stackAndUpgrade') {
  787. self.runUpgradeMethodChecks(version);
  788. }
  789. }).always(function () {
  790. self.set('runningCheckRequests', runningCheckRequests.rejectProperty('type', 'ALL'));
  791. });
  792. request.type = 'ALL';
  793. this.get('runningCheckRequests').push(request);
  794. }
  795. return App.ModalPopup.show({
  796. encodeBody: false,
  797. primary: isInUpgradeWizard ? Em.I18n.t('ok') : Em.I18n.t('common.proceed'),
  798. primaryClass: 'btn-success',
  799. classNames: ['upgrade-options-popup'],
  800. header: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.header'),
  801. bodyClass: Em.View.extend({
  802. templateName: require('templates/main/admin/stack_upgrade/upgrade_options'),
  803. didInsertElement: function () {
  804. App.tooltip($(".failure-tolerance-tooltip"), {
  805. placement: "top",
  806. title: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.tolerance.tooltip')
  807. });
  808. Em.run.later(this, function () {
  809. App.tooltip($(".thumbnail.check-failed"), {
  810. placement: "bottom",
  811. title: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.preCheck.failed.tooltip')
  812. });
  813. App.tooltip($(".not-allowed-by-version"), {
  814. placement: "bottom",
  815. title: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.notAllowed')
  816. });
  817. }, 1000);
  818. },
  819. upgradeMethods: function () {
  820. self.updateSelectedMethod(isInUpgradeWizard);
  821. return self.get('upgradeMethods');
  822. }.property().volatile(),
  823. isInUpgradeWizard: isInUpgradeWizard,
  824. versionText: isInUpgradeWizard ? '' : Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.bodyMsg.version').format(version.get('displayName')),
  825. selectMethod: function (event) {
  826. if (isInUpgradeWizard || !event.context.get('allowed') || event.context.get('isPrecheckFailed')) return;
  827. var selectedMethod = event.context;
  828. self.updateSelectedMethod(isInUpgradeWizard);
  829. self.get('upgradeMethods').forEach(function (method) {
  830. method.set('selected', false);
  831. });
  832. selectedMethod.set('selected', true);
  833. this.set('parentView.selectedMethod', selectedMethod);
  834. },
  835. runAction: function (event) {
  836. var method = event.context,
  837. action = method.get('action');
  838. if (action) {
  839. this.get(action)(event);
  840. }
  841. },
  842. rerunCheck: function (event) {
  843. self.runPreUpgradeCheckOnly({
  844. value: version.get('repositoryVersion'),
  845. label: version.get('displayName'),
  846. type: event.context.get('type')
  847. });
  848. },
  849. openMessage: function (event) {
  850. if (isInUpgradeWizard || !event.context.get('allowed')) return;
  851. var data = event.context.get('precheckResultsData');
  852. var failTitle = Em.I18n.t('popup.clusterCheck.Upgrade.fail.title');
  853. var failAlert = new Em.Handlebars.SafeString(Em.I18n.t('popup.clusterCheck.Upgrade.fail.alert'));
  854. var bypassedFailures = data.items.filterProperty('UpgradeChecks.status', 'BYPASS').length > 0;
  855. if (data.items.filterProperty('UpgradeChecks.status', 'ERROR').length == 0 && bypassedFailures) {
  856. failTitle = Em.I18n.t('popup.clusterCheck.Upgrade.bypassed-failures.title');
  857. failAlert = new Em.Handlebars.SafeString(Em.I18n.t('popup.clusterCheck.Upgrade.bypassed-failures.alert'));
  858. }
  859. var header = Em.I18n.t('popup.clusterCheck.Upgrade.header').format(version.get('displayName')),
  860. warningTitle = Em.I18n.t('popup.clusterCheck.Upgrade.warning.title'),
  861. warningAlert = new Em.Handlebars.SafeString(Em.I18n.t('popup.clusterCheck.Upgrade.warning.alert')),
  862. configsMergeWarning = data.items.findProperty('UpgradeChecks.id', "CONFIG_MERGE"),
  863. popupData = {
  864. items: data.items.rejectProperty('UpgradeChecks.id', 'CONFIG_MERGE')
  865. },
  866. configs = self.getConfigsWarnings(configsMergeWarning);
  867. App.showClusterCheckPopup(popupData, {
  868. header: header,
  869. failTitle: failTitle,
  870. failAlert: failAlert,
  871. warningTitle: warningTitle,
  872. warningAlert: warningAlert,
  873. primary: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.preCheck.rerun'),
  874. secondary: Em.I18n.t('common.cancel'),
  875. bypassedFailures: bypassedFailures,
  876. callback: function () {
  877. self.runPreUpgradeCheckOnly.call(self, {
  878. value: version.get('repositoryVersion'),
  879. label: version.get('displayName'),
  880. type: event.context.get('type')
  881. });
  882. }
  883. }, configs, version.get('displayName'));
  884. }
  885. }),
  886. /**
  887. * @type {Em.Object}
  888. * @default null
  889. */
  890. selectedMethod: null,
  891. skipComponentFailures: self.get('failuresTolerance.skipComponentFailures'),
  892. skipSCFailures: self.get('failuresTolerance.skipSCFailures'),
  893. disablePrimary: function () {
  894. if (isInUpgradeWizard) return false;
  895. var selectedMethod = this.get('selectedMethod');
  896. return (selectedMethod ? (selectedMethod.get('isPrecheckFailed') || selectedMethod.get('isCheckRequestInProgress')) : true);
  897. }.property('selectedMethod', 'selectedMethod.isPrecheckFailed', 'selectedMethod.isCheckRequestInProgress'),
  898. onPrimary: function () {
  899. this.hide();
  900. if (isInUpgradeWizard) {
  901. return App.ajax.send({
  902. name: 'admin.upgrade.update.options',
  903. sender: self,
  904. data: {
  905. upgradeId: self.get('upgradeId'),
  906. skipComponentFailures: Boolean(this.get('skipComponentFailures')).toString(),
  907. skipSCFailures: Boolean(this.get('skipSCFailures')).toString()
  908. },
  909. success: 'updateOptionsSuccessCallback'
  910. });
  911. } else {
  912. var upgradeMethod = self.get('upgradeMethods').findProperty('selected');
  913. version.upgradeType = upgradeMethod.get('type');
  914. version.upgradeTypeDisplayName = upgradeMethod.get('displayName');
  915. version.skipComponentFailures = this.get('skipComponentFailures');
  916. version.skipSCFailures = this.get('skipSCFailures');
  917. var fromVersion = self.get('upgradeVersion') || App.RepositoryVersion.find().findProperty('status', 'CURRENT').get('displayName');
  918. var toVersion = version.get('displayName');
  919. var bodyMessage = Em.Object.create({
  920. confirmButton: Em.I18n.t('yes'),
  921. confirmMsg: upgradeMethod.get('type') === 'ROLLING' ?
  922. Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.RU.confirm.msg').format(fromVersion, toVersion) :
  923. Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.EU.confirm.msg').format(fromVersion, toVersion)
  924. });
  925. return App.showConfirmationFeedBackPopup(function (query) {
  926. return self.runPreUpgradeCheck.call(self, version);
  927. }, bodyMessage);
  928. }
  929. }
  930. });
  931. },
  932. /**
  933. * open upgrade options from upgrade wizard
  934. */
  935. openUpgradeOptions: function () {
  936. if (this.get('isDowngrade')) return;
  937. this.upgradeOptions(true, null);
  938. },
  939. /**
  940. * upgrade confirmation popup including upgrade options: upgrade type and failures tolerance
  941. * @param {object} version
  942. * @return App.ModalPopup
  943. */
  944. confirmUpgrade: function (version) {
  945. this.upgradeOptions(false, version);
  946. },
  947. /**
  948. * send request for pre upgrade check only
  949. */
  950. runPreUpgradeCheckOnly: function (data) {
  951. if (App.get('supports.preUpgradeCheck')) {
  952. var method = this.get('upgradeMethods').findProperty('type', data.type);
  953. method.setProperties({
  954. isCheckComplete: false,
  955. isCheckRequestInProgress: true,
  956. action: ''
  957. });
  958. var request = App.ajax.send({
  959. name: "admin.upgrade.pre_upgrade_check",
  960. sender: this,
  961. data: data,
  962. success: 'runPreUpgradeCheckOnlySuccess',
  963. error: 'runPreUpgradeCheckOnlyError',
  964. callback: function () {
  965. var runningCheckRequests = this.sender.get('runningCheckRequests');
  966. method.set('isCheckRequestInProgress', false);
  967. this.sender.set('runningCheckRequests', runningCheckRequests.rejectProperty('type', this.data.type));
  968. }
  969. });
  970. request.type = data.type;
  971. this.get('runningCheckRequests').push(request);
  972. }
  973. },
  974. /**
  975. * send request to get available upgrade tye names
  976. */
  977. getSupportedUpgradeTypes: function(data) {
  978. return App.ajax.send({
  979. name: "admin.upgrade.get_supported_upgradeTypes",
  980. sender: this,
  981. data: data,
  982. success: "getSupportedUpgradeTypesSuccess"
  983. });
  984. },
  985. /**
  986. * success callback of <code>getSupportedUpgradeTypes()</code>
  987. * @param data {object}
  988. */
  989. getSupportedUpgradeTypesSuccess: function (data) {
  990. var supportedUpgradeTypes = data.items[0] && data.items[0].CompatibleRepositoryVersions.upgrade_types;
  991. this.get('upgradeMethods').forEach(function (method) {
  992. method.set('allowed', supportedUpgradeTypes && !!supportedUpgradeTypes.contains(method.get('type')));
  993. });
  994. },
  995. /**
  996. * success callback of <code>runPreUpgradeCheckOnly()</code>
  997. * Show a message how many fails/warnings/bypass/passed
  998. * on clicking that message a popup window show up
  999. * @param data {object}
  1000. * @param opt {object}
  1001. * @param params {object}
  1002. */
  1003. runPreUpgradeCheckOnlySuccess: function (data, opt, params) {
  1004. var self = this;
  1005. var message = '';
  1006. var messageClass = 'GREEN';
  1007. var allowedToStart = true;
  1008. var messageIconClass = 'icon-ok';
  1009. var bypassedFailures = false;
  1010. if (data.items.someProperty('UpgradeChecks.status', 'WARNING')) {
  1011. message = message + data.items.filterProperty('UpgradeChecks.status', 'WARNING').length + ' Warning ';
  1012. messageClass = 'ORANGE';
  1013. allowedToStart = true;
  1014. messageIconClass = 'icon-warning-sign';
  1015. }
  1016. if (data.items.someProperty('UpgradeChecks.status', 'BYPASS')) {
  1017. message = data.items.filterProperty('UpgradeChecks.status', 'BYPASS').length + ' Error ' + message;
  1018. messageClass = 'RED';
  1019. allowedToStart = true;
  1020. messageIconClass = 'icon-remove';
  1021. bypassedFailures = true;
  1022. }
  1023. if (data.items.someProperty('UpgradeChecks.status', 'FAIL')) {
  1024. message = data.items.filterProperty('UpgradeChecks.status', 'FAIL').length + ' Required ' + message;
  1025. messageClass = 'RED';
  1026. allowedToStart = false;
  1027. messageIconClass = 'icon-remove';
  1028. }
  1029. if (!message) {
  1030. message = Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.preCheck.allPassed');
  1031. }
  1032. var method = self.get('upgradeMethods').findProperty('type', params.type);
  1033. method.setProperties({
  1034. precheckResultsMessage: message,
  1035. precheckResultsMessageClass: messageClass,
  1036. precheckResultsTitle: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.preCheck.msg.title'),
  1037. isPrecheckFailed: allowedToStart === false,
  1038. precheckResultsMessageIconClass: messageIconClass,
  1039. precheckResultsData: data,
  1040. isCheckComplete: true,
  1041. bypassedFailures: bypassedFailures,
  1042. action: 'openMessage'
  1043. });
  1044. this.updateSelectedMethod(false);
  1045. Em.run.later(this, function () {
  1046. // add tooltip for the type with preCheck errors
  1047. App.tooltip($(".thumbnail.check-failed"), {
  1048. placement: "bottom",
  1049. title: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.preCheck.failed.tooltip')
  1050. });
  1051. // destroy the tooltip for the type wo preCheck errors
  1052. $(".thumbnail").not(".check-failed").not(".not-allowed-by-version").tooltip("destroy");
  1053. }, 1000);
  1054. },
  1055. runPreUpgradeCheckOnlyError: function (request, ajaxOptions, error, data, params) {
  1056. var method = this.get('upgradeMethods').findProperty('type', params.type);
  1057. method.setProperties({
  1058. precheckResultsMessage: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.preCheck.msg.failed.link'),
  1059. precheckResultsTitle: Em.I18n.t('admin.stackVersions.version.upgrade.upgradeOptions.preCheck.msg.failed.title'),
  1060. precheckResultsMessageClass: 'RED',
  1061. isPrecheckFailed: true,
  1062. precheckResultsMessageIconClass: 'icon-warning-sign',
  1063. action: 'rerunCheck'
  1064. });
  1065. },
  1066. /**
  1067. * In Upgrade Wizard: update which method already been selected on open
  1068. * Not in upgrade wizard: de-select the method with pre-check errors
  1069. * @param isInUpgradeWizard {boolean}
  1070. */
  1071. updateSelectedMethod: function(isInUpgradeWizard) {
  1072. var self = this;
  1073. if (isInUpgradeWizard) {
  1074. this.get('upgradeMethods').forEach(function(method){
  1075. if (method.get('type') == self.get('upgradeType')) {
  1076. method.set('selected', true);
  1077. } else {
  1078. method.set('selected', false);
  1079. }
  1080. });
  1081. } else {
  1082. var ruMethod = this.get('upgradeMethods').findProperty('type', 'ROLLING');
  1083. var euMethod = this.get('upgradeMethods').findProperty('type', 'NON_ROLLING');
  1084. if (ruMethod && ruMethod.get('isPrecheckFailed')) ruMethod.set('selected', false);
  1085. if (euMethod && euMethod.get('isPrecheckFailed')) euMethod.set('selected', false);
  1086. }
  1087. },
  1088. /**
  1089. * send request for pre upgrade check
  1090. * @param version
  1091. */
  1092. runPreUpgradeCheck: function(version) {
  1093. var params = {
  1094. value: version.get('repositoryVersion'),
  1095. label: version.get('displayName'),
  1096. type: version.get('upgradeType'),
  1097. skipComponentFailures: version.get('skipComponentFailures') ? 'true' : 'false',
  1098. skipSCFailures: version.get('skipSCFailures') ? 'true' : 'false'
  1099. };
  1100. if (App.get('supports.preUpgradeCheck')) {
  1101. this.set('requestInProgress', true);
  1102. App.ajax.send({
  1103. name: "admin.upgrade.pre_upgrade_check",
  1104. sender: this,
  1105. data: params,
  1106. success: "runPreUpgradeCheckSuccess",
  1107. error: "runPreUpgradeCheckError"
  1108. });
  1109. } else {
  1110. this.upgrade(params);
  1111. }
  1112. },
  1113. /**
  1114. * success callback of <code>runPreUpgradeCheckSuccess()</code>
  1115. * if there are some fails - it shows popup else run upgrade
  1116. * @param data {object}
  1117. * @param opt {object}
  1118. * @param params {object}
  1119. * @returns {App.ModalPopup|undefined}
  1120. */
  1121. runPreUpgradeCheckSuccess: function (data, opt, params) {
  1122. var self = this;
  1123. if (data.items.someProperty('UpgradeChecks.status', 'FAIL') || data.items.someProperty('UpgradeChecks.status', 'WARNING') || data.items.someProperty('UpgradeChecks.status', 'BYPASS')) {
  1124. this.set('requestInProgress', false);
  1125. var hasFails = data.items.someProperty('UpgradeChecks.status', 'FAIL'),
  1126. header = Em.I18n.t('popup.clusterCheck.Upgrade.header').format(params.label),
  1127. failTitle = Em.I18n.t('popup.clusterCheck.Upgrade.fail.title'),
  1128. failAlert = new Em.Handlebars.SafeString(Em.I18n.t('popup.clusterCheck.Upgrade.fail.alert')),
  1129. warningTitle = Em.I18n.t('popup.clusterCheck.Upgrade.warning.title'),
  1130. warningAlert = new Em.Handlebars.SafeString(Em.I18n.t('popup.clusterCheck.Upgrade.warning.alert')),
  1131. bypassedFailures = data.items.someProperty('UpgradeChecks.status', 'BYPASS').length > 0,
  1132. configsMergeWarning = data.items.findProperty('UpgradeChecks.id', 'CONFIG_MERGE'),
  1133. popupData = {
  1134. items: data.items.rejectProperty('UpgradeChecks.id', 'CONFIG_MERGE')
  1135. },
  1136. configs = this.getConfigsWarnings(configsMergeWarning);
  1137. App.showClusterCheckPopup(popupData, {
  1138. header: header,
  1139. failTitle: failTitle,
  1140. failAlert: failAlert,
  1141. warningTitle: warningTitle,
  1142. warningAlert: warningAlert,
  1143. bypassedFailures: bypassedFailures,
  1144. noCallbackCondition: hasFails,
  1145. callback: function () {
  1146. self.upgrade(params);
  1147. }
  1148. }, configs, params.label);
  1149. } else {
  1150. this.upgrade(params);
  1151. }
  1152. },
  1153. runPreUpgradeCheckError: function() {
  1154. this.set('requestInProgress', false);
  1155. },
  1156. confirmRetryUpgrade: function (version) {
  1157. var self = this;
  1158. return App.showConfirmationPopup(
  1159. function () {
  1160. self.retryUpgrade();
  1161. },
  1162. Em.I18n.t('admin.stackUpgrade.upgrade.retry.confirm.body').format(version.get('displayName')),
  1163. null,
  1164. Em.I18n.t('admin.stackUpgrade.dialog.header').format(version.get('upgradeTypeDislayName'), version.get('displayName'))
  1165. );
  1166. },
  1167. confirmRetryDowngrade: function () {
  1168. var self = this,
  1169. currentVersion = this.get('currentVersion');
  1170. return App.showConfirmationPopup(
  1171. function() {
  1172. self.retryUpgrade();
  1173. },
  1174. Em.I18n.t('admin.stackUpgrade.downgrade.retry.body').format(currentVersion.repository_name),
  1175. null,
  1176. Em.I18n.t('admin.stackUpgrade.dialog.downgrade.header').format(currentVersion.repository_name),
  1177. Em.I18n.t('admin.stackUpgrade.downgrade.proceed')
  1178. );
  1179. },
  1180. /**
  1181. * confirmation popup before install repository version
  1182. */
  1183. installRepoVersionConfirmation: function (repo) {
  1184. var self = this;
  1185. return App.showConfirmationPopup(function () {
  1186. self.installRepoVersion(repo);
  1187. },
  1188. Em.I18n.t('admin.stackVersions.version.install.confirm').format(repo.get('displayName'))
  1189. );
  1190. },
  1191. /**
  1192. * sends request to install repoVersion to the cluster
  1193. * and create clusterStackVersion resourse
  1194. * @param {Em.Object} repo
  1195. * @return {$.ajax}
  1196. * @method installRepoVersion
  1197. */
  1198. installRepoVersion: function (repo) {
  1199. this.set('requestInProgress', true);
  1200. this.set('requestInProgressRepoId', repo.get('id'));
  1201. var data = {
  1202. ClusterStackVersions: {
  1203. stack: repo.get('stackVersionType'),
  1204. version: repo.get('stackVersionNumber'),
  1205. repository_version: repo.get('repositoryVersion')
  1206. },
  1207. id: repo.get('id')
  1208. };
  1209. return App.ajax.send({
  1210. name: 'admin.stack_version.install.repo_version',
  1211. sender: this,
  1212. data: data,
  1213. success: 'installRepoVersionSuccess',
  1214. error: 'installRepoVersionError',
  1215. callback: function() {
  1216. this.sender.set('requestInProgress', false);
  1217. this.sender.set('requestInProgressRepoId', null);
  1218. }
  1219. });
  1220. },
  1221. /**
  1222. * transform repo data into json for
  1223. * saving changes to repository version
  1224. * @param {Em.Object} repo
  1225. * @returns {{operating_systems: Array}}
  1226. */
  1227. prepareRepoForSaving: function(repo) {
  1228. var repoVersion = { "operating_systems": [] };
  1229. repo.get('operatingSystems').forEach(function (os, k) {
  1230. repoVersion.operating_systems.push({
  1231. "OperatingSystems": {
  1232. "os_type": os.get("osType")
  1233. },
  1234. "repositories": []
  1235. });
  1236. os.get('repositories').forEach(function (repository) {
  1237. repoVersion.operating_systems[k].repositories.push({
  1238. "Repositories": {
  1239. "base_url": repository.get('baseUrl'),
  1240. "repo_id": repository.get('repoId'),
  1241. "repo_name": repository.get('repoName')
  1242. }
  1243. });
  1244. });
  1245. });
  1246. return repoVersion;
  1247. },
  1248. /**
  1249. * Return stack version for the repo object
  1250. * @param {Em.Object} repo
  1251. * */
  1252. getStackVersionNumber: function(repo){
  1253. var stackVersionNumber = repo.get('stackVersion');
  1254. if(null == stackVersionNumber)
  1255. stackVersionNumber = App.get('currentStackVersion');
  1256. return stackVersionNumber;
  1257. },
  1258. /**
  1259. * perform validation if <code>skip<code> is false and run save if
  1260. * validation successfull or run save without validation is <code>skip<code> is true
  1261. * @param {Em.Object} repo
  1262. * @param {boolean} skip
  1263. * @returns {$.Deferred}
  1264. */
  1265. saveRepoOS: function (repo, skip) {
  1266. var self = this;
  1267. var deferred = $.Deferred();
  1268. this.validateRepoVersions(repo, skip).done(function(data) {
  1269. if (data.length > 0) {
  1270. deferred.resolve(data);
  1271. } else {
  1272. var repoVersion = self.prepareRepoForSaving(repo);
  1273. var stackVersionNumber = self.getStackVersionNumber(repo);
  1274. App.ajax.send({
  1275. name: 'admin.stack_versions.edit.repo',
  1276. sender: this,
  1277. data: {
  1278. stackName: App.get('currentStackName'),
  1279. stackVersion: stackVersionNumber,
  1280. repoVersionId: repo.get('repoVersionId'),
  1281. repoVersion: repoVersion
  1282. }
  1283. }).success(function() {
  1284. deferred.resolve([]);
  1285. });
  1286. }
  1287. });
  1288. return deferred.promise();
  1289. },
  1290. /**
  1291. * send request for validation for each repository
  1292. * @param {Em.Object} repo
  1293. * @param {boolean} skip
  1294. * @returns {*}
  1295. */
  1296. validateRepoVersions: function(repo, skip) {
  1297. var deferred = $.Deferred(),
  1298. totalCalls = 0,
  1299. invalidUrls = [];
  1300. if (skip) {
  1301. deferred.resolve(invalidUrls);
  1302. } else {
  1303. var stackVersionNumber = this.getStackVersionNumber(repo);
  1304. repo.get('operatingSystems').forEach(function (os) {
  1305. if (os.get('isSelected')) {
  1306. os.get('repositories').forEach(function (repo) {
  1307. totalCalls++;
  1308. App.ajax.send({
  1309. name: 'admin.stack_versions.validate.repo',
  1310. sender: this,
  1311. data: {
  1312. repo: repo,
  1313. repoId: repo.get('repoId'),
  1314. baseUrl: repo.get('baseUrl'),
  1315. osType: os.get('osType'),
  1316. stackName: App.get('currentStackName'),
  1317. stackVersion: stackVersionNumber
  1318. }
  1319. })
  1320. .success(function () {
  1321. totalCalls--;
  1322. if (totalCalls === 0) deferred.resolve(invalidUrls);
  1323. })
  1324. .error(function () {
  1325. repo.set('hasError', true);
  1326. invalidUrls.push(repo);
  1327. totalCalls--;
  1328. if (totalCalls === 0) deferred.resolve(invalidUrls);
  1329. });
  1330. });
  1331. } else {
  1332. return deferred.resolve(invalidUrls);
  1333. }
  1334. });
  1335. }
  1336. return deferred.promise();
  1337. },
  1338. /**
  1339. * success callback for <code>installRepoVersion()<code>
  1340. * saves request id to the db
  1341. * @param data
  1342. * @param opt
  1343. * @param params
  1344. * @method installStackVersionSuccess
  1345. */
  1346. installRepoVersionSuccess: function (data, opt, params) {
  1347. if(data && data.statusText == "timeout") {
  1348. App.showAlertPopup(Em.I18n.t('admin.stackVersions.upgrade.installPackage.fail.title'), Em.I18n.t('admin.stackVersions.upgrade.installPackage.fail.timeout'));
  1349. return false;
  1350. }
  1351. var version = App.RepositoryVersion.find(params.id);
  1352. App.db.set('repoVersionInstall', 'id', [data.Requests.id]);
  1353. App.clusterStatus.setClusterStatus({
  1354. wizardControllerName: this.get('name'),
  1355. localdb: App.db.data
  1356. });
  1357. version.set('defaultStatus', 'INSTALLING');
  1358. if (version.get('stackVersion')) {
  1359. version.set('stackVersion.state', 'INSTALLING');
  1360. }
  1361. },
  1362. /**
  1363. * error callback for <code>installRepoVersion()<code>
  1364. * show the error message
  1365. * @param data
  1366. * @method installStackVersionSuccess
  1367. */
  1368. installRepoVersionError: function (data) {
  1369. var header = Em.I18n.t('admin.stackVersions.upgrade.installPackage.fail.title');
  1370. var body = "";
  1371. if(data && data.responseText){
  1372. try {
  1373. var json = $.parseJSON(data.responseText);
  1374. body = json.message;
  1375. } catch (err) {}
  1376. }
  1377. if(data && data.statusText == "timeout") {
  1378. body = Em.I18n.t('admin.stackVersions.upgrade.installPackage.fail.timeout');
  1379. }
  1380. App.showAlertPopup(header, body);
  1381. },
  1382. /**
  1383. * opens a popup with installations state per host
  1384. * @param {Em.Object} version
  1385. * @method showProgressPopup
  1386. */
  1387. showProgressPopup: function(version) {
  1388. var popupTitle = Em.I18n.t('admin.stackVersions.details.install.hosts.popup.title').format(version.get('displayName'));
  1389. var requestIds = App.get('testMode') ? [1] : App.db.get('repoVersionInstall', 'id');
  1390. var hostProgressPopupController = App.router.get('highAvailabilityProgressPopupController');
  1391. hostProgressPopupController.initPopup(popupTitle, requestIds, this);
  1392. },
  1393. /**
  1394. * reset upgradeState to INIT when upgrade is COMPLETED
  1395. * and clean auxiliary data
  1396. */
  1397. finish: function () {
  1398. if (App.get('upgradeState') === 'COMPLETED') {
  1399. var upgradeVersion = this.get('upgradeVersion') && this.get('upgradeVersion').match(/[a-zA-Z]+\-\d+\.\d+/);
  1400. this.setDBProperties({
  1401. upgradeId: undefined,
  1402. upgradeState: 'INIT',
  1403. upgradeVersion: undefined,
  1404. currentVersion: undefined,
  1405. upgradeTypeDisplayName: undefined,
  1406. upgradeType: undefined,
  1407. failuresTolerance: undefined,
  1408. isDowngrade: undefined,
  1409. downgradeAllowed: undefined
  1410. });
  1411. App.clusterStatus.setClusterStatus({
  1412. localdb: App.db.data
  1413. });
  1414. if (upgradeVersion && upgradeVersion[0]) {
  1415. App.set('currentStackVersion', upgradeVersion[0]);
  1416. }
  1417. App.set('upgradeState', 'INIT');
  1418. }
  1419. }.observes('App.upgradeState'),
  1420. /**
  1421. * Check <code>App.upgradeState</code> for HOLDING
  1422. * If it is, send request to check if current item is Finalize
  1423. * @method updateFinalize
  1424. */
  1425. updateFinalize: function () {
  1426. var upgradeState = App.get('upgradeState');
  1427. if (upgradeState === 'HOLDING') {
  1428. return App.ajax.send({
  1429. name: 'admin.upgrade.finalizeContext',
  1430. sender: this,
  1431. success: 'updateFinalizeSuccessCallback',
  1432. error: 'updateFinalizeErrorCallback'
  1433. })
  1434. }
  1435. else {
  1436. this.set('isFinalizeItem', false);
  1437. }
  1438. }.observes('App.upgradeState'),
  1439. /**
  1440. *
  1441. * @param {object|null} data
  1442. * @method updateFinalizeSuccessCallback
  1443. */
  1444. updateFinalizeSuccessCallback: function (data) {
  1445. var context = data ? Em.get(data, 'items.firstObject.upgrade_groups.firstObject.upgrade_items.firstObject.UpgradeItem.context') : '';
  1446. this.set('isFinalizeItem', context === this.get('finalizeContext'));
  1447. },
  1448. updateFinalizeErrorCallback: function() {
  1449. this.set('isFinalizeItem', false);
  1450. },
  1451. /**
  1452. * show dialog with tasks of upgrade
  1453. * @return {App.ModalPopup}
  1454. */
  1455. openUpgradeDialog: function () {
  1456. App.router.transitionTo('admin.stackUpgrade');
  1457. },
  1458. /**
  1459. * returns url to get data for repoVersion or clusterStackVersion
  1460. * @param {Boolean} stack true if load clusterStackVersion
  1461. * @param {Boolean} fullLoad true if load all data
  1462. * @returns {String}
  1463. * @method getUrl
  1464. */
  1465. getUrl: function(stack, fullLoad) {
  1466. if (App.get('testMode')) {
  1467. return stack ? this.get('mockStackUrl') : this.get('mockRepoUrl')
  1468. } else {
  1469. if (fullLoad) {
  1470. return stack ? this.get('realStackUrl') : this.get('realRepoUrl');
  1471. } else {
  1472. return this.get('realUpdateUrl');
  1473. }
  1474. }
  1475. },
  1476. /**
  1477. * get stack versions from server and push it to model
  1478. * @return {*}
  1479. * @method loadStackVersionsToModel
  1480. */
  1481. loadStackVersionsToModel: function (fullLoad) {
  1482. var dfd = $.Deferred();
  1483. App.HttpClient.get(this.getUrl(true, fullLoad), App.stackVersionMapper, {
  1484. complete: function () {
  1485. dfd.resolve();
  1486. }
  1487. });
  1488. return dfd.promise();
  1489. },
  1490. /**
  1491. * get repo versions from server and push it to model
  1492. * @return {*}
  1493. * @params {Boolean} isUpdate - if true loads part of data that need to be updated
  1494. * @method loadRepoVersionsToModel()
  1495. */
  1496. loadRepoVersionsToModel: function () {
  1497. var dfd = $.Deferred();
  1498. App.HttpClient.get(this.getUrl(false, true), App.repoVersionMapper, {
  1499. complete: function () {
  1500. dfd.resolve();
  1501. }
  1502. });
  1503. return dfd.promise();
  1504. },
  1505. /**
  1506. * set status to Upgrade item
  1507. * @param item
  1508. * @param status
  1509. */
  1510. setUpgradeItemStatus: function(item, status) {
  1511. this.set('requestInProgress', true);
  1512. return App.ajax.send({
  1513. name: 'admin.upgrade.upgradeItem.setState',
  1514. sender: this,
  1515. data: {
  1516. upgradeId: item.get('request_id'),
  1517. itemId: item.get('stage_id'),
  1518. groupId: item.get('group_id'),
  1519. status: status
  1520. },
  1521. callback: function() {
  1522. this.sender.set('requestInProgress', false);
  1523. }
  1524. }).done(function () {
  1525. item.set('status', status);
  1526. });
  1527. },
  1528. currentVersionObserver: function () {
  1529. var versionNumber = this.get('currentVersion.repository_version');
  1530. var currentVersionObject = App.RepositoryVersion.find().findProperty('status', 'CURRENT');
  1531. var versionName = currentVersionObject && currentVersionObject.get('stackVersionType');
  1532. App.set('isStormMetricsSupported', versionName != 'HDP' || stringUtils.compareVersions(versionNumber, '2.2.2') > -1 || !versionNumber);
  1533. }.observes('currentVersion.repository_version'),
  1534. /**
  1535. * get the installed repositories of HDP from server
  1536. */
  1537. loadRepositories: function () {
  1538. if (App.router.get('clusterController.isLoaded')) {
  1539. var nameVersionCombo = App.get('currentStackVersion');
  1540. var stackName = nameVersionCombo.split('-')[0];
  1541. var stackVersion = nameVersionCombo.split('-')[1];
  1542. App.ajax.send({
  1543. name: 'cluster.load_repositories',
  1544. sender: this,
  1545. data: {
  1546. stackName: stackName,
  1547. stackVersion: stackVersion
  1548. },
  1549. success: 'loadRepositoriesSuccessCallback',
  1550. error: 'loadRepositoriesErrorCallback'
  1551. });
  1552. }
  1553. }.observes('App.router.clusterController.isLoaded'),
  1554. loadRepositoriesSuccessCallback: function (data) {
  1555. var allRepos = [];
  1556. data.items.forEach(function (os) {
  1557. os.repositories.forEach(function (repository) {
  1558. var osType = repository.Repositories.os_type;
  1559. var repo = Em.Object.create({
  1560. baseUrl: repository.Repositories.base_url,
  1561. osType: osType,
  1562. repoId: repository.Repositories.repo_id,
  1563. repoName : repository.Repositories.repo_name,
  1564. stackName : repository.Repositories.stack_name,
  1565. stackVersion : repository.Repositories.stack_version,
  1566. isFirst: false
  1567. });
  1568. var group = allRepos.findProperty('name', osType);
  1569. if (!group) {
  1570. group = {
  1571. name: osType,
  1572. repositories: []
  1573. };
  1574. repo.set('isFirst', true);
  1575. allRepos.push(group);
  1576. }
  1577. group.repositories.push(repo);
  1578. });
  1579. }, this);
  1580. allRepos.stackVersion = App.get('currentStackVersionNumber');
  1581. this.set('allRepos', allRepos);
  1582. },
  1583. loadRepositoriesErrorCallback: function (request, ajaxOptions, error) {
  1584. },
  1585. /**
  1586. * @returns {$.ajax}
  1587. */
  1588. suspendUpgrade: function () {
  1589. var self = this;
  1590. return this.abortUpgradeWithSuspend().done(function () {
  1591. App.set('upgradeState', 'ABORTED');
  1592. self.setDBProperty('upgradeState', 'ABORTED');
  1593. App.clusterStatus.setClusterStatus({
  1594. wizardControllerName: self.get('name'),
  1595. localdb: App.db.data
  1596. });
  1597. });
  1598. },
  1599. /**
  1600. * @returns {$.ajax}
  1601. */
  1602. resumeUpgrade: function() {
  1603. var self = this;
  1604. this.retryUpgrade().done(function () {
  1605. App.set('upgradeState', 'PENDING');
  1606. App.propertyDidChange('upgradeSuspended');
  1607. self.setDBProperty('upgradeState', 'PENDING');
  1608. App.clusterStatus.setClusterStatus({
  1609. wizardControllerName: self.get('name'),
  1610. localdb: App.db.data
  1611. });
  1612. });
  1613. },
  1614. /**
  1615. * restore last Upgrade data
  1616. * @param {object} lastUpgradeData
  1617. */
  1618. restoreLastUpgrade: function(lastUpgradeData) {
  1619. var self = this;
  1620. var upgradeType = this.get('upgradeMethods').findProperty('type', lastUpgradeData.Upgrade.upgrade_type);
  1621. this.setDBProperties({
  1622. upgradeId: lastUpgradeData.Upgrade.request_id,
  1623. isDowngrade: lastUpgradeData.Upgrade.direction === 'DOWNGRADE',
  1624. upgradeState: lastUpgradeData.Upgrade.request_status,
  1625. upgradeType: lastUpgradeData.Upgrade.upgrade_type,
  1626. downgradeAllowed: lastUpgradeData.Upgrade.downgrade_allowed,
  1627. upgradeTypeDisplayName: upgradeType.get('displayName'),
  1628. failuresTolerance: Em.Object.create({
  1629. skipComponentFailures: lastUpgradeData.Upgrade.skip_failures,
  1630. skipSCFailures: lastUpgradeData.Upgrade.skip_service_check_failures
  1631. })
  1632. });
  1633. this.loadRepoVersionsToModel().done(function () {
  1634. var toVersion = App.RepositoryVersion.find().findProperty('repositoryVersion', lastUpgradeData.Upgrade.to_version);
  1635. self.setDBProperty('upgradeVersion', toVersion && toVersion.get('displayName'));
  1636. self.initDBProperties();
  1637. self.loadUpgradeData(true);
  1638. });
  1639. },
  1640. /**
  1641. * Build table from configs list and open new window to show this table
  1642. * @param configs
  1643. */
  1644. openConfigsInNewWindow: function (configs) {
  1645. var newWindow;
  1646. var output = '';
  1647. output += '<table style="text-align: left;"><thead><tr>' +
  1648. '<th>' + Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.configType') + '</th>' +
  1649. '<th>' + Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.propertyName') + '</th>' +
  1650. '<th>' + Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.currentValue') + '</th>' +
  1651. '<th>' + Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.recommendedValue') + '</th>' +
  1652. '<th>' + Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.resultingValue') + '</th>' +
  1653. '</tr></thead><tbody>';
  1654. configs.context.forEach(function (config) {
  1655. output += '<tr>' +
  1656. '<td>' + config.type + '</td>' +
  1657. '<td>' + config.name + '</td>' +
  1658. '<td>' + config.currentValue + '</td>' +
  1659. '<td>' + config.recommendedValue + '</td>' +
  1660. '<td>' + config.resultingValue + '</td>' +
  1661. '</tr>';
  1662. });
  1663. output += '</tbody></table>';
  1664. newWindow = window.open();
  1665. newWindow.document.write(output);
  1666. newWindow.focus();
  1667. }
  1668. });