stack_and_upgrade_controller.js 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991
  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 {boolean}
  43. * @default false
  44. */
  45. isDowngrade: false,
  46. /**
  47. * version that currently applied to server
  48. * should be plain object, because stored to localStorage
  49. * @type {object|null}
  50. */
  51. currentVersion: null,
  52. /**
  53. * versions to which cluster could be upgraded
  54. * @type {Array}
  55. */
  56. targetVersions: [],
  57. /**
  58. * @type {boolean} true if some request that should disable actions is in progress
  59. */
  60. requestInProgress: false,
  61. /**
  62. * @type {boolean} true while no updated upgrade info is loaded after retry
  63. */
  64. isRetryPending: false,
  65. /**
  66. * properties that stored to localStorage to resume wizard progress
  67. */
  68. wizardStorageProperties: ['upgradeId', 'upgradeVersion', 'currentVersion', 'isDowngrade'],
  69. /**
  70. * mutable properties of Upgrade Task
  71. * @type {Array}
  72. */
  73. taskDetailsProperties: ['status', 'stdout', 'stderr', 'error_log', 'host_name', 'output_log'],
  74. /**
  75. * Context for Finalize item
  76. * @type {string}
  77. */
  78. finalizeContext: 'Confirm Finalize',
  79. /**
  80. * Check if current item is Finalize
  81. * @type {boolean}
  82. */
  83. isFinalizeItem: false,
  84. isLoadUpgradeDataPending: false,
  85. /**
  86. * path to the mock json
  87. * @type {String}
  88. */
  89. mockRepoUrl: '/data/stack_versions/repo_versions_all.json',
  90. /**
  91. * api to get RepoVersions
  92. * @type {String}
  93. */
  94. realRepoUrl: function () {
  95. return App.get('apiPrefix') + App.get('stackVersionURL') +
  96. '/compatible_repository_versions?fields=*,operating_systems/*,operating_systems/repositories/*';
  97. }.property('App.stackVersionURL'),
  98. /**
  99. * path to the mock json
  100. * @type {String}
  101. */
  102. mockStackUrl: '/data/stack_versions/stack_version_all.json',
  103. /**
  104. * api to get ClusterStackVersions with repository_versions (use to init data load)
  105. * @type {String}
  106. */
  107. realStackUrl: function () {
  108. return App.get('apiPrefix') + '/clusters/' + App.get('clusterName') +
  109. '/stack_versions?fields=*,repository_versions/*,repository_versions/operating_systems/repositories/*';
  110. }.property('App.clusterName'),
  111. /**
  112. * api to get ClusterStackVersions without repository_versions (use to update data)
  113. * @type {String}
  114. */
  115. realUpdateUrl: function () {
  116. return App.get('apiPrefix') + '/clusters/' + App.get('clusterName') + '/stack_versions?fields=ClusterStackVersions/*';
  117. }.property('App.clusterName'),
  118. /**
  119. * Determines if list of services with checks that failed and were skipped by user during the upgrade is loaded
  120. * @type {boolean}
  121. */
  122. areSkippedServiceChecksLoaded: false,
  123. /**
  124. * List of services with checks that failed and were skipped by user during the upgrade
  125. * @type {array}
  126. */
  127. skippedServiceChecks: [],
  128. /**
  129. * status of tasks/items/groups which should be grayed out and disabled
  130. * @type {Array}
  131. */
  132. nonActiveStates: ['PENDING', 'ABORTED'],
  133. init: function () {
  134. this.initDBProperties();
  135. },
  136. /**
  137. * restore data from localStorage
  138. */
  139. initDBProperties: function () {
  140. var props = this.getDBProperties(this.get('wizardStorageProperties'));
  141. Em.keys(props).forEach(function (k) {
  142. if (props[k]) {
  143. this.set(k, props[k]);
  144. }
  145. }, this);
  146. },
  147. /**
  148. * load all data:
  149. * - upgrade data
  150. * - stack versions
  151. * - repo versions
  152. */
  153. load: function () {
  154. var dfd = $.Deferred();
  155. var self = this;
  156. this.loadUpgradeData(true).done(function() {
  157. self.loadStackVersionsToModel(true).done(function () {
  158. self.loadRepoVersionsToModel().done(function() {
  159. var currentVersion = App.StackVersion.find().findProperty('state', 'CURRENT');
  160. if (currentVersion) {
  161. self.set('currentVersion', {
  162. repository_version: currentVersion.get('repositoryVersion.repositoryVersion'),
  163. repository_name: currentVersion.get('repositoryVersion.displayName')
  164. });
  165. }
  166. dfd.resolve();
  167. });
  168. });
  169. });
  170. return dfd.promise();
  171. },
  172. /**
  173. * load upgrade tasks by upgrade id
  174. * @return {$.Deferred}
  175. * @param {boolean} onlyState
  176. */
  177. loadUpgradeData: function (onlyState) {
  178. var upgradeId = this.get('upgradeId'),
  179. deferred = $.Deferred(),
  180. self = this;
  181. if (Em.isNone(upgradeId)) {
  182. deferred.resolve();
  183. console.log('Upgrade in INIT state');
  184. } else {
  185. this.set('isLoadUpgradeDataPending', true);
  186. App.ajax.send({
  187. name: (onlyState) ? 'admin.upgrade.state' : 'admin.upgrade.data',
  188. sender: this,
  189. data: {
  190. id: upgradeId
  191. },
  192. success: 'loadUpgradeDataSuccessCallback'
  193. }).then(deferred.resolve).complete(function () {
  194. self.set('isLoadUpgradeDataPending', false);
  195. });
  196. }
  197. return deferred.promise();
  198. },
  199. /**
  200. * parse and push upgrade tasks to controller
  201. * @param data
  202. */
  203. loadUpgradeDataSuccessCallback: function (data) {
  204. if (Em.isNone(data)) return;
  205. App.set('upgradeState', data.Upgrade.request_status);
  206. this.setDBProperty('upgradeState', data.Upgrade.request_status);
  207. if (data.upgrade_groups) {
  208. this.updateUpgradeData(data);
  209. }
  210. if (this.get('isRetryPending') && data.Upgrade.request_status != 'ABORTED') {
  211. this.setProperties({
  212. requestInProgress: false,
  213. isRetryPending: false
  214. });
  215. }
  216. },
  217. /**
  218. * update data of Upgrade
  219. * @param {object} newData
  220. */
  221. updateUpgradeData: function (newData) {
  222. var oldData = this.get('upgradeData'),
  223. nonActiveStates = this.get('nonActiveStates'),
  224. groupsMap = {},
  225. itemsMap = {};
  226. if (Em.isNone(oldData) || (newData.upgrade_groups.length !== oldData.upgradeGroups.length)) {
  227. this.initUpgradeData(newData);
  228. } else {
  229. //create entities maps
  230. newData.upgrade_groups.forEach(function (newGroup) {
  231. groupsMap[newGroup.UpgradeGroup.group_id] = newGroup.UpgradeGroup;
  232. newGroup.upgrade_items.forEach(function (item) {
  233. itemsMap[item.UpgradeItem.stage_id] = item.UpgradeItem;
  234. })
  235. });
  236. //update existed entities with new data
  237. oldData.upgradeGroups.forEach(function (oldGroup) {
  238. oldGroup.set('status', groupsMap[oldGroup.get('group_id')].status);
  239. oldGroup.set('progress_percent', groupsMap[oldGroup.get('group_id')].progress_percent);
  240. oldGroup.set('completed_task_count', groupsMap[oldGroup.get('group_id')].completed_task_count);
  241. oldGroup.upgradeItems.forEach(function (item) {
  242. item.set('status', itemsMap[item.get('stage_id')].status);
  243. item.set('progress_percent', itemsMap[item.get('stage_id')].progress_percent);
  244. });
  245. var hasExpandableItems = oldGroup.upgradeItems.some(function (item) {
  246. return !nonActiveStates.contains(item.get('status'));
  247. });
  248. oldGroup.set('hasExpandableItems', hasExpandableItems);
  249. });
  250. oldData.set('Upgrade', newData.Upgrade);
  251. }
  252. },
  253. /**
  254. * change structure of Upgrade
  255. * In order to maintain nested views in template object should have direct link to its properties, for example
  256. * item.UpgradeItem.<properties> -> item.<properties>
  257. * @param {object} newData
  258. */
  259. initUpgradeData: function (newData) {
  260. var upgradeGroups = [],
  261. nonActiveStates = this.get('nonActiveStates');
  262. //wrap all entities into App.upgradeEntity
  263. newData.upgrade_groups.forEach(function (newGroup) {
  264. var hasExpandableItems = newGroup.upgrade_items.some(function (item) {
  265. return !nonActiveStates.contains(item.UpgradeItem.status);
  266. }),
  267. oldGroup = App.upgradeEntity.create({type: 'GROUP', hasExpandableItems: hasExpandableItems}, newGroup.UpgradeGroup),
  268. upgradeItems = [];
  269. newGroup.upgrade_items.forEach(function (item) {
  270. var oldItem = App.upgradeEntity.create({type: 'ITEM'}, item.UpgradeItem);
  271. oldItem.set('tasks', []);
  272. upgradeItems.pushObject(oldItem);
  273. });
  274. upgradeItems.reverse();
  275. oldGroup.set('upgradeItems', upgradeItems);
  276. upgradeGroups.pushObject(oldGroup);
  277. });
  278. upgradeGroups.reverse();
  279. this.set('upgradeData', Em.Object.create({
  280. upgradeGroups: upgradeGroups,
  281. Upgrade: newData.Upgrade
  282. }));
  283. },
  284. /**
  285. * request Upgrade Item and its tasks from server
  286. * @return {$.ajax}
  287. */
  288. getUpgradeItem: function (item) {
  289. return App.ajax.send({
  290. name: 'admin.upgrade.upgrade_item',
  291. sender: this,
  292. data: {
  293. upgradeId: item.get('request_id'),
  294. groupId: item.get('group_id'),
  295. stageId: item.get('stage_id')
  296. },
  297. success: 'getUpgradeItemSuccessCallback'
  298. });
  299. },
  300. /**
  301. * success callback of <code>getTasks</code>
  302. * @param {object} data
  303. */
  304. getUpgradeItemSuccessCallback: function (data) {
  305. this.get('upgradeData.upgradeGroups').forEach(function (group) {
  306. if (group.get('group_id') === data.UpgradeItem.group_id) {
  307. group.get('upgradeItems').forEach(function (item) {
  308. if (item.get('stage_id') === data.UpgradeItem.stage_id) {
  309. if (item.get('tasks.length')) {
  310. item.set('isTasksLoaded', true);
  311. data.tasks.forEach(function (task) {
  312. var currentTask = item.get('tasks').findProperty('id', task.Tasks.id);
  313. this.get('taskDetailsProperties').forEach(function (property) {
  314. currentTask.set(property, task.Tasks[property]);
  315. }, this);
  316. }, this);
  317. } else {
  318. var tasks = [];
  319. data.tasks.forEach(function (task) {
  320. tasks.pushObject(App.upgradeEntity.create({type: 'TASK'}, task.Tasks));
  321. });
  322. item.set('tasks', tasks);
  323. }
  324. item.set('isTasksLoaded', true);
  325. }
  326. }, this);
  327. }
  328. }, this);
  329. },
  330. /**
  331. * downgrade confirmation popup
  332. * @param {object} event
  333. */
  334. confirmDowngrade: function (event) {
  335. var self = this;
  336. var currentVersion = this.get('currentVersion');
  337. return App.showConfirmationPopup(
  338. function() {
  339. self.downgrade.call(self, currentVersion, event);
  340. },
  341. Em.I18n.t('admin.stackUpgrade.downgrade.body').format(currentVersion.repository_name),
  342. null,
  343. Em.I18n.t('admin.stackUpgrade.dialog.downgrade.header').format(currentVersion.repository_name),
  344. Em.I18n.t('admin.stackUpgrade.downgrade.proceed')
  345. );
  346. },
  347. /**
  348. * make call to start downgrade process
  349. * @param {object} currentVersion
  350. * @param {object} event
  351. */
  352. downgrade: function (currentVersion, event) {
  353. this.set('requestInProgress', true);
  354. this.abortUpgrade();
  355. App.ajax.send({
  356. name: 'admin.downgrade.start',
  357. sender: this,
  358. data: {
  359. from: App.RepositoryVersion.find().findProperty('displayName', this.get('upgradeVersion')).get('repositoryVersion'),
  360. value: currentVersion.repository_version,
  361. label: currentVersion.repository_name,
  362. isDowngrade: true
  363. },
  364. success: 'upgradeSuccessCallback',
  365. callback: function() {
  366. this.sender.set('requestInProgress', false);
  367. }
  368. });
  369. },
  370. /**
  371. * abort upgrade (in order to start Downgrade)
  372. */
  373. abortUpgrade: function () {
  374. return App.ajax.send({
  375. name: 'admin.upgrade.abort',
  376. sender: this,
  377. data: {
  378. upgradeId: this.get('upgradeId')
  379. }
  380. });
  381. },
  382. retryUpgrade: function () {
  383. this.setProperties({
  384. requestInProgress: true,
  385. isRetryPending: true
  386. });
  387. return App.ajax.send({
  388. name: 'admin.upgrade.retry',
  389. sender: this,
  390. data: {
  391. upgradeId: this.get('upgradeId')
  392. }
  393. });
  394. },
  395. /**
  396. * make call to start upgrade process and show popup with current progress
  397. * @param {object} version
  398. */
  399. upgrade: function (version) {
  400. this.set('requestInProgress', true);
  401. App.ajax.send({
  402. name: 'admin.upgrade.start',
  403. sender: this,
  404. data: version,
  405. success: 'upgradeSuccessCallback',
  406. callback: function() {
  407. this.sender.set('requestInProgress', false);
  408. }
  409. });
  410. this.setDBProperty('currentVersion', this.get('currentVersion'));
  411. },
  412. /**
  413. * success callback of <code>upgrade()</code>
  414. * @param {object} data
  415. */
  416. upgradeSuccessCallback: function (data, opt, params) {
  417. this.set('upgradeData', null);
  418. this.set('upgradeId', data.resources[0].Upgrade.request_id);
  419. this.set('upgradeVersion', params.label);
  420. this.set('isDowngrade', !!params.isDowngrade);
  421. this.setDBProperties({
  422. upgradeVersion: params.label,
  423. upgradeId: data.resources[0].Upgrade.request_id,
  424. upgradeState: 'PENDING',
  425. isDowngrade: !!params.isDowngrade
  426. });
  427. App.set('upgradeState', 'PENDING');
  428. App.clusterStatus.setClusterStatus({
  429. wizardControllerName: this.get('name'),
  430. localdb: App.db.data
  431. });
  432. this.load();
  433. this.openUpgradeDialog();
  434. },
  435. /**
  436. * upgrade confirmation popup
  437. * @param {object} version
  438. * @return App.ModalPopup
  439. */
  440. confirmUpgrade: function (version) {
  441. var self = this;
  442. return App.showConfirmationPopup(
  443. function () {
  444. self.runPreUpgradeCheck.call(self, version);
  445. },
  446. Em.I18n.t('admin.stackUpgrade.upgrade.confirm.body').format(version.get('displayName')),
  447. null,
  448. Em.I18n.t('admin.stackUpgrade.dialog.header').format(version.get('displayName'))
  449. );
  450. },
  451. /**
  452. * send request for pre upgrade check
  453. * @param version
  454. */
  455. runPreUpgradeCheck: function(version) {
  456. var params = {
  457. value: version.get('repositoryVersion'),
  458. label: version.get('displayName')
  459. };
  460. if (App.get('supports.preUpgradeCheck')) {
  461. this.set('requestInProgress', true);
  462. App.ajax.send({
  463. name: "admin.rolling_upgrade.pre_upgrade_check",
  464. sender: this,
  465. data: params,
  466. success: "runPreUpgradeCheckSuccess",
  467. error: "runPreUpgradeCheckError"
  468. });
  469. } else {
  470. this.upgrade(params);
  471. }
  472. },
  473. /**
  474. * success callback of <code>runPreUpgradeCheckSuccess()</code>
  475. * if there are some fails - it shows popup else run upgrade
  476. * @param data {object}
  477. * @param opt {object}
  478. * @param params {object}
  479. * @returns {App.ModalPopup|undefined}
  480. */
  481. runPreUpgradeCheckSuccess: function (data, opt, params) {
  482. var self = this;
  483. if (data.items.someProperty('UpgradeChecks.status', 'FAIL') || data.items.someProperty('UpgradeChecks.status', 'WARNING')) {
  484. this.set('requestInProgress', false);
  485. var header = Em.I18n.t('popup.clusterCheck.Upgrade.header').format(params.label),
  486. failTitle = Em.I18n.t('popup.clusterCheck.Upgrade.fail.title'),
  487. failAlert = new Em.Handlebars.SafeString(Em.I18n.t('popup.clusterCheck.Upgrade.fail.alert')),
  488. warningTitle = Em.I18n.t('popup.clusterCheck.Upgrade.warning.title'),
  489. warningAlert = new Em.Handlebars.SafeString(Em.I18n.t('popup.clusterCheck.Upgrade.warning.alert')),
  490. configsMergeWarning = data.items.findProperty('UpgradeChecks.id', "CONFIG_MERGE"),
  491. configs = [];
  492. if (configsMergeWarning && Em.get(configsMergeWarning, 'UpgradeChecks.status') === 'WARNING') {
  493. data.items = data.items.rejectProperty('UpgradeChecks.id', 'CONFIG_MERGE');
  494. var configsMergeCheckData = Em.get(configsMergeWarning, 'UpgradeChecks.failed_detail');
  495. if (configsMergeCheckData) {
  496. configs = configsMergeCheckData.map(function (item) {
  497. var isDeprecated = Em.isNone(item.new_stack_value),
  498. willBeRemoved = Em.isNone(item.result_value);
  499. return {
  500. type: item.type,
  501. name: item.property,
  502. currentValue: item.current,
  503. recommendedValue: isDeprecated ? Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.deprecated') : item.new_stack_value,
  504. isDeprecated: isDeprecated,
  505. resultingValue: willBeRemoved ? Em.I18n.t('popup.clusterCheck.Upgrade.configsMerge.willBeRemoved') : item.result_value,
  506. willBeRemoved: willBeRemoved
  507. };
  508. });
  509. }
  510. }
  511. App.showClusterCheckPopup(data, header, failTitle, failAlert, warningTitle, warningAlert, function () {
  512. self.upgrade(params);
  513. }, configs, params.label);
  514. } else {
  515. this.upgrade(params);
  516. }
  517. },
  518. runPreUpgradeCheckError: function() {
  519. this.set('requestInProgress', false);
  520. },
  521. confirmRetryUpgrade: function (version) {
  522. var self = this;
  523. return App.showConfirmationPopup(
  524. function () {
  525. self.retryUpgrade();
  526. },
  527. Em.I18n.t('admin.stackUpgrade.upgrade.retry.confirm.body').format(version.get('displayName')),
  528. null,
  529. Em.I18n.t('admin.stackUpgrade.dialog.header').format(version.get('displayName'))
  530. );
  531. },
  532. confirmRetryDowngrade: function () {
  533. var self = this,
  534. currentVersion = this.get('currentVersion');
  535. return App.showConfirmationPopup(
  536. function() {
  537. self.retryUpgrade();
  538. },
  539. Em.I18n.t('admin.stackUpgrade.downgrade.retry.body').format(currentVersion.repository_name),
  540. null,
  541. Em.I18n.t('admin.stackUpgrade.dialog.downgrade.header').format(currentVersion.repository_name),
  542. Em.I18n.t('admin.stackUpgrade.downgrade.proceed')
  543. );
  544. },
  545. /**
  546. * confirmation popup before install repository version
  547. */
  548. installRepoVersionConfirmation: function (repo) {
  549. var self = this;
  550. return App.showConfirmationPopup(function () {
  551. self.installRepoVersion(repo);
  552. },
  553. Em.I18n.t('admin.stackVersions.version.install.confirm').format(repo.get('displayName'))
  554. );
  555. },
  556. /**
  557. * sends request to install repoVersion to the cluster
  558. * and create clusterStackVersion resourse
  559. * @param {Em.Object} repo
  560. * @return {$.ajax}
  561. * @method installRepoVersion
  562. */
  563. installRepoVersion: function (repo) {
  564. this.set('requestInProgress', true);
  565. var data = {
  566. ClusterStackVersions: {
  567. stack: repo.get('stackVersionType'),
  568. version: repo.get('stackVersionNumber'),
  569. repository_version: repo.get('repositoryVersion')
  570. },
  571. id: repo.get('id')
  572. };
  573. return App.ajax.send({
  574. name: 'admin.stack_version.install.repo_version',
  575. sender: this,
  576. data: data,
  577. success: 'installRepoVersionSuccess',
  578. callback: function() {
  579. this.sender.set('requestInProgress', false);
  580. }
  581. });
  582. },
  583. /**
  584. * transform repo data into json for
  585. * saving changes to repository version
  586. * @param {Em.Object} repo
  587. * @returns {{operating_systems: Array}}
  588. */
  589. prepareRepoForSaving: function(repo) {
  590. var repoVersion = { "operating_systems": [] };
  591. repo.get('operatingSystems').forEach(function (os, k) {
  592. repoVersion.operating_systems.push({
  593. "OperatingSystems": {
  594. "os_type": os.get("osType")
  595. },
  596. "repositories": []
  597. });
  598. os.get('repositories').forEach(function (repository) {
  599. repoVersion.operating_systems[k].repositories.push({
  600. "Repositories": {
  601. "base_url": repository.get('baseUrl'),
  602. "repo_id": repository.get('repoId'),
  603. "repo_name": repository.get('repoName')
  604. }
  605. });
  606. });
  607. });
  608. return repoVersion;
  609. },
  610. /**
  611. * Return stack version for the repo object
  612. * @param {Em.Object} repo
  613. * */
  614. getStackVersionNumber: function(repo){
  615. var stackVersionNumber = repo.get('stackVersion');
  616. if(null == stackVersionNumber)
  617. stackVersionNumber = App.get('currentStackVersion');
  618. return stackVersionNumber;
  619. },
  620. /**
  621. * perform validation if <code>skip<code> is false and run save if
  622. * validation successfull or run save without validation is <code>skip<code> is true
  623. * @param {Em.Object} repo
  624. * @param {boolean} skip
  625. * @returns {$.Deferred}
  626. */
  627. saveRepoOS: function (repo, skip) {
  628. var self = this;
  629. var deferred = $.Deferred();
  630. this.validateRepoVersions(repo, skip).done(function(data) {
  631. if (data.length > 0) {
  632. deferred.resolve(data);
  633. } else {
  634. var repoVersion = self.prepareRepoForSaving(repo);
  635. var stackVersionNumber = self.getStackVersionNumber(repo);
  636. console.log("Repository stack version:"+stackVersionNumber);
  637. App.ajax.send({
  638. name: 'admin.stack_versions.edit.repo',
  639. sender: this,
  640. data: {
  641. stackName: App.get('currentStackName'),
  642. stackVersion: stackVersionNumber,
  643. repoVersionId: repo.get('repoVersionId'),
  644. repoVersion: repoVersion
  645. }
  646. }).success(function() {
  647. deferred.resolve([]);
  648. });
  649. }
  650. });
  651. return deferred.promise();
  652. },
  653. /**
  654. * send request for validation for each repository
  655. * @param {Em.Object} repo
  656. * @param {boolean} skip
  657. * @returns {*}
  658. */
  659. validateRepoVersions: function(repo, skip) {
  660. var deferred = $.Deferred(),
  661. totalCalls = 0,
  662. invalidUrls = [];
  663. if (skip) {
  664. deferred.resolve(invalidUrls);
  665. } else {
  666. var stackVersionNumber = this.getStackVersionNumber(repo);
  667. repo.get('operatingSystems').forEach(function (os) {
  668. if (os.get('isSelected')) {
  669. os.get('repositories').forEach(function (repo) {
  670. totalCalls++;
  671. App.ajax.send({
  672. name: 'admin.stack_versions.validate.repo',
  673. sender: this,
  674. data: {
  675. repo: repo,
  676. repoId: repo.get('repoId'),
  677. baseUrl: repo.get('baseUrl'),
  678. osType: os.get('osType'),
  679. stackName: App.get('currentStackName'),
  680. stackVersion: stackVersionNumber
  681. }
  682. })
  683. .success(function () {
  684. totalCalls--;
  685. if (totalCalls === 0) deferred.resolve(invalidUrls);
  686. })
  687. .error(function () {
  688. repo.set('hasError', true);
  689. invalidUrls.push(repo);
  690. totalCalls--;
  691. if (totalCalls === 0) deferred.resolve(invalidUrls);
  692. });
  693. });
  694. } else {
  695. return deferred.resolve(invalidUrls);
  696. }
  697. });
  698. }
  699. return deferred.promise();
  700. },
  701. /**
  702. * success callback for <code>installRepoVersion()<code>
  703. * saves request id to the db
  704. * @param data
  705. * @param opt
  706. * @param params
  707. * @method installStackVersionSuccess
  708. */
  709. installRepoVersionSuccess: function (data, opt, params) {
  710. var version = App.RepositoryVersion.find(params.id);
  711. App.db.set('repoVersionInstall', 'id', [data.Requests.id]);
  712. App.clusterStatus.setClusterStatus({
  713. wizardControllerName: this.get('name'),
  714. localdb: App.db.data
  715. });
  716. version.set('defaultStatus', 'INSTALLING');
  717. if (version.get('stackVersion')) {
  718. version.set('stackVersion.state', 'INSTALLING');
  719. }
  720. },
  721. /**
  722. * opens a popup with installations state per host
  723. * @param {Em.Object} version
  724. * @method showProgressPopup
  725. */
  726. showProgressPopup: function(version) {
  727. var popupTitle = Em.I18n.t('admin.stackVersions.details.install.hosts.popup.title').format(version.get('displayName'));
  728. var requestIds = App.get('testMode') ? [1] : App.db.get('repoVersionInstall', 'id');
  729. var hostProgressPopupController = App.router.get('highAvailabilityProgressPopupController');
  730. hostProgressPopupController.initPopup(popupTitle, requestIds, this);
  731. },
  732. /**
  733. * reset upgradeState to INIT when upgrade is COMPLETED
  734. * and clean auxiliary data
  735. */
  736. finish: function () {
  737. if (App.get('upgradeState') === 'COMPLETED') {
  738. this.setDBProperties({
  739. upgradeId: undefined,
  740. upgradeState: 'INIT',
  741. upgradeVersion: undefined,
  742. currentVersion: undefined,
  743. isDowngrade: undefined
  744. });
  745. App.clusterStatus.setClusterStatus({
  746. localdb: App.db.data
  747. });
  748. App.set('upgradeState', 'INIT');
  749. }
  750. }.observes('App.upgradeState'),
  751. /**
  752. * Check <code>App.upgradeState</code> for HOLDING
  753. * If it is, send request to check if current item is Finalize
  754. * @method updateFinalize
  755. */
  756. updateFinalize: function () {
  757. var upgradeState = App.get('upgradeState');
  758. if (upgradeState === 'HOLDING') {
  759. return App.ajax.send({
  760. name: 'admin.upgrade.finalizeContext',
  761. sender: this,
  762. success: 'updateFinalizeSuccessCallback',
  763. error: 'updateFinalizeErrorCallback'
  764. })
  765. }
  766. else {
  767. this.set('isFinalizeItem', false);
  768. }
  769. }.observes('App.upgradeState'),
  770. /**
  771. *
  772. * @param {object|null} data
  773. * @method updateFinalizeSuccessCallback
  774. */
  775. updateFinalizeSuccessCallback: function (data) {
  776. var context = data ? Em.get(data, 'upgrade_groups.firstObject.upgrade_items.firstObject.UpgradeItem.context') : '';
  777. this.set('isFinalizeItem', context === this.get('finalizeContext'));
  778. },
  779. updateFinalizeErrorCallback: function() {
  780. this.set('isFinalizeItem', false);
  781. },
  782. /**
  783. * show dialog with tasks of upgrade
  784. * @return {App.ModalPopup}
  785. */
  786. openUpgradeDialog: function () {
  787. App.router.transitionTo('admin.stackUpgrade');
  788. },
  789. /**
  790. * returns url to get data for repoVersion or clusterStackVersion
  791. * @param {Boolean} stack true if load clusterStackVersion
  792. * @param {Boolean} fullLoad true if load all data
  793. * @returns {String}
  794. * @method getUrl
  795. */
  796. getUrl: function(stack, fullLoad) {
  797. if (App.get('testMode')) {
  798. return stack ? this.get('mockStackUrl') : this.get('mockRepoUrl')
  799. } else {
  800. if (fullLoad) {
  801. return stack ? this.get('realStackUrl') : this.get('realRepoUrl');
  802. } else {
  803. return this.get('realUpdateUrl');
  804. }
  805. }
  806. },
  807. /**
  808. * get stack versions from server and push it to model
  809. * @return {*}
  810. * @method loadStackVersionsToModel
  811. */
  812. loadStackVersionsToModel: function (fullLoad) {
  813. var dfd = $.Deferred();
  814. App.HttpClient.get(this.getUrl(true, fullLoad), App.stackVersionMapper, {
  815. complete: function () {
  816. dfd.resolve();
  817. }
  818. });
  819. return dfd.promise();
  820. },
  821. /**
  822. * get repo versions from server and push it to model
  823. * @return {*}
  824. * @params {Boolean} isUpdate - if true loads part of data that need to be updated
  825. * @method loadRepoVersionsToModel()
  826. */
  827. loadRepoVersionsToModel: function () {
  828. var dfd = $.Deferred();
  829. App.HttpClient.get(this.getUrl(false, true), App.repoVersionMapper, {
  830. complete: function () {
  831. dfd.resolve();
  832. }
  833. });
  834. return dfd.promise();
  835. },
  836. /**
  837. * set status to Upgrade item
  838. * @param item
  839. * @param status
  840. */
  841. setUpgradeItemStatus: function(item, status) {
  842. this.set('requestInProgress', true);
  843. return App.ajax.send({
  844. name: 'admin.upgrade.upgradeItem.setState',
  845. sender: this,
  846. data: {
  847. upgradeId: item.get('request_id'),
  848. itemId: item.get('stage_id'),
  849. groupId: item.get('group_id'),
  850. status: status
  851. },
  852. callback: function() {
  853. this.sender.set('requestInProgress', false);
  854. }
  855. }).done(function () {
  856. item.set('status', status);
  857. });
  858. },
  859. currentVersionObserver: function () {
  860. var versionNumber = this.get('currentVersion.repository_version');
  861. var currentVersionObject = App.RepositoryVersion.find().findProperty('status', 'CURRENT');
  862. var versionName = currentVersionObject && currentVersionObject.get('stackVersionType');
  863. App.set('isStormMetricsSupported', versionName != 'HDP' || stringUtils.compareVersions(versionNumber, '2.2.2') > -1 || !versionNumber);
  864. }.observes('currentVersion.repository_version'),
  865. /**
  866. * get the installed repositories of HDP from server
  867. */
  868. loadRepositories: function () {
  869. if (App.router.get('clusterController.isLoaded')) {
  870. var nameVersionCombo = App.get('currentStackVersion');
  871. var stackName = nameVersionCombo.split('-')[0];
  872. var stackVersion = nameVersionCombo.split('-')[1];
  873. App.ajax.send({
  874. name: 'cluster.load_repositories',
  875. sender: this,
  876. data: {
  877. stackName: stackName,
  878. stackVersion: stackVersion
  879. },
  880. success: 'loadRepositoriesSuccessCallback',
  881. error: 'loadRepositoriesErrorCallback'
  882. });
  883. }
  884. }.observes('App.router.clusterController.isLoaded'),
  885. loadRepositoriesSuccessCallback: function (data) {
  886. var allRepos = [];
  887. data.items.forEach(function (os) {
  888. os.repositories.forEach(function (repository) {
  889. var osType = repository.Repositories.os_type;
  890. var repo = Em.Object.create({
  891. baseUrl: repository.Repositories.base_url,
  892. osType: osType,
  893. repoId: repository.Repositories.repo_id,
  894. repoName : repository.Repositories.repo_name,
  895. stackName : repository.Repositories.stack_name,
  896. stackVersion : repository.Repositories.stack_version,
  897. isFirst: false
  898. });
  899. var group = allRepos.findProperty('name', osType);
  900. if (!group) {
  901. group = {
  902. name: osType,
  903. repositories: []
  904. };
  905. repo.set('isFirst', true);
  906. allRepos.push(group);
  907. }
  908. group.repositories.push(repo);
  909. });
  910. }, this);
  911. allRepos.stackVersion = App.get('currentStackVersionNumber');
  912. this.set('allRepos', allRepos);
  913. },
  914. loadRepositoriesErrorCallback: function (request, ajaxOptions, error) {
  915. console.log('Error message is: ' + request.responseText);
  916. }
  917. });