 b76c734d16
			
		
	
	b76c734d16
	
	
	
		
			
			This helps video playback do a slightly better job of keeping up in the browser. It's not a dramatic effect, but enough to start making video playback almost tolerable.
		
			
				
	
	
		
			472 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			472 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| "use strict";
 | |
| /*
 | |
|    Copyright (C) 2012 by Jeremy P. White <jwhite@codeweavers.com>
 | |
| 
 | |
|    This file is part of spice-html5.
 | |
| 
 | |
|    spice-html5 is free software: you can redistribute it and/or modify
 | |
|    it under the terms of the GNU Lesser General Public License as published by
 | |
|    the Free Software Foundation, either version 3 of the License, or
 | |
|    (at your option) any later version.
 | |
| 
 | |
|    spice-html5 is distributed in the hope that it will be useful,
 | |
|    but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|    GNU Lesser General Public License for more details.
 | |
| 
 | |
|    You should have received a copy of the GNU Lesser General Public License
 | |
|    along with spice-html5.  If not, see <http://www.gnu.org/licenses/>.
 | |
| */
 | |
| 
 | |
| /*----------------------------------------------------------------------------
 | |
| **  SpiceConn
 | |
| **      This is the base Javascript class for establishing and
 | |
| **  managing a connection to a Spice Server.
 | |
| **  It is used to provide core functionality to the Spice main,
 | |
| **  display, inputs, and cursor channels.  See main.js for 
 | |
| **  usage.
 | |
| **--------------------------------------------------------------------------*/
 | |
| function SpiceConn(o)
 | |
| {
 | |
|     if (o === undefined || o.uri === undefined || ! o.uri)
 | |
|         throw new Error("You must specify a uri");
 | |
| 
 | |
|     this.ws = new WebSocket(o.uri, 'binary');
 | |
| 
 | |
|     if (! this.ws.binaryType)
 | |
|         throw new Error("WebSocket doesn't support binaryType.  Try a different browser.");
 | |
| 
 | |
|     this.connection_id = o.connection_id !== undefined ? o.connection_id : 0;
 | |
|     this.type = o.type !== undefined ? o.type : SPICE_CHANNEL_MAIN;
 | |
|     this.chan_id = o.chan_id !== undefined ? o.chan_id : 0;
 | |
|     if (o.parent !== undefined)
 | |
|     {
 | |
|         this.parent = o.parent;
 | |
|         this.message_id = o.parent.message_id;
 | |
|         this.password = o.parent.password;
 | |
|     }
 | |
|     if (o.screen_id !== undefined)
 | |
|         this.screen_id = o.screen_id;
 | |
|     if (o.dump_id !== undefined)
 | |
|         this.dump_id = o.dump_id;
 | |
|     if (o.message_id !== undefined)
 | |
|         this.message_id = o.message_id;
 | |
|     if (o.password !== undefined)
 | |
|         this.password = o.password;
 | |
|     if (o.onerror !== undefined)
 | |
|         this.onerror = o.onerror;
 | |
|     if (o.onsuccess !== undefined)
 | |
|         this.onsuccess = o.onsuccess;
 | |
|     if (o.onagent !== undefined)
 | |
|         this.onagent = o.onagent;
 | |
| 
 | |
|     this.state = "connecting";
 | |
|     this.ws.parent = this;
 | |
|     this.wire_reader = new SpiceWireReader(this, this.process_inbound);
 | |
|     this.messages_sent = 0;
 | |
|     this.warnings = [];
 | |
| 
 | |
|     this.ws.addEventListener('open', function(e) {
 | |
|         DEBUG > 0 && console.log(">> WebSockets.onopen");
 | |
|         DEBUG > 0 && console.log("id " + this.parent.connection_id +"; type " + this.parent.type);
 | |
| 
 | |
|         /***********************************************************************
 | |
|         **          WHERE IT ALL REALLY BEGINS
 | |
|         ***********************************************************************/
 | |
|         this.parent.send_hdr();
 | |
|         this.parent.wire_reader.request(SpiceLinkHeader.prototype.buffer_size());
 | |
|         this.parent.state = "start";
 | |
|     });
 | |
|     this.ws.addEventListener('error', function(e) {
 | |
|         if ('url' in e.target) {
 | |
|             this.parent.log_err("WebSocket error: Can't connect to websocket on URL: " + e.target.url);
 | |
|         }
 | |
|         this.parent.report_error(e);
 | |
|     });
 | |
|     this.ws.addEventListener('close', function(e) {
 | |
|         DEBUG > 0 && console.log(">> WebSockets.onclose");
 | |
|         DEBUG > 0 && console.log("id " + this.parent.connection_id +"; type " + this.parent.type);
 | |
|         DEBUG > 0 && console.log(e);
 | |
|         if (this.parent.state != "closing" && this.parent.state != "error" && this.parent.onerror !== undefined)
 | |
|         {
 | |
|             var e;
 | |
|             if (this.parent.state == "connecting")
 | |
|                 e = new Error("Connection refused.");
 | |
|             else if (this.parent.state == "start" || this.parent.state == "link")
 | |
|                 e = new Error("Unexpected protocol mismatch.");
 | |
|             else if (this.parent.state == "ticket")
 | |
|                 e = new Error("Bad password.");
 | |
|             else
 | |
|                 e = new Error("Unexpected close while " + this.parent.state);
 | |
| 
 | |
|             this.parent.onerror(e);
 | |
|             this.parent.log_err(e.toString());
 | |
|         }
 | |
|     });
 | |
| 
 | |
|     if (this.ws.readyState == 2 || this.ws.readyState == 3)
 | |
|         throw new Error("Unable to connect to " + o.uri);
 | |
| 
 | |
|     this.timeout = window.setTimeout(spiceconn_timeout, SPICE_CONNECT_TIMEOUT, this);
 | |
| }
 | |
