stack_and_upgrade_controller.js 53 KB

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