stack_and_upgrade_controller.js 28 KB

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