// Copyright 2013 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. /** * @fileoverview * Connect set-up state machine for Me2Me and IT2Me */ 'use strict'; /** @suppress {duplicate} */ var remoting = remoting || {}; /** * @param {Element} pluginParent The node under which to add the client plugin. * @param {function(remoting.ClientSession):void} onOk Callback on success. * @param {function(remoting.Error):void} onError Callback on error. * @constructor */ remoting.SessionConnector = function(pluginParent, onOk, onError) { /** * @type {Element} * @private */ this.pluginParent_ = pluginParent; /** * @type {function(remoting.ClientSession):void} * @private */ this.onOk_ = onOk; /** * @type {function(remoting.Error):void} * @private */ this.onError_ = onError; /** * @type {remoting.ClientSession.Mode} * @private */ this.connectionMode_ = remoting.ClientSession.Mode.ME2ME; /** * String used to identify the host to which to connect. For IT2Me, this is * the first 7 digits of the access code; for Me2Me it is the host identifier. * * @type {string} * @private */ this.hostId_ = ''; /** * String used to authenticate to the host on connection. For IT2Me, this is * the access code; for Me2Me it is the PIN. * * @type {string} * @private */ this.passPhrase_ = ''; /** * @type {string} * @private */ this.hostJid_ = ''; /** * @type {string} * @private */ this.hostPublicKey_ = ''; /** * @type {string} * @private */ this.clientJid_ = ''; /** * @type {boolean} * @private */ this.refreshHostJidIfOffline_ = true; /** * @type {remoting.ClientSession} * @private */ this.clientSession_ = null; /** * @type {XMLHttpRequest} * @private */ this.pendingXhr_ = null; /** * A timer that polls for an updated access token. * @type {number} * @private */ this.wcsAccessTokenRefreshTimer_ = 0; // Pre-load WCS to improve connection time. remoting.identity.callWithToken(this.loadWcs_.bind(this), this.onError_); }; /** * Initiate a Me2Me connection. * * @param {remoting.Host} host The Me2Me host to which to connect. * @param {string} pin The PIN as entered by the user. * @return {void} Nothing. */ remoting.SessionConnector.prototype.connectMe2Me = function(host, pin) { this.hostId_ = host.hostId; this.hostJid_ = host.jabberId; this.passPhrase_ = pin; this.createSessionIfReady_(); }; /** * Initiate an IT2Me connection. * * @param {string} accessCode The access code as entered by the user. * @return {void} Nothing. */ remoting.SessionConnector.prototype.connectIT2Me = function(accessCode) { var kSupportIdLen = 7; var kHostSecretLen = 5; var kAccessCodeLen = kSupportIdLen + kHostSecretLen; var normalizedAccessCode = this.normalizeAccessCode_(accessCode); if (normalizedAccessCode.length != kAccessCodeLen) { this.onError_(remoting.Error.INVALID_ACCESS_CODE); return; } this.hostId_ = normalizedAccessCode.substring(0, kSupportIdLen); this.passPhrase_ = normalizedAccessCode; this.connectionMode_ = remoting.ClientSession.Mode.IT2ME; remoting.identity.callWithToken(this.connectIT2MeWithToken_.bind(this), this.onError_); }; /** * Cancel a connection-in-progress. */ remoting.SessionConnector.prototype.cancel = function() { if (this.clientSession_) { this.clientSession_.removePlugin(); this.clientSession_ = null; } if (this.pendingXhr_) { this.pendingXhr_.abort(); this.pendingXhr_ = null; } this.hostId_ = ''; this.hostJid_ = ''; this.passPhrase_ = ''; this.hostPublicKey_ = ''; this.refreshHostJidIfOffline_ = true; }; /** * Get the connection mode (Me2Me or IT2Me) * * @return {remoting.ClientSession.Mode} */ remoting.SessionConnector.prototype.getConnectionMode = function() { return this.connectionMode_; }; /** * Continue an IT2Me connection once an access token has been obtained. * * @param {string} token An OAuth2 access token. * @return {void} Nothing. * @private */ remoting.SessionConnector.prototype.connectIT2MeWithToken_ = function(token) { // Resolve the host id to get the host JID. this.pendingXhr_ = remoting.xhr.get( remoting.settings.DIRECTORY_API_BASE_URL + '/support-hosts/' + encodeURIComponent(this.hostId_), this.onIT2MeHostInfo_.bind(this), '', { 'Authorization': 'OAuth ' + token }); }; /** * Continue an IT2Me connection once the host JID has been looked up. * * @param {XMLHttpRequest} xhr The server response to the support-hosts query. * @return {void} Nothing. * @private */ remoting.SessionConnector.prototype.onIT2MeHostInfo_ = function(xhr) { this.pendingXhr_ = null; if (xhr.status == 200) { var host = /** @type {{data: {jabberId: string, publicKey: string}}} */ jsonParseSafe(xhr.responseText); if (host && host.data && host.data.jabberId && host.data.publicKey) { this.hostJid_ = host.data.jabberId; this.hostPublicKey_ = host.data.publicKey; this.createSessionIfReady_(); return; } else { console.error('Invalid "support-hosts" response from server.'); } } else { this.onError_(this.translateSupportHostsError(xhr.status)); } }; /** * Load the WCS driver script. * * @param {string} token An OAuth2 access token. * @return {void} Nothing. * @private */ remoting.SessionConnector.prototype.loadWcs_ = function(token) { remoting.wcsSandbox.setOnReady(this.onWcsLoaded_.bind(this)); remoting.wcsSandbox.setOnError(this.onError_); remoting.wcsSandbox.setAccessToken(token); this.startAccessTokenRefreshTimer_(); }; /** * Continue an IT2Me or Me2Me connection once WCS has been loaded. * * @param {string} clientJid The full JID of the WCS client. * @return {void} Nothing. * @private */ remoting.SessionConnector.prototype.onWcsLoaded_ = function(clientJid) { this.clientJid_ = clientJid; this.createSessionIfReady_(); }; /** * If both the client and host JIDs are available, create a session and connect. * * @return {void} Nothing. * @private */ remoting.SessionConnector.prototype.createSessionIfReady_ = function() { if (!this.clientJid_ || !this.hostJid_) { return; } var securityTypes = 'spake2_hmac,spake2_plain'; this.clientSession_ = new remoting.ClientSession( this.hostJid_, this.clientJid_, this.hostPublicKey_, this.passPhrase_, securityTypes, this.hostId_, this.connectionMode_); this.clientSession_.logHostOfflineErrors(!this.refreshHostJidIfOffline_); this.clientSession_.setOnStateChange(this.onStateChange_.bind(this)); this.clientSession_.createPluginAndConnect(this.pluginParent_); }; /** * Handle a change in the state of the client session prior to successful * connection (after connection, this class no longer handles state change * events). Errors that occur while connecting either trigger a reconnect * or notify the onError handler. * * @param {number} oldState The previous state of the plugin. * @param {number} newState The current state of the plugin. * @return {void} Nothing. * @private */ remoting.SessionConnector.prototype.onStateChange_ = function(oldState, newState) { switch (newState) { case remoting.ClientSession.State.CONNECTED: // When the connection succeeds, deregister for state-change callbacks // and pass the session to the onOk callback. It is expected that it // will register a new state-change callback to handle disconnect // or error conditions. this.clientSession_.setOnStateChange(null); this.onOk_(this.clientSession_); break; case remoting.ClientSession.State.CREATED: console.log('Created plugin'); break; case remoting.ClientSession.State.BAD_PLUGIN_VERSION: this.onError_(remoting.Error.BAD_PLUGIN_VERSION); break; case remoting.ClientSession.State.CONNECTING: console.log('Connecting as ' + remoting.identity.getCachedEmail()); break; case remoting.ClientSession.State.INITIALIZING: console.log('Initializing connection'); break; case remoting.ClientSession.State.CLOSED: // This class deregisters for state-change callbacks when the CONNECTED // state is reached, so it only sees the CLOSED state in exceptional // circumstances. For example, a CONNECTING -> CLOSED transition happens // if the host closes the connection without an error message instead of // accepting it. Since there's no way of knowing exactly what went wrong, // we rely on server-side logs in this case and report a generic error // message. this.onError_(remoting.Error.UNEXPECTED); break; case remoting.ClientSession.State.FAILED: var error = this.clientSession_.getError(); console.error('Client plugin reported connection failed: ' + error); if (error == null) { error = remoting.Error.UNEXPECTED; } if (error == remoting.Error.HOST_IS_OFFLINE && this.refreshHostJidIfOffline_) { this.refreshHostJidIfOffline_ = false; this.clientSession_.removePlugin(); this.clientSession_ = null; remoting.hostList.refresh(this.onHostListRefresh_.bind(this)); } else { this.onError_(error); } break; default: console.error('Unexpected client plugin state: ' + newState); // This should only happen if the web-app and client plugin get out of // sync, and even then the version check should ensure compatibility. this.onError_(remoting.Error.MISSING_PLUGIN); } }; /** * @param {boolean} success True if the host list was successfully refreshed; * false if an error occurred. * @private */ remoting.SessionConnector.prototype.onHostListRefresh_ = function(success) { if (success) { var host = remoting.hostList.getHostForId(this.hostId_); if (host) { this.connectMe2Me(host, this.passPhrase_); return; } } this.onError_(remoting.Error.HOST_IS_OFFLINE); }; /** * Start a timer to periodically refresh the access token used by WCS. Access * tokens have a limited lifespan, and since the WCS driver runs in a sandbox, * it can't obtain a new one directly. * * @return {void} Nothing. * @private */ remoting.SessionConnector.prototype.startAccessTokenRefreshTimer_ = function() { if (this.wcsAccessTokenRefreshTimer_ != 0) { return; } /** @type {remoting.SessionConnector} */ var that = this; var refreshAccessToken = function() { remoting.identity.callWithToken( remoting.wcsSandbox.setAccessToken.bind(remoting.wcsSandbox), that.onError_); }; /** * A timer that polls for an updated access token. * @type {number} * @private */ this.wcsAccessTokenRefreshTimer_ = setInterval(refreshAccessToken, 60 * 1000); } /** * @param {number} error An HTTP error code returned by the support-hosts * endpoint. * @return {remoting.Error} The equivalent remoting.Error code. * @private */ remoting.SessionConnector.prototype.translateSupportHostsError = function(error) { switch (error) { case 0: return remoting.Error.NO_RESPONSE; case 404: return remoting.Error.INVALID_ACCESS_CODE; case 502: // No break case 503: return remoting.Error.SERVICE_UNAVAILABLE; default: return remoting.Error.UNEXPECTED; } }; /** * Normalize the access code entered by the user. * * @param {string} accessCode The access code, as entered by the user. * @return {string} The normalized form of the code (whitespace removed). */ remoting.SessionConnector.prototype.normalizeAccessCode_ = function(accessCode) { // Trim whitespace. return accessCode.replace(/\s/g, ''); };