stack_and_upgrade_controller.js 58 KB

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