| 
 | |
| SpiceConn.prototype =
 | |
| {
 | |
|     send_hdr : function ()
 | |
|     {
 | |
|         var hdr = new SpiceLinkHeader;
 | |
|         var msg = new SpiceLinkMess;
 | |
| 
 | |
|         msg.connection_id = this.connection_id;
 | |
|         msg.channel_type = this.type;
 | |
|         // FIXME - we're not setting a channel_id...
 | |
|         msg.common_caps.push(
 | |
|             (1 << SPICE_COMMON_CAP_PROTOCOL_AUTH_SELECTION) |
 | |
|             (1 << SPICE_COMMON_CAP_MINI_HEADER)
 | |
|             );
 | |
| 
 | |
|         if (msg.channel_type == SPICE_CHANNEL_PLAYBACK)
 | |
|             msg.channel_caps.push(
 | |
|                 (1 << SPICE_PLAYBACK_CAP_OPUS)
 | |
|             );
 | |
|         else if (msg.channel_type == SPICE_CHANNEL_MAIN)
 | |
|             msg.channel_caps.push(
 | |
|                 (1 << SPICE_MAIN_CAP_AGENT_CONNECTED_TOKENS)
 | |
|             );
 | |
|         else if (msg.channel_type == SPICE_CHANNEL_DISPLAY)
 | |
|             msg.channel_caps.push(
 | |
|                 (1 << SPICE_DISPLAY_CAP_SIZED_STREAM) |
 | |
|                 (1 << SPICE_DISPLAY_CAP_STREAM_REPORT)
 | |
|             );
 | |
| 
 | |
|         hdr.size = msg.buffer_size();
 | |
| 
 | |
|         var mb = new ArrayBuffer(hdr.buffer_size() + msg.buffer_size());
 | |
|         hdr.to_buffer(mb);
 | |
|         msg.to_buffer(mb, hdr.buffer_size());
 | |
| 
 | |
|         DEBUG > 1 && console.log("Sending header:");
 | |
|         DEBUG > 2 && hexdump_buffer(mb);
 | |
|         this.ws.send(mb);
 | |
|     },
 | |
| 
 | |
|     send_ticket: function(ticket)
 | |
|     {
 | |
|         var hdr = new SpiceLinkAuthTicket();
 | |
|         hdr.auth_mechanism = SPICE_COMMON_CAP_AUTH_SPICE;
 | |
|         // FIXME - we need to implement RSA to make this work right
 | |
|         hdr.encrypted_data = ticket;
 | |
|         var mb = new ArrayBuffer(hdr.buffer_size());
 | |
| 
 | |
|         hdr.to_buffer(mb);
 | |
|         DEBUG > 1 && console.log("Sending ticket:");
 | |
|         DEBUG > 2 && hexdump_buffer(mb);
 | |
|         this.ws.send(mb);
 | |
|     },
 | |
| 
 | |
|     send_msg: function(msg)
 | |
|     {
 | |
|         var mb = new ArrayBuffer(msg.buffer_size());
 | |
|         msg.to_buffer(mb);
 | |
|         this.messages_sent++;
 | |
|         DEBUG > 0 && console.log(">> hdr " + this.channel_type() + " type " + msg.type + " size " + mb.byteLength);
 | |
|         DEBUG > 2 && hexdump_buffer(mb);
 | |
|         this.ws.send(mb);
 | |
|     },
 | |
| 
 | |
|     process_inbound: function(mb, saved_header)
 | |
|     {
 | |
|         DEBUG > 2 && console.log(this.type + ": processing message of size " + mb.byteLength + "; state is " + this.state);
 | |
|         if (this.state == "ready")
 | |
|         {
 | |
|             if (saved_header == undefined)
 | |
|             {
 | |
|                 var msg = new SpiceMiniData(mb);
 | |
| 
 | |
|                 if (msg.type > 500)
 | |
|                 {
 | |
|                     alert("Something has gone very wrong; we think we have message of type " + msg.type);
 | |
|                     debugger;
 | |
|                 }
 | |
| 
 | |
|                 if (msg.size == 0)
 | |
|                 {
 | |
|                     this.process_message(msg);
 | |
|                     this.wire_reader.request(SpiceMiniData.prototype.buffer_size());
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     this.wire_reader.request(msg.size);
 | |
|                     this.wire_reader.save_header(msg);
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 saved_header.data = mb;
 | |
|                 this.process_message(saved_header);
 | |
|                 this.wire_reader.request(SpiceMiniData.prototype.buffer_size());
 | |
|                 this.wire_reader.save_header(undefined);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         else if (this.state == "start")
 | |
|         {
 | |
|             this.reply_hdr = new SpiceLinkHeader(mb);
 | |
|             if (this.reply_hdr.magic != SPICE_MAGIC)
 | |
|             {
 | |
|                 this.state = "error";
 | |
|                 var e = new Error('Error: magic mismatch: ' + this.reply_hdr.magic);
 | |
|                 this.report_error(e);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 // FIXME - Determine major/minor version requirements
 | |
|                 this.wire_reader.request(this.reply_hdr.size);
 | |
|                 this.state = "link";
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         else if (this.state == "link")
 | |
|         {
 | |
|             this.reply_link = new SpiceLinkReply(mb);
 | |
|              // FIXME - Screen the caps - require minihdr at least, right?
 | |
|             if (this.reply_link.error)
 | |
|             {
 | |
|                 this.state = "error";
 | |
|                 var e = new Error('Error: reply link error ' + this.reply_link.error);
 | |
|                 this.report_error(e);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 this.send_ticket(rsa_encrypt(this.reply_link.pub_key, this.password + String.fromCharCode(0)));
 | |
|                 this.state = "ticket";
 | |
|                 this.wire_reader.request(SpiceLinkAuthReply.prototype.buffer_size());
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         else if (this.state == "ticket")
 | |
|         {
 | |
|             this.auth_reply = new SpiceLinkAuthReply(mb);
 | |
|             if (this.auth_reply.auth_code == SPICE_LINK_ERR_OK)
 | |
|             {
 | |
|                 DEBUG > 0 && console.log(this.type + ': Connected');
 | |
| 
 | |
|                 if (this.type == SPICE_CHANNEL_DISPLAY)
 | |
|                 {
 | |
|                     // FIXME - pixmap and glz dictionary config info?
 | |
|                     var dinit = new SpiceMsgcDisplayInit();
 | |
|                     var reply = new SpiceMiniData();
 | |
|                     reply.build_msg(SPICE_MSGC_DISPLAY_INIT, dinit);
 | |
|                     DEBUG > 0 && console.log("Request display init");
 | |
|                     this.send_msg(reply);
 | |
|                 }
 | |
|                 this.state = "ready";
 | |
|                 this.wire_reader.request(SpiceMiniData.prototype.buffer_size());
 | |
|                 if (this.timeout)
 | |
|                 {
 | |
|                     window.clearTimeout(this.timeout);
 | |
|                     delete this.timeout;
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 this.state = "error";
 | |
|                 if (this.auth_reply.auth_code == SPICE_LINK_ERR_PERMISSION_DENIED)
 | |
|                 {
 | |
|                     var e = new Error("Permission denied.");
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     var e = new Error("Unexpected link error " + this.auth_reply.auth_code);
 | |
|                 }
 | |
|                 this.report_error(e);
 | |
|             }
 | |
|         }
 | |
|     },
 | |
| 
 | |
|     process_common_messages : function(msg)
 | |
|     {
 | |
|         if (msg.type == SPICE_MSG_SET_ACK)
 | |
|         {
 | |
|             var ack = new SpiceMsgSetAck(msg.data);
 | |
|             // FIXME - what to do with generation?
 | |
|             this.ack_window = ack.window;
 | |
|             DEBUG > 1 && console.log(this.type + ": set ack to " + ack.window);
 | |
|             this.msgs_until_ack = this.ack_window;
 | |
|             var ackack = new SpiceMsgcAckSync(ack);
 | |
|             var reply = new SpiceMiniData();
 | |
|             reply.build_msg(SPICE_MSGC_ACK_SYNC, ackack);
 | |
|             this.send_msg(reply);
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         if (msg.type == SPICE_MSG_PING)
 | |
|         {
 | |
|             DEBUG > 1 && console.log("ping!");
 | |
|             var pong = new SpiceMiniData;
 | |
|             pong.type = SPICE_MSGC_PONG;
 | |
|             if (msg.data)
 | |
|             {
 | |
|                 pong.data = msg.data.slice(0, 12);
 | |
|             }
 | |
|             pong.size = pong.buffer_size();
 | |
|             this.send_msg(pong);
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         if (msg.type == SPICE_MSG_NOTIFY)
 | |
|         {
 | |
|             // FIXME - Visibility + what
 | |
|             var notify = new SpiceMsgNotify(msg.data);
 | |
|             if (notify.severity == SPICE_NOTIFY_SEVERITY_ERROR)
 | |
|                 this.log_err(notify.message);
 | |
|             else if (notify.severity == SPICE_NOTIFY_SEVERITY_WARN )
 | |
|                 this.log_warn(notify.message);
 | |
|             else
 | |
|                 this.log_info(notify.message);
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
| 
 | |
|     },
 | |
| 
 | |
|     process_message: function(msg)
 | |
|     {
 | |
|         var rc;
 | |
|         DEBUG > 0 && console.log("<< hdr " + this.channel_type() + " type " + msg.type + " size " + (msg.data && msg.data.byteLength));
 | |
|         rc = this.process_common_messages(msg);
 | |
|         if (! rc)
 | |
|         {
 | |
|             if (this.process_channel_message)
 | |
|             {
 | |
|                 rc = this.process_channel_message(msg);
 | |
|                 if (! rc)
 | |
|                     this.log_warn(this.type + ": Unknown message type " + msg.type + "!");
 | |
|             }
 | |
|             else
 | |
|                 this.log_err(this.type + ": No message handlers for this channel; message " + msg.type);
 | |
|         }
 | |
| 
 | |
|         if (this.msgs_until_ack !== undefined && this.ack_window)
 | |
|         {
 | |
|             this.msgs_until_ack--;
 | |
|             if (this.msgs_until_ack <= 0)
 | |
|             {
 | |
|                 this.msgs_until_ack = this.ack_window;
 | |
|                 var ack = new SpiceMiniData();
 | |
|                 ack.type = SPICE_MSGC_ACK;
 | |
|                 this.send_msg(ack);
 | |
|                 DEBUG > 1 && console.log(this.type + ": sent ack");
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return rc;
 | |
|     },
 | |
| 
 | |
|     channel_type: function()
 | |
|     {
 | |
|         if (this.type == SPICE_CHANNEL_MAIN)
 | |
|             return "main";
 | |
|         else if (this.type == SPICE_CHANNEL_DISPLAY)
 | |
|             return "display";
 | |
|         else if (this.type == SPICE_CHANNEL_INPUTS)
 | |
|             return "inputs";
 | |
|         else if (this.type == SPICE_CHANNEL_CURSOR)
 | |
|             return "cursor";
 | |
|         return "unknown-" + this.type;
 | |
| 
 | |
|     },
 | |
| 
 | |
|     log_info: function()
 | |
|     {
 | |
|         var msg = Array.prototype.join.call(arguments, " ");
 | |
|         console.log(msg);
 | |
|         if (this.message_id)
 | |
|         {
 | |
|             var p = document.createElement("p");
 | |
|             p.appendChild(document.createTextNode(msg));
 | |
|             p.className += "spice-message-info";
 | |
|             document.getElementById(this.message_id).appendChild(p);
 | |
|         }
 | |
|     },
 | |
| 
 | |
|     log_warn: function()
 | |
|     {
 | |
|         var msg = Array.prototype.join.call(arguments, " ");
 | |
|         console.log("WARNING: " + msg);
 | |
|         if (this.message_id)
 | |
|         {
 | |
|             var p = document.createElement("p");
 | |
|             p.appendChild(document.createTextNode(msg));
 | |
|             p.className += "spice-message-warning";
 | |
|             document.getElementById(this.message_id).appendChild(p);
 | |
|         }
 | |
|     },
 | |
| 
 | |
|     log_err: function()
 | |
|     {
 | |
|         var msg = Array.prototype.join.call(arguments, " ");
 | |
|         console.log("ERROR: " + msg);
 | |
|         if (this.message_id)
 | |
|         {
 | |
|             var p = document.createElement("p");
 | |
|             p.appendChild(document.createTextNode(msg));
 | |
|             p.className += "spice-message-error";
 | |
|             document.getElementById(this.message_id).appendChild(p);
 | |
|         }
 | |
|     },
 | |
| 
 | |
|     known_unimplemented: function(type, msg)
 | |
|     {
 | |
|         if ( (!this.warnings[type]) || DEBUG > 1)
 | |
|         {
 | |
|             var str = "";
 | |
|             if (DEBUG <= 1)
 | |
|                 str = " [ further notices suppressed ]";
 | |
|             this.log_warn("Unimplemented function " + type + "(" + msg + ")" + str);
 | |
|             this.warnings[type] = true;
 | |
|         }
 | |
|     },
 | |
| 
 | |
|     report_error: function(e)
 | |
|     {
 | |
|         this.log_err(e.toString());
 | |
|         if (this.onerror != undefined)
 | |
|             this.onerror(e);
 | |
|         else
 | |
|             throw(e);
 | |
|     },
 | |
| 
 | |
|     report_success: function(m)
 | |
|     {
 | |
|         if (this.onsuccess != undefined)
 | |
|             this.onsuccess(m);
 | |
|     },
 | |
| 
 | |
|     cleanup: function()
 | |
|     {
 | |
|         if (this.timeout)
 | |
|         {
 | |
|             window.clearTimeout(this.timeout);
 | |
|             delete this.timeout;
 | |
|         }
 | |
|         if (this.ws)
 | |
|         {
 | |
|             this.ws.close();
 | |
|             this.ws = undefined;
 | |
|         }
 | |
|     },
 | |
| 
 | |
|     handle_timeout: function()
 | |
|     {
 | |
|         var e = new Error("Connection timed out.");
 | |
|         this.report_error(e);
 | |
|     },
 | |
| }
 | |
| 
 | |
| function spiceconn_timeout(sc)
 | |
| {
 | |
|     SpiceConn.prototype.handle_timeout.call(sc);
 | |
| }
 |