stack_and_upgrade_controller.js 57 KB

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