stack_and_upgrade_controller.js 57 KB

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