stack_and_upgrade_controller.js 27 KB

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