stack_and_upgrade_controller.js 52 KB

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