431 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			JavaScript
		
	
	
			
		
		
	
	
			431 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			JavaScript
		
	
	
| /*
 | |
| Copyright 2015, 2016 OpenMarket Ltd
 | |
| Copyright 2017 Vector Creations Ltd
 | |
| 
 | |
| Licensed under the Apache License, Version 2.0 (the "License");
 | |
| you may not use this file except in compliance with the License.
 | |
| You may obtain a copy of the License at
 | |
| 
 | |
|     http://www.apache.org/licenses/LICENSE-2.0
 | |
| 
 | |
| Unless required by applicable law or agreed to in writing, software
 | |
| distributed under the License is distributed on an "AS IS" BASIS,
 | |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| See the License for the specific language governing permissions and
 | |
| limitations under the License.
 | |
| */
 | |
| 
 | |
| import q from 'q';
 | |
| import Matrix from 'matrix-js-sdk';
 | |
| 
 | |
| import MatrixClientPeg from './MatrixClientPeg';
 | |
| import Notifier from './Notifier';
 | |
| import UserActivity from './UserActivity';
 | |
| import Presence from './Presence';
 | |
| import dis from './dispatcher';
 | |
| import DMRoomMap from './utils/DMRoomMap';
 | |
| import RtsClient from './RtsClient';
 | |
| import Modal from './Modal';
 | |
| import sdk from './index';
 | |
| 
 | |
| /**
 | |
|  * Called at startup, to attempt to build a logged-in Matrix session. It tries
 | |
|  * a number of things:
 | |
|  *
 | |
|  * 0. if it looks like we are in the middle of a registration process, it does
 | |
|  *    nothing.
 | |
|  *
 | |
|  * 1. if we have a loginToken in the (real) query params, it uses that to log
 | |
|  *    in.
 | |
|  *
 | |
|  * 2. if we have a guest access token in the fragment query params, it uses
 | |
|  *    that.
 | |
|  *
 | |
|  * 3. if an access token is stored in local storage (from a previous session),
 | |
|  *    it uses that.
 | |
|  *
 | |
|  * 4. it attempts to auto-register as a guest user.
 | |
|  *
 | |
|  * If any of steps 1-4 are successful, it will call {setLoggedIn}, which in
 | |
|  * turn will raise on_logged_in and will_start_client events.
 | |
|  *
 | |
|  * It returns a promise which resolves when the above process completes.
 | |
|  *
 | |
|  * @param {object} opts.realQueryParams: string->string map of the
 | |
|  *     query-parameters extracted from the real query-string of the starting
 | |
|  *     URI.
 | |
|  *
 | |
|  * @param {object} opts.fragmentQueryParams: string->string map of the
 | |
|  *     query-parameters extracted from the #-fragment of the starting URI.
 | |
|  *
 | |
|  * @param {boolean} opts.enableGuest: set to true to enable guest access tokens
 | |
|  *     and auto-guest registrations.
 | |
|  *
 | |
|  * @params {string} opts.guestHsUrl: homeserver URL. Only used if enableGuest is
 | |
|  *     true; defines the HS to register against.
 | |
|  *
 | |
|  * @params {string} opts.guestIsUrl: homeserver URL. Only used if enableGuest is
 | |
|  *     true; defines the IS to use.
 | |
|  *
 | |
|  */
 | |
