stack_and_upgrade_controller.js 73 KB

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