installer.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627
  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. App.InstallerController = App.WizardController.extend({
  20. name: 'installerController',
  21. totalSteps: 11,
  22. content: Em.Object.create({
  23. cluster: null,
  24. installOptions: null,
  25. hosts: null,
  26. services: null,
  27. slaveComponentHosts: null,
  28. masterComponentHosts: null,
  29. serviceConfigProperties: null,
  30. advancedServiceConfig: null,
  31. configGroups: [],
  32. slaveGroupProperties: null,
  33. installedHosts: [],
  34. stacks: null,
  35. controllerName: 'installerController'
  36. }),
  37. /**
  38. * Wizard properties in local storage, which should be cleaned right after wizard has been finished
  39. */
  40. dbPropertiesToClean: [
  41. 'service',
  42. 'hosts',
  43. 'masterComponentHosts',
  44. 'slaveComponentHosts',
  45. 'cluster',
  46. 'allHostNames',
  47. 'installOptions',
  48. 'allHostNamesPattern',
  49. 'serviceComponents',
  50. 'advancedServiceConfig',
  51. 'clientInfo',
  52. 'selectedServiceNames',
  53. 'serviceConfigGroups',
  54. 'serviceConfigProperties',
  55. 'configsToUpdate',
  56. 'bootStatus'
  57. ],
  58. init: function () {
  59. this._super();
  60. this.get('isStepDisabled').setEach('value', true);
  61. this.get('isStepDisabled').pushObject(Ember.Object.create({
  62. step: 0,
  63. value: false
  64. }));
  65. },
  66. /**
  67. * redefined connectOutlet method to avoid view loading by unauthorized user
  68. * @param view
  69. * @param content
  70. */
  71. connectOutlet: function(view, content) {
  72. if(App.db.getAuthenticated()) {
  73. this._super(view, content);
  74. }
  75. },
  76. getCluster: function(){
  77. return jQuery.extend({}, this.get('clusterStatusTemplate'));
  78. },
  79. getInstallOptions: function(){
  80. return jQuery.extend({}, this.get('installOptionsTemplate'));
  81. },
  82. getHosts: function(){
  83. return [];
  84. },
  85. /**
  86. * Remove host from model. Used at <code>Confirm hosts(step2)</code> step
  87. * @param hosts Array of hosts, which we want to delete
  88. */
  89. removeHosts: function (hosts) {
  90. var dbHosts = this.getDBProperty('hosts');
  91. hosts.forEach(function (_hostInfo) {
  92. var host = _hostInfo.hostName;
  93. delete dbHosts[host];
  94. });
  95. this.setDBProperty('hosts', dbHosts);
  96. },
  97. /**
  98. * Load confirmed hosts.
  99. * Will be used at <code>Assign Masters(step5)</code> step
  100. */
  101. loadConfirmedHosts: function () {
  102. this.set('content.hosts', this.getDBProperty('hosts') || []);
  103. },
  104. /**
  105. * Load services data. Will be used at <code>Select services(step4)</code> step
  106. */
  107. loadServices: function () {
  108. var servicesInfo = this.getDBProperty('service');
  109. if(servicesInfo && servicesInfo.length) {
  110. servicesInfo.forEach(function (item, index) {
  111. servicesInfo[index] = Em.Object.create(item);
  112. servicesInfo[index].isInstalled = false;
  113. });
  114. this.set('content.services', servicesInfo);
  115. console.log('installerController.loadServices: loaded data ', JSON.stringify(servicesInfo));
  116. console.log('selected services ', servicesInfo.filterProperty('isSelected', true).mapProperty('serviceName'));
  117. } else {
  118. console.log("Failed to load Services");
  119. }
  120. },
  121. /**
  122. * total set of hosts registered to cluster, analog of App.Host model,
  123. * used in Installer wizard until hosts are installed
  124. */
  125. allHosts: function () {
  126. var rawHosts = this.get('content.hosts');
  127. var masterComponents = this.get('content.masterComponentHosts');
  128. var slaveComponents = this.get('content.slaveComponentHosts');
  129. var hosts = [];
  130. masterComponents.forEach(function (component) {
  131. var host = rawHosts[component.hostName];
  132. if (host.hostComponents) {
  133. host.hostComponents.push(Em.Object.create({
  134. componentName: component.component,
  135. displayName: component.display_name
  136. }));
  137. } else {
  138. rawHosts[component.hostName].hostComponents = [
  139. Em.Object.create({
  140. componentName: component.component,
  141. displayName: component.display_name
  142. })
  143. ]
  144. }
  145. });
  146. slaveComponents.forEach(function (component) {
  147. component.hosts.forEach(function (rawHost) {
  148. var host = rawHosts[rawHost.hostName];
  149. if (host.hostComponents) {
  150. host.hostComponents.push(Em.Object.create({
  151. componentName: component.componentName,
  152. displayName: component.displayName
  153. }));
  154. } else {
  155. rawHosts[rawHost.hostName].hostComponents = [
  156. Em.Object.create({
  157. componentName: component.componentName,
  158. displayName: component.displayName
  159. })
  160. ]
  161. }
  162. });
  163. });
  164. for (var hostName in rawHosts) {
  165. var host = rawHosts[hostName];
  166. var disksOverallCapacity = 0;
  167. var diskFree = 0;
  168. host.disk_info.forEach(function (disk) {
  169. disksOverallCapacity += parseFloat(disk.size);
  170. diskFree += parseFloat(disk.available);
  171. });
  172. hosts.pushObject(Em.Object.create({
  173. id: host.name,
  174. ip: host.ip,
  175. osType: host.os_type,
  176. osArch: host.os_arch,
  177. hostName: host.name,
  178. publicHostName: host.name,
  179. cpu: host.cpu,
  180. memory: host.memory,
  181. diskInfo: host.disk_info,
  182. diskTotal: disksOverallCapacity / (1024 * 1024),
  183. diskFree: diskFree / (1024 * 1024),
  184. hostComponents: host.hostComponents
  185. }
  186. ))
  187. }
  188. return hosts;
  189. }.property('content.hosts'),
  190. /**
  191. * Load service components.
  192. */
  193. loadServiceComponentsDb: function () {
  194. var serviceComponents = this.getDBProperty('serviceComponents');
  195. if(serviceComponents && serviceComponents.items && serviceComponents.items.length) {
  196. App.stackServiceComponentMapper.map(serviceComponents);
  197. } else {
  198. console.log("Failed to load Service components");
  199. }
  200. },
  201. stacks: [],
  202. /**
  203. * Load stacks data from server or take exist data from local db
  204. */
  205. loadStacks: function () {
  206. var stacks = App.db.getStacks();
  207. if (stacks && stacks.length) {
  208. var convertedStacks = [];
  209. stacks.forEach(function (stack) {
  210. convertedStacks.pushObject(Ember.Object.create(stack));
  211. });
  212. App.set('currentStackVersion', convertedStacks.findProperty('isSelected').get('name'));
  213. this.set('content.stacks', convertedStacks);
  214. } else {
  215. App.ajax.send({
  216. name: 'wizard.stacks',
  217. sender: this,
  218. success: 'loadStacksSuccessCallback',
  219. error: 'loadStacksErrorCallback'
  220. });
  221. }
  222. },
  223. /**
  224. * Send queries to load versions for each stack
  225. */
  226. loadStacksSuccessCallback: function (data) {
  227. var stacks = data.items;
  228. var result;
  229. this.get('stacks').clear();
  230. stacks.forEach(function (stack) {
  231. App.ajax.send({
  232. name: 'wizard.stacks_versions',
  233. sender: this,
  234. data: {
  235. stackName: stack.Stacks.stack_name
  236. },
  237. success: 'loadStacksVersionsSuccessCallback',
  238. error: 'loadStacksVersionsErrorCallback'
  239. });
  240. }, this);
  241. result = this.get('stacks');
  242. if (!result.length) {
  243. console.log('Error: therea are no active stacks');
  244. } else {
  245. var defaultStackVersion = result.findProperty('name', App.defaultStackVersion);
  246. if (defaultStackVersion) {
  247. defaultStackVersion.set('isSelected', true)
  248. } else {
  249. result.objectAt(0).set('isSelected', true);
  250. }
  251. }
  252. App.db.setStacks(result);
  253. this.set('content.stacks', result);
  254. },
  255. /**
  256. * onError callback for loading stacks data
  257. */
  258. loadStacksErrorCallback: function () {
  259. console.log('Error in loading stacks');
  260. },
  261. /**
  262. * Parse loaded data and create array of stacks objects
  263. */
  264. loadStacksVersionsSuccessCallback: function (data) {
  265. var result = [];
  266. var stackVersions = data.items.filterProperty('Versions.active');
  267. stackVersions.sortProperty('Versions.stack_version').reverse().forEach(function (version) {
  268. /*
  269. * operatingSystems:[
  270. * {
  271. * osType: 'centos5',
  272. * baseUrl: 'http://...',
  273. * originalBaseUrl: 'http://...',
  274. * defaultBaseUrl: 'http://...',
  275. * latestBaseUrl: 'http://...',
  276. * mirrorsList: '';
  277. * },
  278. * {
  279. * osType: 'centos6',
  280. * baseUrl: 'http://...',
  281. * originalBaseUrl: 'http://...',
  282. * defaultBaseUrl: 'http://...',
  283. * latestBaseUrl: 'http://...',
  284. * mirrorsList: '';
  285. * },
  286. * ]
  287. */
  288. var oses = [];
  289. if (version.operatingSystems) {
  290. version.operatingSystems.forEach(function (os) {
  291. if (os.repositories) {
  292. os.repositories.forEach(function (repo) {
  293. var defaultBaseUrl = repo.Repositories.default_base_url || repo.Repositories.base_url;
  294. var latestBaseUrl = repo.Repositories.latest_base_url || defaultBaseUrl;
  295. if (!App.supports.ubuntu && os.OperatingSystems.os_type == 'debian12') return; // @todo: remove after Ubuntu support confirmation
  296. oses.push({
  297. osType: os.OperatingSystems.os_type,
  298. baseUrl: latestBaseUrl,
  299. latestBaseUrl: latestBaseUrl,
  300. originalLatestBaseUrl: latestBaseUrl,
  301. originalBaseUrl: repo.Repositories.base_url,
  302. defaultBaseUrl: defaultBaseUrl,
  303. mirrorsList: repo.Repositories.mirrors_list,
  304. id: os.OperatingSystems.os_type + repo.Repositories.repo_name,
  305. repoId: repo.Repositories.repo_id,
  306. selected: true
  307. });
  308. });
  309. }
  310. });
  311. }
  312. result.push(
  313. Em.Object.create({
  314. name: version.Versions.stack_name + "-" + version.Versions.stack_version,
  315. isSelected: false,
  316. operatingSystems: oses
  317. })
  318. );
  319. }, this);
  320. this.get('stacks').pushObjects(result);
  321. },
  322. /**
  323. * onError callback for loading stacks data
  324. */
  325. loadStacksVersionsErrorCallback: function () {
  326. console.log('Error in loading stacks');
  327. },
  328. /**
  329. * check server version and web client version
  330. */
  331. checkServerClientVersion: function () {
  332. var dfd = $.Deferred();
  333. var self = this;
  334. self.getServerVersion().done(function () {
  335. dfd.resolve();
  336. });
  337. return dfd.promise();
  338. },
  339. getServerVersion: function(){
  340. return App.ajax.send({
  341. name: 'ambari.service.load_server_version',
  342. sender: this,
  343. success: 'getServerVersionSuccessCallback',
  344. error: 'getServerVersionErrorCallback'
  345. });
  346. },
  347. getServerVersionSuccessCallback: function (data) {
  348. var clientVersion = App.get('version');
  349. var serverVersion = (data.RootServiceComponents.component_version).toString();
  350. this.set('ambariServerVersion', serverVersion);
  351. if (clientVersion) {
  352. this.set('versionConflictAlertBody', Em.I18n.t('app.versionMismatchAlert.body').format(serverVersion, clientVersion));
  353. this.set('isServerClientVersionMismatch', clientVersion != serverVersion);
  354. } else {
  355. this.set('isServerClientVersionMismatch', false);
  356. }
  357. },
  358. getServerVersionErrorCallback: function () {
  359. console.log('ERROR: Cannot load Ambari server version');
  360. },
  361. /**
  362. * Save data to model
  363. * @param stepController App.WizardStep4Controller
  364. */
  365. saveServices: function (stepController) {
  366. var serviceNames = [];
  367. this.setDBProperty('service', stepController.get('content'));
  368. stepController.filterProperty('isSelected', true).forEach(function (item) {
  369. serviceNames.push(item.serviceName);
  370. });
  371. this.set('content.selectedServiceNames', serviceNames);
  372. this.setDBProperty('selectedServiceNames', serviceNames);
  373. console.log('installerController.saveServices: saved data ', serviceNames);
  374. },
  375. /**
  376. * Save Master Component Hosts data to Main Controller
  377. * @param stepController App.WizardStep5Controller
  378. */
  379. saveMasterComponentHosts: function (stepController) {
  380. var obj = stepController.get('selectedServicesMasters');
  381. var masterComponentHosts = [];
  382. obj.forEach(function (_component) {
  383. masterComponentHosts.push({
  384. display_name: _component.get('display_name'),
  385. component: _component.get('component_name'),
  386. hostName: _component.get('selectedHost'),
  387. serviceId: _component.get('serviceId'),
  388. isInstalled: false
  389. });
  390. });
  391. console.log("installerController.saveMasterComponentHosts: saved hosts ", masterComponentHosts);
  392. this.setDBProperty('masterComponentHosts', masterComponentHosts);
  393. this.set('content.masterComponentHosts', masterComponentHosts);
  394. },
  395. /**
  396. * Load master component hosts data for using in required step controllers
  397. */
  398. loadMasterComponentHosts: function () {
  399. var masterComponentHosts = this.getDBProperty('masterComponentHosts') || [];
  400. this.set("content.masterComponentHosts", masterComponentHosts);
  401. console.log("InstallerController.loadMasterComponentHosts: loaded hosts ", masterComponentHosts);
  402. },
  403. /**
  404. * Load master component hosts data for using in required step controllers
  405. */
  406. loadSlaveComponentHosts: function () {
  407. var slaveComponentHosts = this.getDBProperty('slaveComponentHosts') || null;
  408. this.set("content.slaveComponentHosts", slaveComponentHosts);
  409. console.log("InstallerController.loadSlaveComponentHosts: loaded hosts ", slaveComponentHosts);
  410. },
  411. /**
  412. * Load serviceConfigProperties to model
  413. */
  414. loadServiceConfigProperties: function () {
  415. var serviceConfigProperties = this.getDBProperty('serviceConfigProperties');
  416. this.set('content.serviceConfigProperties', serviceConfigProperties);
  417. console.log("InstallerController.loadServiceConfigProperties: loaded config ", serviceConfigProperties);
  418. this.set('content.advancedServiceConfig', this.getDBProperty('advancedServiceConfig'));
  419. },
  420. /**
  421. * Generate clients list for selected services and save it to model
  422. * @param stepController step4WizardController
  423. */
  424. saveClients: function (stepController) {
  425. var clients = [];
  426. var serviceComponents = App.StackServiceComponent.find();
  427. stepController.get('content').filterProperty('isSelected', true).forEach(function (_service) {
  428. var client = serviceComponents.filterProperty('serviceName', _service.serviceName).findProperty('isClient', true);
  429. if (client) {
  430. clients.pushObject({
  431. component_name: client.get('componentName'),
  432. display_name: client.get('displayName'),
  433. isInstalled: false
  434. });
  435. }
  436. }, this);
  437. this.setDBProperty('clientInfo', clients);
  438. this.set('content.clients', clients);
  439. console.log("InstallerController.saveClients: saved list ", clients);
  440. },
  441. /**
  442. * Save stacks data to local db
  443. * @param stepController step1WizardController
  444. */
  445. saveStacks: function (stepController) {
  446. var stacks = stepController.get('content.stacks');
  447. if (stacks.length) {
  448. App.set('currentStackVersion', stacks.findProperty('isSelected').get('name'));
  449. } else {
  450. App.set('currentStackVersion', App.defaultStackVersion);
  451. }
  452. App.db.setStacks(stacks);
  453. this.set('content.stacks', stacks);
  454. },
  455. /**
  456. * Check validation of the customized local urls
  457. */
  458. checkRepoURL: function () {
  459. var selectedStack = this.get('content.stacks').findProperty('isSelected', true);
  460. selectedStack.set('reload', true);
  461. var nameVersionCombo = selectedStack.name;
  462. var stackName = nameVersionCombo.split('-')[0];
  463. var stackVersion = nameVersionCombo.split('-')[1];
  464. if (selectedStack && selectedStack.operatingSystems) {
  465. this.set('validationCnt', selectedStack.get('operatingSystems').filterProperty('selected', true).length);
  466. this.set('invalidCnt', 0);
  467. selectedStack.operatingSystems.forEach(function (os) {
  468. os.errorTitle = null;
  469. os.errorContent = null;
  470. var verifyBaseUrl = os.skipValidation ? false: true;
  471. if (os.selected ) {
  472. os.validation = 'icon-repeat';
  473. selectedStack.set('reload', !selectedStack.get('reload'));
  474. App.ajax.send({
  475. name: 'wizard.advanced_repositories.valid_url',
  476. sender: this,
  477. data: {
  478. stackName: stackName,
  479. stackVersion: stackVersion,
  480. repoId: os.repoId,
  481. osType: os.osType,
  482. osId: os.id,
  483. data: {
  484. 'Repositories': {
  485. 'base_url': os.baseUrl,
  486. "verify_base_url": verifyBaseUrl
  487. }
  488. }
  489. },
  490. success: 'checkRepoURLSuccessCallback',
  491. error: 'checkRepoURLErrorCallback'
  492. });
  493. }
  494. }, this);
  495. }
  496. },
  497. setInvalidUrlCnt: function () {
  498. var selectedStack = this.get('content.stacks').findProperty('isSelected', true);
  499. selectedStack.set('invalidCnt', this.get('invalidCnt'));
  500. }.observes('invalidCnt'),
  501. /**
  502. * onSuccess callback for check Repo URL.
  503. */
  504. checkRepoURLSuccessCallback: function (response, request, data) {
  505. console.log('Success in check Repo URL. data osType: ' + data.osType );
  506. var selectedStack = this.get('content.stacks').findProperty('isSelected', true);
  507. if (selectedStack && selectedStack.operatingSystems) {
  508. var os = selectedStack.operatingSystems.findProperty('id', data.osId);
  509. os.validation = 'icon-ok';
  510. selectedStack.set('reload', !selectedStack.get('reload'));
  511. this.set('validationCnt', this.get('validationCnt') - 1);
  512. }
  513. },
  514. /**
  515. * onError callback for check Repo URL.
  516. */
  517. checkRepoURLErrorCallback: function (request, ajaxOptions, error, data, params) {
  518. console.log('Error in check Repo URL. The baseURL sent is: ' + data.data);
  519. var selectedStack = this.get('content.stacks').findProperty('isSelected', true);
  520. if (selectedStack && selectedStack.operatingSystems) {
  521. var os = selectedStack.operatingSystems.findProperty('id', params.osId);
  522. os.validation = 'icon-exclamation-sign';
  523. os.errorTitle = request.status + ":" + request.statusText;
  524. os.errorContent = $.parseJSON(request.responseText) ? $.parseJSON(request.responseText).message : "";
  525. selectedStack.set('reload', !selectedStack.get('reload'));
  526. this.set('validationCnt', this.get('validationCnt') - 1);
  527. this.set('invalidCnt', this.get('invalidCnt') + 1);
  528. }
  529. },
  530. /**
  531. * Load data for all steps until <code>current step</code>
  532. */
  533. loadAllPriorSteps: function () {
  534. var step = this.get('currentStep');
  535. switch (step) {
  536. case '10':
  537. case '9':
  538. case '8':
  539. case '7':
  540. this.loadServiceConfigGroups();
  541. this.loadServiceConfigProperties();
  542. case '6':
  543. this.loadSlaveComponentHosts();
  544. this.loadClients();
  545. case '5':
  546. this.loadMasterComponentHosts();
  547. this.loadConfirmedHosts();
  548. case '4':
  549. this.loadServices();
  550. this.loadServiceComponentsDb();
  551. case '3':
  552. this.loadConfirmedHosts();
  553. case '2':
  554. this.load('installOptions');
  555. case '1':
  556. this.loadStacks();
  557. case '0':
  558. this.load('cluster');
  559. }
  560. },
  561. /**
  562. * Clear all temporary data
  563. */
  564. finish: function () {
  565. this.setCurrentStep('0');
  566. this.clearStorageData();
  567. var persists = App.router.get('applicationController').persistKey();
  568. App.router.get('applicationController').postUserPref(persists,true);
  569. },
  570. setStepsEnable: function () {
  571. for (var i = 0; i <= this.totalSteps; i++) {
  572. var step = this.get('isStepDisabled').findProperty('step', i);
  573. if (i <= this.get('currentStep')) {
  574. step.set('value', false);
  575. } else {
  576. step.set('value', true);
  577. }
  578. }
  579. }.observes('currentStep'),
  580. setLowerStepsDisable: function (stepNo) {
  581. for (var i = 0; i < stepNo; i++) {
  582. var step = this.get('isStepDisabled').findProperty('step', i);
  583. step.set('value', true);
  584. }
  585. }
  586. });