| export function loadSession(opts) {
 | |
|     const realQueryParams = opts.realQueryParams || {};
 | |
|     const fragmentQueryParams = opts.fragmentQueryParams || {};
 | |
|     let enableGuest = opts.enableGuest || false;
 | |
|     const guestHsUrl = opts.guestHsUrl;
 | |
|     const guestIsUrl = opts.guestIsUrl;
 | |
|     const defaultDeviceDisplayName = opts.defaultDeviceDisplayName;
 | |
| 
 | |
|     if (fragmentQueryParams.client_secret && fragmentQueryParams.sid) {
 | |
|         // this happens during email validation: the email contains a link to the
 | |
|         // IS, which in turn redirects back to vector. We let MatrixChat create a
 | |
|         // Registration component which completes the next stage of registration.
 | |
|         console.log("Not registering as guest: registration already in progress.");
 | |
|         return q();
 | |
|     }
 | |
| 
 | |
|     if (!guestHsUrl) {
 | |
|         console.warn("Cannot enable guest access: can't determine HS URL to use");
 | |
|         enableGuest = false;
 | |
|     }
 | |
| 
 | |
|     if (realQueryParams.loginToken) {
 | |
|         if (!realQueryParams.homeserver) {
 | |
|             console.warn("Cannot log in with token: can't determine HS URL to use");
 | |
|         } else {
 | |
|             return _loginWithToken(realQueryParams, defaultDeviceDisplayName);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (enableGuest &&
 | |
|         fragmentQueryParams.guest_user_id &&
 | |
|         fragmentQueryParams.guest_access_token
 | |
|        ) {
 | |
|         console.log("Using guest access credentials");
 | |
|         setLoggedIn({
 | |
|             userId: fragmentQueryParams.guest_user_id,
 | |
|             accessToken: fragmentQueryParams.guest_access_token,
 | |
|             homeserverUrl: guestHsUrl,
 | |
|             identityServerUrl: guestIsUrl,
 | |
|             guest: true,
 | |
|         });
 | |
|         return q();
 | |
|     }
 | |
| 
 | |
|     return _restoreFromLocalStorage().then((success) => {
 | |
|         if (success) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         if (enableGuest) {
 | |
|             return _registerAsGuest(guestHsUrl, guestIsUrl, defaultDeviceDisplayName);
 | |
|         }
 | |
| 
 | |
|         // fall back to login screen
 | |
|     });
 | |
| }
 | |
| 
 | |
| function _loginWithToken(queryParams, defaultDeviceDisplayName) {
 | |
|     // create a temporary MatrixClient to do the login
 | |
|     var client = Matrix.createClient({
 | |
|         baseUrl: queryParams.homeserver,
 | |
|     });
 | |
| 
 | |
|     return client.login(
 | |
|         "m.login.token", {
 | |
|             token: queryParams.loginToken,
 | |
|             initial_device_display_name: defaultDeviceDisplayName,
 | |
|         },
 | |
|     ).then(function(data) {
 | |
|         console.log("Logged in with token");
 | |
|         setLoggedIn({
 | |
|             userId: data.user_id,
 | |
|             deviceId: data.device_id,
 | |
|             accessToken: data.access_token,
 | |
|             homeserverUrl: queryParams.homeserver,
 | |
|             identityServerUrl: queryParams.identityServer,
 | |
|             guest: false,
 | |
|         });
 | |
|     }, (err) => {
 | |
|         console.error("Failed to log in with login token: " + err + " " +
 | |
|                       err.data);
 | |
|     });
 | |
| }
 | |
| 
 | |
| function _registerAsGuest(hsUrl, isUrl, defaultDeviceDisplayName) {
 | |
|     console.log("Doing guest login on %s", hsUrl);
 | |
| 
 | |
|     // TODO: we should probably de-duplicate this and Login.loginAsGuest.
 | |
|     // Not really sure where the right home for it is.
 | |
| 
 | |
|     // create a temporary MatrixClient to do the login
 | |
|     var client = Matrix.createClient({
 | |
|         baseUrl: hsUrl,
 | |
|     });
 | |
| 
 | |
|     return client.registerGuest({
 | |
|         body: {
 | |
|             initial_device_display_name: defaultDeviceDisplayName,
 | |
|         },
 | |
|     }).then((creds) => {
 | |
|         console.log("Registered as guest: %s", creds.user_id);
 | |
|         setLoggedIn({
 | |
|             userId: creds.user_id,
 | |
|             deviceId: creds.device_id,
 | |
|             accessToken: creds.access_token,
 | |
|             homeserverUrl: hsUrl,
 | |
|             identityServerUrl: isUrl,
 | |
|             guest: true,
 | |
|         });
 | |
|     }, (err) => {
 | |
|         console.error("Failed to register as guest: " + err + " " + err.data);
 | |
|     });
 | |
| }
 | |
| 
 | |
| // returns a promise which resolves to true if a session is found in
 | |
| // localstorage
 | |
| function _restoreFromLocalStorage() {
 | |
|     if (!localStorage) {
 | |
|         return q(false);
 | |
|     }
 | |
|     const hs_url = localStorage.getItem("mx_hs_url");
 | |
|     const is_url = localStorage.getItem("mx_is_url") || 'https://matrix.org';
 | |
|     const access_token = localStorage.getItem("mx_access_token");
 | |
|     const user_id = localStorage.getItem("mx_user_id");
 | |
|     const device_id = localStorage.getItem("mx_device_id");
 | |
| 
 | |
|     let is_guest;
 | |
|     if (localStorage.getItem("mx_is_guest") !== null) {
 | |
|         is_guest = localStorage.getItem("mx_is_guest") === "true";
 | |
|     } else {
 | |
|         // legacy key name
 | |
|         is_guest = localStorage.getItem("matrix-is-guest") === "true";
 | |
|     }
 | |
| 
 | |
|     if (access_token && user_id && hs_url) {
 | |
|         console.log("Restoring session for %s", user_id);
 | |
|         try {
 | |
|             setLoggedIn({
 | |
|                 userId: user_id,
 | |
|                 deviceId: device_id,
 | |
|                 accessToken: access_token,
 | |
|                 homeserverUrl: hs_url,
 | |
|                 identityServerUrl: is_url,
 | |
|                 guest: is_guest,
 | |
|             });
 | |
|             return q(true);
 | |
|         } catch (e) {
 | |
|             return _handleRestoreFailure(e);
 | |
|         }
 | |
|     } else {
 | |
|         console.log("No previous session found.");
 | |
|         return q(false);
 | |
|     }
 | |
| }
 | |
| 
 | |
| function _handleRestoreFailure(e) {
 | |
|     console.log("Unable to restore session", e);
 | |
| 
 | |
|     let msg = e.message;
 | |
|     if (msg == "OLM.BAD_LEGACY_ACCOUNT_PICKLE") {
 | |
|         msg = "You need to log back in to generate end-to-end encryption keys "
 | |
|             + "for this device and submit the public key to your homeserver. "
 | |
|             + "This is a once off; sorry for the inconvenience.";
 | |
| 
 | |
|         _clearLocalStorage();
 | |
| 
 | |
|         return q.reject(new Error(
 | |
|             "Unable to restore previous session: " + msg,
 | |
|         ));
 | |
|     }
 | |
| 
 | |
|     const def = q.defer();
 | |
|     const SessionRestoreErrorDialog =
 | |
|           sdk.getComponent('views.dialogs.SessionRestoreErrorDialog');
 | |
| 
 | |
|     Modal.createDialog(SessionRestoreErrorDialog, {
 | |
|         error: msg,
 | |
|         onFinished: (success) => {
 | |
|             def.resolve(success);
 | |
|         },
 | |
|     });
 | |
| 
 | |
|     return def.promise.then((success) => {
 | |
|         if (success) {
 | |
|             // user clicked continue.
 | |
|             _clearLocalStorage();
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         // try, try again
 | |
|         return _restoreFromLocalStorage();
 | |
|     });
 | |
| }
 | |
| 
 | |
| let rtsClient = null;
 | |
| export function initRtsClient(url) {
 | |
|     rtsClient = new RtsClient(url);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Transitions to a logged-in state using the given credentials
 | |
|  * @param {MatrixClientCreds} credentials The credentials to use
 | |
|  */
 | |
| export function setLoggedIn(credentials) {
 | |
|     credentials.guest = Boolean(credentials.guest);
 | |
|     console.log("setLoggedIn => %s (guest=%s) hs=%s",
 | |
|                 credentials.userId, credentials.guest,
 | |
|                 credentials.homeserverUrl);
 | |
|     // This is dispatched to indicate that the user is still in the process of logging in
 | |
|     // because `teamPromise` may take some time to resolve, breaking the assumption that
 | |
|     // `setLoggedIn` takes an "instant" to complete, and dispatch `on_logged_in` a few ms
 | |
|     // later than MatrixChat might assume.
 | |
|     dis.dispatch({action: 'on_logging_in'});
 | |
| 
 | |
|     // Resolves by default
 | |
|     let teamPromise = Promise.resolve(null);
 | |
| 
 | |
|     // persist the session
 | |
|     if (localStorage) {
 | |
|         try {
 | |
|             localStorage.setItem("mx_hs_url", credentials.homeserverUrl);
 | |
|             localStorage.setItem("mx_is_url", credentials.identityServerUrl);
 | |
|             localStorage.setItem("mx_user_id", credentials.userId);
 | |
|             localStorage.setItem("mx_access_token", credentials.accessToken);
 | |
|             localStorage.setItem("mx_is_guest", JSON.stringify(credentials.guest));
 | |
| 
 | |
|             // if we didn't get a deviceId from the login, leave mx_device_id unset,
 | |
|             // rather than setting it to "undefined".
 | |
|             //
 | |
|             // (in this case MatrixClient doesn't bother with the crypto stuff
 | |
|             // - that's fine for us).
 | |
|             if (credentials.deviceId) {
 | |
|                 localStorage.setItem("mx_device_id", credentials.deviceId);
 | |
|             }
 | |
| 
 | |
|             console.log("Session persisted for %s", credentials.userId);
 | |
|         } catch (e) {
 | |
|             console.warn("Error using local storage: can't persist session!", e);
 | |
|         }
 | |
| 
 | |
|         if (rtsClient && !credentials.guest) {
 | |
|             teamPromise = rtsClient.login(credentials.userId).then((body) => {
 | |
|                 if (body.team_token) {
 | |
|                     localStorage.setItem("mx_team_token", body.team_token);
 | |
|                 }
 | |
|                 return body.team_token;
 | |
|             });
 | |
|         }
 | |
|     } else {
 | |
|         console.warn("No local storage available: can't persist session!");
 | |
|     }
 | |
| 
 | |
|     // stop any running clients before we create a new one with these new credentials
 | |
|     stopMatrixClient();
 | |
| 
 | |
|     MatrixClientPeg.replaceUsingCreds(credentials);
 | |
| 
 | |
|     teamPromise.then((teamToken) => {
 | |
|         dis.dispatch({action: 'on_logged_in', teamToken: teamToken});
 | |
|     }, (err) => {
 | |
|         console.warn("Failed to get team token on login", err);
 | |
|         dis.dispatch({action: 'on_logged_in', teamToken: null});
 | |
|     });
 | |
| 
 | |
|     startMatrixClient();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Logs the current session out and transitions to the logged-out state
 | |
|  */
 | |
| export function logout() {
 | |
|     if (MatrixClientPeg.get().isGuest()) {
 | |
|         // logout doesn't work for guest sessions
 | |
|         // Also we sometimes want to re-log in a guest session
 | |
|         // if we abort the login
 | |
| 
 | |
|         // use settimeout to avoid racing with react unmounting components
 | |
|         // which need a valid matrixclientpeg
 | |
|         setTimeout(()=>{
 | |
|             onLoggedOut();
 | |
|         }, 0);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     return MatrixClientPeg.get().logout().then(onLoggedOut,
 | |
|         (err) => {
 | |
|             // Just throwing an error here is going to be very unhelpful
 | |
|             // if you're trying to log out because your server's down and
 | |
|             // you want to log into a different server, so just forget the
 | |
|             // access token. It's annoying that this will leave the access
 | |
|             // token still valid, but we should fix this by having access
 | |
|             // tokens expire (and if you really think you've been compromised,
 | |
|             // change your password).
 | |
|             console.log("Failed to call logout API: token will not be invalidated");
 | |
|             onLoggedOut();
 | |
|         }
 | |
|     );
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Starts the matrix client and all other react-sdk services that
 | |
|  * listen for events while a session is logged in.
 | |
|  */
 | |
| export function startMatrixClient() {
 | |
|     // dispatch this before starting the matrix client: it's used
 | |
|     // to add listeners for the 'sync' event so otherwise we'd have
 | |
|     // a race condition (and we need to dispatch synchronously for this
 | |
|     // to work).
 | |
|     dis.dispatch({action: 'will_start_client'}, true);
 | |
| 
 | |
|     Notifier.start();
 | |
|     UserActivity.start();
 | |
|     Presence.start();
 | |
|     DMRoomMap.makeShared().start();
 | |
| 
 | |
|     MatrixClientPeg.start();
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Stops a running client and all related services, used after
 | |
|  * a session has been logged out / ended.
 | |
|  */
 | |
| export function onLoggedOut() {
 | |
|     _clearLocalStorage();
 | |
|     stopMatrixClient();
 | |
|     dis.dispatch({action: 'on_logged_out'});
 | |
| }
 | |
| 
 | |
| function _clearLocalStorage() {
 | |
|     if (!window.localStorage) {
 | |
|         return;
 | |
|     }
 | |
|     const hsUrl = window.localStorage.getItem("mx_hs_url");
 | |
|     const isUrl = window.localStorage.getItem("mx_is_url");
 | |
|     window.localStorage.clear();
 | |
| 
 | |
|     // preserve our HS & IS URLs for convenience
 | |
|     // N.B. we cache them in hsUrl/isUrl and can't really inline them
 | |
|     // as getCurrentHsUrl() may call through to localStorage.
 | |
|     // NB. We do clear the device ID (as well as all the settings)
 | |
|     if (hsUrl) window.localStorage.setItem("mx_hs_url", hsUrl);
 | |
|     if (isUrl) window.localStorage.setItem("mx_is_url", isUrl);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Stop all the background processes related to the current client
 | |
|  */
 | |
| export function stopMatrixClient() {
 | |
|     Notifier.stop();
 | |
|     UserActivity.stop();
 | |
|     Presence.stop();
 | |
|     if (DMRoomMap.shared()) DMRoomMap.shared().stop();
 | |
|     var cli = MatrixClientPeg.get();
 | |
|     if (cli) {
 | |
|         cli.stopClient();
 | |
|         cli.removeAllListeners();
 | |
|         cli.store.deleteAllData();
 | |
|         MatrixClientPeg.unset();
 | |
|     }
 | |
| }
 |