router.js 19 KB

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