blob: 9ac7cc4836d8eeb9ce51890647aca74f10f4fe24 [file] [log] [blame]
// Copyright 2014 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.
'use strict';
/** @suppress {duplicate} */
var remoting = remoting || {};
/**
* WCS-based SignalStrategy implementation. Used instead of XMPPConnection
* when XMPP cannot be used (e.g. in V1 app).
*
* @param {function(remoting.SignalStrategy.State):void} onStateChangedCallback
* Callback to call on state change.
* @constructor
* @implements {remoting.SignalStrategy}
*/
remoting.WcsAdapter =
function(onStateChangedCallback) {
/** @private */
this.onStateChangedCallback_ = onStateChangedCallback;
/** @type {?function(Element):void} @private */
this.onIncomingStanzaCallback_ = null;
/** @private */
this.state_ = remoting.SignalStrategy.State.NOT_CONNECTED;
/** @private */
this.jid_ = '';
/** @private */
this.error_ = remoting.Error.NONE;
}
/**
* @param {?function(Element):void} onIncomingStanzaCallback Callback to call on
* incoming messages.
*/
remoting.WcsAdapter.prototype.setIncomingStanzaCallback =
function(onIncomingStanzaCallback) {
this.onIncomingStanzaCallback_ = onIncomingStanzaCallback;
}
remoting.WcsAdapter.prototype.connect = function(server, authToken) {
remoting.wcsSandbox.setOnIq(this.onIncomingStanza_.bind(this));
remoting.wcsSandbox.connect(this.onWcsConnected_.bind(this),
this.onError_.bind(this));
}
/** @return {remoting.SignalStrategy.State} Current state */
remoting.WcsAdapter.prototype.getState = function() {
return this.state_;
}
/** @return {remoting.Error} Error when in FAILED state. */
remoting.WcsAdapter.prototype.getError = function() {
return this.error_;
}
/** @return {string} Current JID when in CONNECTED state. */
remoting.WcsAdapter.prototype.getJid = function() {
return this.jid_;
}
remoting.WcsAdapter.prototype.dispose = function() {
this.setState_(remoting.SignalStrategy.State.CLOSED);
remoting.wcsSandbox.setOnIq(null);
}
/** @param {string} message */
remoting.WcsAdapter.prototype.sendMessage = function(message) {
// Extract the session id, so we can close the session later.
// HACK: Add 'x' prefix to the IDs of the outgoing messages to make sure that
// stanza IDs used by host and client do not match. This is necessary to
// workaround bug in the signaling endpoint used by chromoting.
// TODO(sergeyu): Remove this hack once the server-side bug is fixed.
var parser = new DOMParser();
var iqNode = parser.parseFromString(message, 'text/xml').firstChild;
var type = iqNode.getAttribute('type');
if (type == 'set') {
var id = iqNode.getAttribute('id');
iqNode.setAttribute('id', 'x' + id);
message = (new XMLSerializer()).serializeToString(iqNode);
}
// Send the stanza.
remoting.wcsSandbox.sendIq(message);
}
/** @param {string} jid */
remoting.WcsAdapter.prototype.onWcsConnected_ = function(jid) {
this.jid_ = jid;
this.setState_(remoting.SignalStrategy.State.CONNECTED);
}
/** @param {string} stanza */
remoting.WcsAdapter.prototype.onIncomingStanza_ = function(stanza) {
var parser = new DOMParser();
var parsed = parser.parseFromString(stanza, 'text/xml').firstChild;
// HACK: Remove 'x' prefix added to the id in sendMessage().
try {
var type = parsed.getAttribute('type');
var id = parsed.getAttribute('id');
if (type != 'set' && id.charAt(0) == 'x') {
parsed.setAttribute('id', id.substr(1));
}
} catch (err) {
// Pass message as is when it is malformed.
}
if (this.onIncomingStanzaCallback_) {
this.onIncomingStanzaCallback_(parsed);
}
}
/** @param {remoting.Error} error */
remoting.WcsAdapter.prototype.onError_ = function(error) {
this.error_ = error;
this.setState_(remoting.SignalStrategy.State.FAILED);
}
/**
* @param {remoting.SignalStrategy.State} newState
* @private
*/
remoting.WcsAdapter.prototype.setState_ = function(newState) {
if (this.state_ != newState) {
this.state_ = newState;
this.onStateChangedCallback_(this.state_);
}
};