2017-04-24 21:23:04 +03:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Packets contain read / write functionality for the different packet types
|
|
|
|
* supported by the debugging protocol, so that a transport can focus on
|
|
|
|
* delivery and queue management without worrying too much about the specific
|
|
|
|
* packet types.
|
|
|
|
*
|
|
|
|
* They are intended to be "one use only", so a new packet should be
|
|
|
|
* instantiated for each incoming or outgoing packet.
|
|
|
|
*
|
|
|
|
* A complete Packet type should expose at least the following:
|
|
|
|
* * read(stream, scriptableStream)
|
|
|
|
* Called when the input stream has data to read
|
|
|
|
* * write(stream)
|
|
|
|
* Called when the output stream is ready to write
|
|
|
|
* * get done()
|
|
|
|
* Returns true once the packet is done being read / written
|
|
|
|
* * destroy()
|
|
|
|
* Called to clean up at the end of use
|
|
|
|
*/
|
|
|
|
|
|
|
|
const {classes: Cc, interfaces: Ci, utils: Cu} = Components;
|
2017-06-30 02:40:24 +03:00
|
|
|
const {StreamUtils} =
|
|
|
|
Cu.import("chrome://marionette/content/stream-utils.js", {});
|
2017-04-24 21:23:04 +03:00
|
|
|
|
|
|
|
const unicodeConverter = Cc["@mozilla.org/intl/scriptableunicodeconverter"]
|
2017-06-30 02:40:24 +03:00
|
|
|
.createInstance(Ci.nsIScriptableUnicodeConverter);
|
2017-04-24 21:23:04 +03:00
|
|
|
unicodeConverter.charset = "UTF-8";
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
const defer = function() {
|
2017-04-24 21:23:04 +03:00
|
|
|
let deferred = {
|
|
|
|
promise: new Promise((resolve, reject) => {
|
|
|
|
deferred.resolve = resolve;
|
|
|
|
deferred.reject = reject;
|
2017-06-30 02:40:24 +03:00
|
|
|
}),
|
2017-04-24 21:23:04 +03:00
|
|
|
};
|
|
|
|
return deferred;
|
|
|
|
};
|
|
|
|
|
|
|
|
this.EXPORTED_SYMBOLS = ["RawPacket", "Packet", "JSONPacket", "BulkPacket"];
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
// The transport's previous check ensured the header length did not
|
|
|
|
// exceed 20 characters. Here, we opt for the somewhat smaller, but still
|
|
|
|
// large limit of 1 TiB.
|
2017-04-24 21:23:04 +03:00
|
|
|
const PACKET_LENGTH_MAX = Math.pow(2, 40);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A generic Packet processing object (extended by two subtypes below).
|
|
|
|
*/
|
|
|
|
function Packet(transport) {
|
|
|
|
this._transport = transport;
|
|
|
|
this._length = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-06-30 02:40:24 +03:00
|
|
|
* Attempt to initialize a new Packet based on the incoming packet header
|
|
|
|
* we've received so far. We try each of the types in succession, trying
|
|
|
|
* JSON packets first since they are much more common.
|
|
|
|
*
|
|
|
|
* @param {string} header
|
|
|
|
* Packet header string to attempt parsing.
|
|
|
|
* @param {DebuggerTransport} transport
|
|
|
|
* Transport instance that will own the packet.
|
|
|
|
*
|
|
|
|
* @return {Packet}
|
|
|
|
* Parsed packet of the matching type, or null if no types matched.
|
2017-04-24 21:23:04 +03:00
|
|
|
*/
|
2017-06-30 02:40:24 +03:00
|
|
|
Packet.fromHeader = function(header, transport) {
|
2017-04-24 21:23:04 +03:00
|
|
|
return JSONPacket.fromHeader(header, transport) ||
|
|
|
|
BulkPacket.fromHeader(header, transport);
|
|
|
|
};
|
|
|
|
|
|
|
|
Packet.prototype = {
|
|
|
|
|
|
|
|
get length() {
|
|
|
|
return this._length;
|
|
|
|
},
|
|
|
|
|
|
|
|
set length(length) {
|
|
|
|
if (length > PACKET_LENGTH_MAX) {
|
|
|
|
throw Error("Packet length " + length + " exceeds the max length of " +
|
|
|
|
PACKET_LENGTH_MAX);
|
|
|
|
}
|
|
|
|
this._length = length;
|
|
|
|
},
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
destroy() {
|
2017-04-24 21:23:04 +03:00
|
|
|
this._transport = null;
|
2017-06-30 02:40:24 +03:00
|
|
|
},
|
2017-04-24 21:23:04 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2017-06-30 02:40:24 +03:00
|
|
|
* With a JSON packet (the typical packet type sent via the transport),
|
|
|
|
* data is transferred as a JSON packet serialized into a string,
|
|
|
|
* with the string length prepended to the packet, followed by a colon
|
|
|
|
* ([length]:[packet]). The contents of the JSON packet are specified in
|
|
|
|
* the Remote Debugging Protocol specification.
|
|
|
|
*
|
|
|
|
* @param {DebuggerTransport} transport
|
|
|
|
* Transport instance that will own the packet.
|
2017-04-24 21:23:04 +03:00
|
|
|
*/
|
|
|
|
function JSONPacket(transport) {
|
|
|
|
Packet.call(this, transport);
|
|
|
|
this._data = "";
|
|
|
|
this._done = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-06-30 02:40:24 +03:00
|
|
|
* Attempt to initialize a new JSONPacket based on the incoming packet
|
|
|
|
* header we've received so far.
|
|
|
|
*
|
|
|
|
* @param {string} header
|
|
|
|
* Packet header string to attempt parsing.
|
|
|
|
* @param {DebuggerTransport} transport
|
|
|
|
* Transport instance that will own the packet.
|
|
|
|
*
|
|
|
|
* @return {JSONPacket}
|
|
|
|
* Parsed packet, or null if it's not a match.
|
2017-04-24 21:23:04 +03:00
|
|
|
*/
|
2017-06-30 02:40:24 +03:00
|
|
|
JSONPacket.fromHeader = function(header, transport) {
|
2017-04-24 21:23:04 +03:00
|
|
|
let match = this.HEADER_PATTERN.exec(header);
|
|
|
|
|
|
|
|
if (!match) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
let packet = new JSONPacket(transport);
|
|
|
|
packet.length = +match[1];
|
|
|
|
return packet;
|
|
|
|
};
|
|
|
|
|
|
|
|
JSONPacket.HEADER_PATTERN = /^(\d+):$/;
|
|
|
|
|
|
|
|
JSONPacket.prototype = Object.create(Packet.prototype);
|
|
|
|
|
|
|
|
Object.defineProperty(JSONPacket.prototype, "object", {
|
|
|
|
/**
|
|
|
|
* Gets the object (not the serialized string) being read or written.
|
|
|
|
*/
|
2017-06-30 02:40:24 +03:00
|
|
|
get() {
|
2017-04-24 21:23:04 +03:00
|
|
|
return this._object;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the object to be sent when write() is called.
|
|
|
|
*/
|
2017-06-30 02:40:24 +03:00
|
|
|
set(object) {
|
2017-04-24 21:23:04 +03:00
|
|
|
this._object = object;
|
|
|
|
let data = JSON.stringify(object);
|
|
|
|
this._data = unicodeConverter.ConvertFromUnicode(data);
|
|
|
|
this.length = this._data.length;
|
2017-06-30 02:40:24 +03:00
|
|
|
},
|
2017-04-24 21:23:04 +03:00
|
|
|
});
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
JSONPacket.prototype.read = function(stream, scriptableStream) {
|
2017-04-24 21:23:04 +03:00
|
|
|
|
|
|
|
// Read in more packet data.
|
|
|
|
this._readData(stream, scriptableStream);
|
|
|
|
|
|
|
|
if (!this.done) {
|
|
|
|
// Don't have a complete packet yet.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let json = this._data;
|
|
|
|
try {
|
|
|
|
json = unicodeConverter.ConvertToUnicode(json);
|
|
|
|
this._object = JSON.parse(json);
|
|
|
|
} catch (e) {
|
|
|
|
let msg = "Error parsing incoming packet: " + json + " (" + e +
|
|
|
|
" - " + e.stack + ")";
|
|
|
|
console.error(msg);
|
|
|
|
dump(msg + "\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._transport._onJSONObjectReady(this._object);
|
|
|
|
};
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
JSONPacket.prototype._readData = function(stream, scriptableStream) {
|
|
|
|
let bytesToRead = Math.min(
|
|
|
|
this.length - this._data.length,
|
|
|
|
stream.available());
|
2017-04-24 21:23:04 +03:00
|
|
|
this._data += scriptableStream.readBytes(bytesToRead);
|
|
|
|
this._done = this._data.length === this.length;
|
|
|
|
};
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
JSONPacket.prototype.write = function(stream) {
|
2017-04-24 21:23:04 +03:00
|
|
|
|
|
|
|
if (this._outgoing === undefined) {
|
|
|
|
// Format the serialized packet to a buffer
|
|
|
|
this._outgoing = this.length + ":" + this._data;
|
|
|
|
}
|
|
|
|
|
|
|
|
let written = stream.write(this._outgoing, this._outgoing.length);
|
|
|
|
this._outgoing = this._outgoing.slice(written);
|
|
|
|
this._done = !this._outgoing.length;
|
|
|
|
};
|
|
|
|
|
|
|
|
Object.defineProperty(JSONPacket.prototype, "done", {
|
2017-06-30 02:40:24 +03:00
|
|
|
get() {
|
2017-04-24 21:23:04 +03:00
|
|
|
return this._done;
|
2017-06-30 02:40:24 +03:00
|
|
|
},
|
2017-04-24 21:23:04 +03:00
|
|
|
});
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
JSONPacket.prototype.toString = function() {
|
2017-04-24 21:23:04 +03:00
|
|
|
return JSON.stringify(this._object, null, 2);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2017-06-30 02:40:24 +03:00
|
|
|
* With a bulk packet, data is transferred by temporarily handing over
|
|
|
|
* the transport's input or output stream to the application layer for
|
|
|
|
* writing data directly. This can be much faster for large data sets,
|
|
|
|
* and avoids various stages of copies and data duplication inherent in
|
|
|
|
* the JSON packet type. The bulk packet looks like:
|
2017-04-24 21:23:04 +03:00
|
|
|
*
|
2017-06-30 02:40:24 +03:00
|
|
|
* bulk [actor] [type] [length]:[data]
|
2017-04-24 21:23:04 +03:00
|
|
|
*
|
2017-06-30 02:40:24 +03:00
|
|
|
* The interpretation of the data portion depends on the kind of actor and
|
|
|
|
* the packet's type. See the Remote Debugging Protocol Stream Transport
|
|
|
|
* spec for more details.
|
|
|
|
*
|
|
|
|
* @param {DebuggerTransport} transport
|
|
|
|
* Transport instance that will own the packet.
|
2017-04-24 21:23:04 +03:00
|
|
|
*/
|
|
|
|
function BulkPacket(transport) {
|
|
|
|
Packet.call(this, transport);
|
|
|
|
this._done = false;
|
|
|
|
this._readyForWriting = defer();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-06-30 02:40:24 +03:00
|
|
|
* Attempt to initialize a new BulkPacket based on the incoming packet
|
|
|
|
* header we've received so far.
|
|
|
|
*
|
|
|
|
* @param {string} header
|
|
|
|
* Packet header string to attempt parsing.
|
|
|
|
* @param {DebuggerTransport} transport
|
|
|
|
* Transport instance that will own the packet.
|
|
|
|
*
|
|
|
|
* @return {BulkPacket}
|
|
|
|
* Parsed packet, or null if it's not a match.
|
2017-04-24 21:23:04 +03:00
|
|
|
*/
|
2017-06-30 02:40:24 +03:00
|
|
|
BulkPacket.fromHeader = function(header, transport) {
|
2017-04-24 21:23:04 +03:00
|
|
|
let match = this.HEADER_PATTERN.exec(header);
|
|
|
|
|
|
|
|
if (!match) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
let packet = new BulkPacket(transport);
|
|
|
|
packet.header = {
|
|
|
|
actor: match[1],
|
|
|
|
type: match[2],
|
2017-06-30 02:40:24 +03:00
|
|
|
length: +match[3],
|
2017-04-24 21:23:04 +03:00
|
|
|
};
|
|
|
|
return packet;
|
|
|
|
};
|
|
|
|
|
|
|
|
BulkPacket.HEADER_PATTERN = /^bulk ([^: ]+) ([^: ]+) (\d+):$/;
|
|
|
|
|
|
|
|
BulkPacket.prototype = Object.create(Packet.prototype);
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
BulkPacket.prototype.read = function(stream) {
|
2017-04-24 21:23:04 +03:00
|
|
|
// Temporarily pause monitoring of the input stream
|
|
|
|
this._transport.pauseIncoming();
|
|
|
|
|
|
|
|
let deferred = defer();
|
|
|
|
|
|
|
|
this._transport._onBulkReadReady({
|
|
|
|
actor: this.actor,
|
|
|
|
type: this.type,
|
|
|
|
length: this.length,
|
|
|
|
copyTo: (output) => {
|
|
|
|
let copying = StreamUtils.copyStream(stream, output, this.length);
|
|
|
|
deferred.resolve(copying);
|
|
|
|
return copying;
|
|
|
|
},
|
2017-06-30 02:40:24 +03:00
|
|
|
stream,
|
|
|
|
done: deferred,
|
2017-04-24 21:23:04 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
// Await the result of reading from the stream
|
|
|
|
deferred.promise.then(() => {
|
|
|
|
this._done = true;
|
|
|
|
this._transport.resumeIncoming();
|
|
|
|
}, this._transport.close);
|
|
|
|
|
|
|
|
// Ensure this is only done once
|
|
|
|
this.read = () => {
|
|
|
|
throw new Error("Tried to read() a BulkPacket's stream multiple times.");
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
BulkPacket.prototype.write = function(stream) {
|
2017-04-24 21:23:04 +03:00
|
|
|
if (this._outgoingHeader === undefined) {
|
|
|
|
// Format the serialized packet header to a buffer
|
|
|
|
this._outgoingHeader = "bulk " + this.actor + " " + this.type + " " +
|
|
|
|
this.length + ":";
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write the header, or whatever's left of it to write.
|
|
|
|
if (this._outgoingHeader.length) {
|
|
|
|
let written = stream.write(this._outgoingHeader,
|
|
|
|
this._outgoingHeader.length);
|
|
|
|
this._outgoingHeader = this._outgoingHeader.slice(written);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Temporarily pause the monitoring of the output stream
|
|
|
|
this._transport.pauseOutgoing();
|
|
|
|
|
|
|
|
let deferred = defer();
|
|
|
|
|
|
|
|
this._readyForWriting.resolve({
|
|
|
|
copyFrom: (input) => {
|
|
|
|
let copying = StreamUtils.copyStream(input, stream, this.length);
|
|
|
|
deferred.resolve(copying);
|
|
|
|
return copying;
|
|
|
|
},
|
2017-06-30 02:40:24 +03:00
|
|
|
stream,
|
|
|
|
done: deferred,
|
2017-04-24 21:23:04 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
// Await the result of writing to the stream
|
|
|
|
deferred.promise.then(() => {
|
|
|
|
this._done = true;
|
|
|
|
this._transport.resumeOutgoing();
|
|
|
|
}, this._transport.close);
|
|
|
|
|
|
|
|
// Ensure this is only done once
|
|
|
|
this.write = () => {
|
|
|
|
throw new Error("Tried to write() a BulkPacket's stream multiple times.");
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
Object.defineProperty(BulkPacket.prototype, "streamReadyForWriting", {
|
2017-06-30 02:40:24 +03:00
|
|
|
get() {
|
2017-04-24 21:23:04 +03:00
|
|
|
return this._readyForWriting.promise;
|
2017-06-30 02:40:24 +03:00
|
|
|
},
|
2017-04-24 21:23:04 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
Object.defineProperty(BulkPacket.prototype, "header", {
|
2017-06-30 02:40:24 +03:00
|
|
|
get() {
|
2017-04-24 21:23:04 +03:00
|
|
|
return {
|
|
|
|
actor: this.actor,
|
|
|
|
type: this.type,
|
2017-06-30 02:40:24 +03:00
|
|
|
length: this.length,
|
2017-04-24 21:23:04 +03:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
set(header) {
|
2017-04-24 21:23:04 +03:00
|
|
|
this.actor = header.actor;
|
|
|
|
this.type = header.type;
|
|
|
|
this.length = header.length;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
Object.defineProperty(BulkPacket.prototype, "done", {
|
2017-06-30 02:40:24 +03:00
|
|
|
get() {
|
2017-04-24 21:23:04 +03:00
|
|
|
return this._done;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
BulkPacket.prototype.toString = function() {
|
2017-04-24 21:23:04 +03:00
|
|
|
return "Bulk: " + JSON.stringify(this.header, null, 2);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* RawPacket is used to test the transport's error handling of malformed
|
|
|
|
* packets, by writing data directly onto the stream.
|
|
|
|
* @param transport DebuggerTransport
|
|
|
|
* The transport instance that will own the packet.
|
|
|
|
* @param data string
|
|
|
|
* The raw string to send out onto the stream.
|
|
|
|
*/
|
|
|
|
function RawPacket(transport, data) {
|
|
|
|
Packet.call(this, transport);
|
|
|
|
this._data = data;
|
|
|
|
this.length = data.length;
|
|
|
|
this._done = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
RawPacket.prototype = Object.create(Packet.prototype);
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
RawPacket.prototype.read = function(stream) {
|
2017-04-24 21:23:04 +03:00
|
|
|
// This hasn't yet been needed for testing.
|
|
|
|
throw Error("Not implmented.");
|
|
|
|
};
|
|
|
|
|
2017-06-30 02:40:24 +03:00
|
|
|
RawPacket.prototype.write = function(stream) {
|
2017-04-24 21:23:04 +03:00
|
|
|
let written = stream.write(this._data, this._data.length);
|
|
|
|
this._data = this._data.slice(written);
|
|
|
|
this._done = !this._data.length;
|
|
|
|
};
|
|
|
|
|
|
|
|
Object.defineProperty(RawPacket.prototype, "done", {
|
2017-06-30 02:40:24 +03:00
|
|
|
get() {
|
2017-04-24 21:23:04 +03:00
|
|
|
return this._done;
|
2017-06-30 02:40:24 +03:00
|
|
|
},
|
2017-04-24 21:23:04 +03:00
|
|
|
});
|