// 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 {HTMLElement} clientContainer Container element for the client view. * @param {function(remoting.ClientSession):void} onConnected Callback on * success. * @param {function(remoting.Error):void} onError Callback on error. * @param {function(string, string):boolean} onExtensionMessage The handler for * protocol extension messages. Returns true if a message is recognized; * false otherwise. * @constructor * @implements {remoting.SessionConnector} */ remoting.SessionConnectorImpl = function(clientContainer, onConnected, onError, onExtensionMessage) { /** * @type {HTMLElement} * @private */ this.clientContainer_ = clientContainer; /** * @type {function(remoting.ClientSession):void} * @private */ this.onConnected_ = onConnected; /** * @type {function(remoting.Error):void} * @private */ this.onError_ = onError; /** * @type {function(string, string):boolean} * @private */ this.onExtensionMessage_ = onExtensionMessage; /** * @type {string} * @private */ this.clientJid_ = ''; /** * @type {remoting.ClientSession.Mode} * @private */ this.connectionMode_ = remoting.ClientSession.Mode.ME2ME; /** * @type {remoting.SignalStrategy} * @private */ this.signalStrategy_ = null; /** * @type {remoting.SmartReconnector} * @private */ this.reconnector_ = null; /** * @private */ this.bound_ = { onStateChange : this.onStateChange_.bind(this) }; // Initialize/declare per-connection state. this.reset(); }; /** * Reset the per-connection state so that the object can be re-used for a * second connection. Note the none of the shared WCS state is reset. */ remoting.SessionConnectorImpl.prototype.reset = function() { /** * 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_ = ''; /** * For paired connections, the client id of this device, issued by the host. * * @type {string} * @private */ this.clientPairingId_ = ''; /** * For paired connections, the paired secret for this device, issued by the * host. * * @type {string} * @private */ this.clientPairedSecret_ = ''; /** * 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 {boolean} * @private */ this.refreshHostJidIfOffline_ = false; /** * @type {remoting.ClientSession} * @private */ this.clientSession_ = null; /** * @type {XMLHttpRequest} * @private */ this.pendingXhr_ = null; /** * Function to interactively obtain the PIN from the user. * @type {function(boolean, function(string):void):void} * @private */ this.fetchPin_ = function(onPinFetched) {}; /** * @type {function(string, string, string, * function(string, string):void): void} * @private */ this.fetchThirdPartyToken_ = function( tokenUrl, scope, onThirdPartyTokenFetched) {}; /** * Host 'name', as displayed in the client tool-bar. For a Me2Me connection, * this is the name of the host; for an IT2Me connection, it is the email * address of the person sharing their computer. * * @type {string} * @private */ this.hostDisplayName_ = ''; }; /** * Initiate a Me2Me connection. * * @param {remoting.Host} host The Me2Me host to which to connect. * @param {function(boolean, function(string):void):void} fetchPin Function to * interactively obtain the PIN from the user. * @param {function(string, string, string, * function(string, string): void): void} * fetchThirdPartyToken Function to obtain a token from a third party * authenticaiton server. * @param {string} clientPairingId The client id issued by the host when * this device was paired, if it is already paired. * @param {string} clientPairedSecret The shared secret issued by the host when * this device was paired, if it is already paired. * @return {void} Nothing. */ remoting.SessionConnectorImpl.prototype.connectMe2Me = function(host, fetchPin, fetchThirdPartyToken, clientPairingId, clientPairedSecret) { this.connectMe2MeInternal_( host.hostId, host.jabberId, host.publicKey, host.hostName, fetchPin, fetchThirdPartyToken, clientPairingId, clientPairedSecret, true); }; /** * Update the pairing info so that the reconnect function will work correctly. * * @param {string} clientId The paired client id. * @param {string} sharedSecret The shared secret. */ remoting.SessionConnectorImpl.prototype.updatePairingInfo = function(clientId, sharedSecret) { this.clientPairingId_ = clientId; this.clientPairedSecret_ = sharedSecret; }; /** * Initiate a Me2Me connection. * * @param {string} hostId ID of the Me2Me host. * @param {string} hostJid XMPP JID of the host. * @param {string} hostPublicKey Public Key of the host. * @param {string} hostDisplayName Display name (friendly name) of the host. * @param {function(boolean, function(string):void):void} fetchPin Function to * interactively obtain the PIN from the user. * @param {function(string, string, string, * function(string, string): void): void} * fetchThirdPartyToken Function to obtain a token from a third party * authenticaiton server. * @param {string} clientPairingId The client id issued by the host when * this device was paired, if it is already paired. * @param {string} clientPairedSecret The shared secret issued by the host when * this device was paired, if it is already paired. * @param {boolean} refreshHostJidIfOffline Whether to refresh the JID and retry * the connection if the current JID is offline. * @return {void} Nothing. * @private */ remoting.SessionConnectorImpl.prototype.connectMe2MeInternal_ = function(hostId, hostJid, hostPublicKey, hostDisplayName, fetchPin, fetchThirdPartyToken, clientPairingId, clientPairedSecret, refreshHostJidIfOffline) { // Cancel any existing connect operation. this.cancel(); this.hostId_ = hostId; this.hostJid_ = hostJid; this.hostPublicKey_ = hostPublicKey; this.fetchPin_ = fetchPin; this.fetchThirdPartyToken_ = fetchThirdPartyToken; this.hostDisplayName_ = hostDisplayName; this.connectionMode_ = remoting.ClientSession.Mode.ME2ME; this.refreshHostJidIfOffline_ = refreshHostJidIfOffline; this.updatePairingInfo(clientPairingId, clientPairedSecret); this.connectSignaling_(); } /** * Initiate an IT2Me connection. * * @param {string} accessCode The access code as entered by the user. * @return {void} Nothing. */ remoting.SessionConnectorImpl.prototype.connectIT2Me = function(accessCode) { var kSupportIdLen = 7; var kHostSecretLen = 5; var kAccessCodeLen = kSupportIdLen + kHostSecretLen; // Cancel any existing connect operation. this.cancel(); 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_); }; /** * Reconnect a closed connection. * * @return {void} Nothing. */ remoting.SessionConnectorImpl.prototype.reconnect = function() { if (this.connectionMode_ == remoting.ClientSession.Mode.IT2ME) { console.error('reconnect not supported for IT2Me.'); return; } this.connectMe2MeInternal_( this.hostId_, this.hostJid_, this.hostPublicKey_, this.hostDisplayName_, this.fetchPin_, this.fetchThirdPartyToken_, this.clientPairingId_, this.clientPairedSecret_, true); }; /** * Cancel a connection-in-progress. */ remoting.SessionConnectorImpl.prototype.cancel = function() { if (this.clientSession_) { this.clientSession_.removePlugin(); this.clientSession_ = null; } if (this.pendingXhr_) { this.pendingXhr_.abort(); this.pendingXhr_ = null; } this.reset(); }; /** * Get the connection mode (Me2Me or IT2Me) * * @return {remoting.ClientSession.Mode} */ remoting.SessionConnectorImpl.prototype.getConnectionMode = function() { return this.connectionMode_; }; /** * Get host ID. * * @return {string} */ remoting.SessionConnectorImpl.prototype.getHostId = function() { return this.hostId_; }; /** * @private */ remoting.SessionConnectorImpl.prototype.connectSignaling_ = function() { base.dispose(this.signalStrategy_); this.signalStrategy_ = null; /** @type {remoting.SessionConnectorImpl} */ var that = this; /** @param {string} token */ function connectSignalingWithToken(token) { remoting.identity.getEmail( connectSignalingWithTokenAndEmail.bind(null, token), that.onError_); } /** * @param {string} token * @param {string} email */ function connectSignalingWithTokenAndEmail(token, email) { that.signalStrategy_.connect( remoting.settings.XMPP_SERVER_ADDRESS, email, token); } this.signalStrategy_ = remoting.SignalStrategy.create(this.onSignalingState_.bind(this)); remoting.identity.callWithToken(connectSignalingWithToken, this.onError_); }; /** * @private * @param {remoting.SignalStrategy.State} state */ remoting.SessionConnectorImpl.prototype.onSignalingState_ = function(state) { switch (state) { case remoting.SignalStrategy.State.CONNECTED: // Proceed only if the connection hasn't been canceled. if (this.hostJid_) { this.createSession_(); } break; case remoting.SignalStrategy.State.FAILED: this.onError_(this.signalStrategy_.getError()); break; } }; /** * Continue an IT2Me connection once an access token has been obtained. * * @param {string} token An OAuth2 access token. * @return {void} Nothing. * @private */ remoting.SessionConnectorImpl.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.SessionConnectorImpl.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.hostDisplayName_ = this.hostJid_.split('/')[0]; this.connectSignaling_(); return; } else { console.error('Invalid "support-hosts" response from server.'); } } else { this.onError_(this.translateSupportHostsError_(xhr.status)); } }; /** * Creates ClientSession object. */ remoting.SessionConnectorImpl.prototype.createSession_ = function() { // In some circumstances, the WCS