router.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587
  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 misc = require('utils/misc');
  19. var App = require('app');
  20. App.WizardRoute = Em.Route.extend({
  21. gotoStep0: Em.Router.transitionTo('step0'),
  22. gotoStep1: Em.Router.transitionTo('step1'),
  23. gotoStep2: Em.Router.transitionTo('step2'),
  24. gotoStep3: Em.Router.transitionTo('step3'),
  25. gotoStep4: Em.Router.transitionTo('step4'),
  26. gotoStep5: Em.Router.transitionTo('step5'),
  27. gotoStep6: Em.Router.transitionTo('step6'),
  28. gotoStep7: Em.Router.transitionTo('step7'),
  29. gotoStep8: Em.Router.transitionTo('step8'),
  30. gotoStep9: Em.Router.transitionTo('step9'),
  31. gotoStep10: Em.Router.transitionTo('step10'),
  32. isRoutable: function() {
  33. return (typeof this.get('route') === 'string' && App.router.get('loggedIn'));
  34. }.property('App.router.loggedIn')
  35. });
  36. App.Router = Em.Router.extend({
  37. enableLogging: true,
  38. isFwdNavigation: true,
  39. backBtnForHigherStep: false,
  40. transitionInProgress: false,
  41. /**
  42. * Is true, if cluster.provisioning_state is equal to 'INSTALLED'
  43. * @type {Boolean}
  44. */
  45. clusterInstallCompleted: false,
  46. /**
  47. * user prefered path to route
  48. */
  49. preferedPath: null,
  50. setNavigationFlow: function (step) {
  51. var matches = step.match(/\d+$/);
  52. var newStep;
  53. if (matches) {
  54. newStep = parseInt(matches[0]);
  55. }
  56. var previousStep = parseInt(this.getInstallerCurrentStep());
  57. this.set('isFwdNavigation', newStep >= previousStep);
  58. },
  59. clearAllSteps: function () {
  60. this.get('installerController').clear();
  61. this.get('addHostController').clear();
  62. this.get('addServiceController').clear();
  63. this.get('backgroundOperationsController').clear();
  64. for (var i = 1; i < 11; i++) {
  65. this.set('wizardStep' + i + 'Controller.hasSubmitted', false);
  66. this.set('wizardStep' + i + 'Controller.isDisabled', true);
  67. }
  68. },
  69. /**
  70. * Temporary fix for getting cluster name
  71. * @return {*}
  72. */
  73. getClusterName: function () {
  74. return App.router.get('clusterController').get('clusterName');
  75. },
  76. /**
  77. * Get current step of Installer wizard
  78. * @return {*}
  79. */
  80. getInstallerCurrentStep: function () {
  81. return this.getWizardCurrentStep('installer');
  82. },
  83. /**
  84. * Get current step for <code>wizardType</code> wizard
  85. * @param wizardType one of <code>installer</code>, <code>addHost</code>, <code>addServices</code>
  86. */
  87. getWizardCurrentStep: function (wizardType) {
  88. var loginName = this.getLoginName();
  89. var currentStep = App.db.getWizardCurrentStep(wizardType);
  90. console.log('getWizardCurrentStep: loginName=' + loginName + ", currentStep=" + currentStep);
  91. if (!currentStep) {
  92. currentStep = wizardType === 'installer' ? '0' : '1';
  93. }
  94. console.log('returning currentStep=' + currentStep);
  95. return currentStep;
  96. },
  97. loggedIn: !!App.db.getAuthenticated(),
  98. loginName: function() {
  99. return this.getLoginName();
  100. }.property('loggedIn'),
  101. getAuthenticated: function () {
  102. var dfd = $.Deferred();
  103. var self = this;
  104. var auth = App.db.getAuthenticated();
  105. App.ajax.send({
  106. name: 'router.login.clusters',
  107. sender: this,
  108. success: 'onAuthenticationSuccess',
  109. error: 'onAuthenticationError'
  110. }).complete(function (xhr) {
  111. if (xhr.isResolved()) {
  112. // if server knows the user and user authenticated by UI
  113. if (auth && auth === true) {
  114. dfd.resolve(self.get('loggedIn'));
  115. // if server knows the user but UI don't, check the response header
  116. // and try to authorize
  117. } else if (xhr.getResponseHeader('User')) {
  118. var user = xhr.getResponseHeader('User');
  119. App.ajax.send({
  120. name: 'router.login',
  121. sender: self,
  122. data: {
  123. usr: user,
  124. loginName: encodeURIComponent(user)
  125. },
  126. success: 'loginSuccessCallback',
  127. error: 'loginErrorCallback'
  128. });
  129. } else {
  130. self.setAuthenticated(false);
  131. dfd.resolve(false);
  132. }
  133. }
  134. });
  135. return dfd.promise();
  136. },
  137. /**
  138. * Response for <code>/clusters?fields=Clusters/provisioning_state</code>
  139. * @type {null|object}
  140. */
  141. clusterData: null,
  142. onAuthenticationSuccess: function (data) {
  143. if (App.db.getAuthenticated() === true) {
  144. this.set('clusterData', data);
  145. this.setAuthenticated(true);
  146. if (data.items.length) {
  147. this.setClusterInstalled(data);
  148. }
  149. }
  150. },
  151. onAuthenticationError: function (data) {
  152. if (data.status === 403) {
  153. this.setAuthenticated(false);
  154. } else {
  155. console.log('error in getAuthenticated');
  156. }
  157. },
  158. setAuthenticated: function (authenticated) {
  159. console.log("TRACE: Entering router:setAuthenticated function");
  160. App.db.setAuthenticated(authenticated);
  161. this.set('loggedIn', authenticated);
  162. },
  163. getLoginName: function () {
  164. return App.db.getLoginName();
  165. },
  166. setLoginName: function (loginName) {
  167. App.db.setLoginName(loginName);
  168. },
  169. /**
  170. * Set user model to local storage
  171. * @param user
  172. */
  173. setUser: function (user) {
  174. App.db.setUser(user);
  175. },
  176. /**
  177. * Get user model from local storage
  178. * @return {*}
  179. */
  180. getUser: function () {
  181. return App.db.getUser();
  182. },
  183. setUserLoggedIn: function(userName) {
  184. this.setAuthenticated(true);
  185. this.setLoginName(userName);
  186. this.setUser(App.User.find().findProperty('id', userName));
  187. },
  188. /**
  189. * Set `clusterInstallCompleted` property based on cluster info response.
  190. *
  191. * @param {Object} clusterObject
  192. **/
  193. setClusterInstalled: function(clusterObject) {
  194. this.set('clusterInstallCompleted', clusterObject.items[0].Clusters.provisioning_state === 'INSTALLED')
  195. },
  196. login: function () {
  197. var controller = this.get('loginController');
  198. var loginName = controller.get('loginName').toLowerCase();
  199. controller.set('loginName', loginName);
  200. var hash = misc.utf8ToB64(loginName + ":" + controller.get('password'));
  201. var usr = '';
  202. if (App.get('testMode')) {
  203. if (loginName === "admin" && controller.get('password') === 'admin') {
  204. usr = 'admin';
  205. } else if (loginName === 'user' && controller.get('password') === 'user') {
  206. usr = 'user';
  207. }
  208. }
  209. App.ajax.send({
  210. name: 'router.login',
  211. sender: this,
  212. data: {
  213. auth: "Basic " + hash,
  214. usr: usr,
  215. loginName: encodeURIComponent(loginName)
  216. },
  217. beforeSend: 'authBeforeSend',
  218. success: 'loginSuccessCallback',
  219. error: 'loginErrorCallback'
  220. });
  221. },
  222. authBeforeSend: function(opt, xhr, data) {
  223. xhr.setRequestHeader("Authorization", data.auth);
  224. },
  225. loginSuccessCallback: function(data, opt, params) {
  226. console.log('login success');
  227. App.usersMapper.map({"items": [data]});
  228. this.setUserLoggedIn(decodeURIComponent(params.loginName));
  229. var requestData = {
  230. loginName: params.loginName,
  231. loginData: data
  232. };
  233. // no need to load cluster data if it's already loaded
  234. if (this.get('clusterData')) {
  235. this.loginGetClustersSuccessCallback(this.get('clusterData'), {}, requestData);
  236. }
  237. else {
  238. App.ajax.send({
  239. name: 'router.login.clusters',
  240. sender: this,
  241. data: requestData,
  242. success: 'loginGetClustersSuccessCallback'
  243. });
  244. }
  245. },
  246. loginErrorCallback: function(request, ajaxOptions, error, opt) {
  247. var controller = this.get('loginController');
  248. console.log("login error: " + error);
  249. this.setAuthenticated(false);
  250. if (request.status == 403) {
  251. var responseMessage = request.responseText;
  252. try{
  253. responseMessage = JSON.parse(request.responseText).message;
  254. }catch(e){}
  255. controller.postLogin(true, false, responseMessage);
  256. } else {
  257. controller.postLogin(false, false, null);
  258. }
  259. },
  260. loginGetClustersSuccessCallback: function (clustersData, opt, params) {
  261. var loginController = this.get('loginController');
  262. var loginData = params.loginData;
  263. var privileges = loginData.privileges || [];
  264. var router = this;
  265. var permissionList = privileges.mapProperty('PrivilegeInfo.permission_name');
  266. var isAdmin = permissionList.contains('AMBARI.ADMIN');
  267. var transitionToApp = false;
  268. if (isAdmin) {
  269. App.set('isAdmin', true);
  270. if (clustersData.items.length) {
  271. router.setClusterInstalled(clustersData);
  272. transitionToApp = true;
  273. } else {
  274. App.ajax.send({
  275. name: 'ambari.service.load_server_version',
  276. sender: this,
  277. success: 'adminViewInfoSuccessCallback'
  278. });
  279. }
  280. } else {
  281. if (clustersData.items.length) {
  282. router.setClusterInstalled(clustersData);
  283. //TODO: Iterate over clusters
  284. var clusterName = clustersData.items[0].Clusters.cluster_name;
  285. var clusterPermissions = privileges.filterProperty('PrivilegeInfo.cluster_name', clusterName).mapProperty('PrivilegeInfo.permission_name');
  286. if (clusterPermissions.contains('CLUSTER.OPERATE')) {
  287. App.setProperties({
  288. isAdmin: true,
  289. isOperator: true
  290. });
  291. transitionToApp = true;
  292. } else if (clusterPermissions.contains('CLUSTER.READ')) {
  293. transitionToApp = true;
  294. }
  295. }
  296. }
  297. App.set('isPermissionDataLoaded', true);
  298. if (transitionToApp) {
  299. if (!Em.isNone(router.get('preferedPath')) &&
  300. router.get('preferedPath') != "#/login") {
  301. window.location = router.get('preferedPath');
  302. router.set('preferedPath', null);
  303. } else {
  304. router.getSection(function (route) {
  305. router.transitionTo(route);
  306. loginController.postLogin(true, true);
  307. });
  308. }
  309. } else {
  310. App.router.get('mainViewsController').loadAmbariViews();
  311. router.transitionTo('main.views.index');
  312. loginController.postLogin(true,true);
  313. }
  314. },
  315. adminViewInfoSuccessCallback: function(data) {
  316. var components = Em.get(data,'components');
  317. if (Em.isArray(components)) {
  318. var mappedVersions = components.map(function(component) {
  319. if (Em.get(component, 'RootServiceComponents.component_version')) {
  320. return Em.get(component, 'RootServiceComponents.component_version');
  321. }
  322. }),
  323. sortedMappedVersions = mappedVersions.sort(),
  324. latestVersion = sortedMappedVersions[sortedMappedVersions.length-1];
  325. window.location.replace('/views/ADMIN_VIEW/' + latestVersion + '/INSTANCE/#/');
  326. }
  327. },
  328. getSection: function (callback) {
  329. if (App.get('testMode')) {
  330. if (App.alwaysGoToInstaller) {
  331. callback('installer');
  332. } else {
  333. callback('main.dashboard.index');
  334. }
  335. } else {
  336. if (this.get('clusterInstallCompleted')) {
  337. App.clusterStatus.updateFromServer(false).complete(function () {
  338. var route = 'main.dashboard.index';
  339. var clusterStatusOnServer = App.clusterStatus.get('value');
  340. if (clusterStatusOnServer) {
  341. var wizardControllerRoutes = require('data/controller_route');
  342. var wizardControllerRoute = wizardControllerRoutes.findProperty('wizardControllerName', clusterStatusOnServer.wizardControllerName);
  343. if (wizardControllerRoute) {
  344. route = wizardControllerRoute.route;
  345. }
  346. }
  347. if (wizardControllerRoute && wizardControllerRoute.wizardControllerName === 'mainAdminStackAndUpgradeController') {
  348. var clusterController = App.router.get('clusterController');
  349. clusterController.loadClusterName().done(function(){
  350. clusterController.restoreUpgradeState().done(function(){
  351. callback(route);
  352. });
  353. });
  354. } else {
  355. callback(route);
  356. }
  357. });
  358. } else {
  359. callback('installer');
  360. }
  361. }
  362. },
  363. logOff: function (context) {
  364. var self = this;
  365. $('title').text(Em.I18n.t('app.name'));
  366. App.router.get('mainController').stopPolling();
  367. // App.db.cleanUp() must be called before router.clearAllSteps().
  368. // otherwise, this.set('installerController.currentStep, 0) would have no effect
  369. // since it's a computed property but we are not setting it as a dependent of App.db.
  370. App.db.cleanUp();
  371. App.setProperties({
  372. isAdmin: false,
  373. isOperator: false,
  374. isPermissionDataLoaded: false
  375. });
  376. this.set('loggedIn', false);
  377. this.clearAllSteps();
  378. console.log("Log off: " + App.router.getClusterName());
  379. this.set('loginController.loginName', '');
  380. this.set('loginController.password', '');
  381. // When logOff is called by Sign Out button, context contains event object. As it is only case we should send logoff request, we are checking context below.
  382. if (!App.get('testMode') && context) {
  383. App.ajax.send({
  384. name: 'router.logoff',
  385. sender: this,
  386. success: 'logOffSuccessCallback',
  387. error: 'logOffErrorCallback'
  388. }).complete(function() {
  389. self.logoffRedirect(context);
  390. });
  391. } else {
  392. this.logoffRedirect();
  393. }
  394. },
  395. logOffSuccessCallback: function () {
  396. console.log("invoked logout on the server successfully");
  397. var applicationController = App.router.get('applicationController');
  398. applicationController.set('isPollerRunning', false);
  399. },
  400. logOffErrorCallback: function () {
  401. console.log("failed to invoke logout on the server");
  402. },
  403. /**
  404. * Redirect function on sign off request.
  405. *
  406. * @param {$.Event} [context=undefined] - triggered event context
  407. */
  408. logoffRedirect: function(context) {
  409. if (App.router.get('clusterController.isLoaded')) {
  410. window.location.reload();
  411. } else {
  412. this.transitionTo('login', context);
  413. }
  414. },
  415. /**
  416. * initialize isAdmin if user is administrator
  417. */
  418. initAdmin: function(){
  419. if (App.db) {
  420. var user = App.db.getUser();
  421. if (user) {
  422. if (user.admin) {
  423. App.set('isAdmin', true);
  424. console.log('Administrator logged in');
  425. }
  426. if (user.operator) {
  427. App.set('isOperator', true);
  428. }
  429. App.set('isPermissionDataLoaded', true);
  430. }
  431. }
  432. },
  433. root: Em.Route.extend({
  434. index: Em.Route.extend({
  435. route: '/',
  436. redirectsTo: 'login'
  437. }),
  438. enter: function(router){
  439. router.initAdmin();
  440. },
  441. login: Em.Route.extend({
  442. route: '/login',
  443. /**
  444. * If the user is already logged in, redirect to where the user was previously
  445. */
  446. enter: function (router, context) {
  447. router.getAuthenticated().done(function (loggedIn) {
  448. var location = router.location.location.hash;
  449. //key to parse URI for prefered path to route
  450. var key = '?targetURI=';
  451. if (loggedIn) {
  452. Ember.run.next(function () {
  453. console.log(router.getLoginName() + ' already authenticated. Redirecting...');
  454. router.getSection(function (route) {
  455. router.transitionTo(route, context);
  456. });
  457. });
  458. } else {
  459. if (location.contains(key)) {
  460. router.set('preferedPath', location.slice(location.indexOf(key) + key.length));
  461. }
  462. }
  463. });
  464. },
  465. connectOutlets: function (router, context) {
  466. $('title').text(Em.I18n.t('app.name'));
  467. console.log('/login:connectOutlet');
  468. console.log('currentStep is: ' + router.getInstallerCurrentStep());
  469. router.get('applicationController').connectOutlet('login');
  470. }
  471. }),
  472. installer: require('routes/installer'),
  473. main: require('routes/main'),
  474. adminView: Em.Route.extend({
  475. route: '/adminView',
  476. enter: function (router) {
  477. if (!router.get('loggedIn') || !App.isAccessible('upgrade_ADMIN') || App.isAccessible('upgrade_OPERATOR')) {
  478. Em.run.next(function () {
  479. router.transitionTo('login');
  480. });
  481. } else {
  482. App.ajax.send({
  483. name: 'ambari.service.load_server_version',
  484. sender: router,
  485. success: 'adminViewInfoSuccessCallback'
  486. });
  487. }
  488. }
  489. }),
  490. experimental: Em.Route.extend({
  491. route: '/experimental',
  492. enter: function (router, context) {
  493. if (App.isAccessible('upgrade_OPERATOR')) {
  494. Em.run.next(function () {
  495. if (router.get('clusterInstallCompleted')) {
  496. router.transitionTo("main.dashboard.widgets");
  497. } else {
  498. router.route("installer");
  499. }
  500. });
  501. } else if (!App.isAccessible('upgrade_ADMIN')) {
  502. Em.run.next(function () {
  503. router.transitionTo("main.views.index");
  504. });
  505. }
  506. },
  507. connectOutlets: function (router, context) {
  508. if (App.isAccessible('upgrade_ONLY_ADMIN')) {
  509. $('title').text(Em.I18n.t('app.name.subtitle.experimental'));
  510. console.log('/experimental:connectOutlet');
  511. router.get('applicationController').connectOutlet('experimental');
  512. }
  513. }
  514. }),
  515. logoff: function (router, context) {
  516. router.logOff(context);
  517. }
  518. })
  519. });