[tor-commits] [meek/extension] Handle connections, do JSON, make requests.
dcf at torproject.org
dcf at torproject.org
Tue Mar 18 18:10:10 UTC 2014
commit 10e4dfeb23cddbb6d367d85f536b0d473a163b45
Author: David Fifield <david at bamsoftware.com>
Date: Sat Mar 15 20:27:38 2014 -0700
Handle connections, do JSON, make requests.
---
firefox/components/main.js | 302 ++++++++++++++++++++++++++++++++++++++------
1 file changed, 263 insertions(+), 39 deletions(-)
diff --git a/firefox/components/main.js b/firefox/components/main.js
index c38eeda..8bda493 100644
--- a/firefox/components/main.js
+++ b/firefox/components/main.js
@@ -2,31 +2,20 @@
// https://developer.mozilla.org/en-US/docs/Mozilla/JavaScript_code_modules/XPCOMUtils.jsm
Components.utils.import("resource://gre/modules/XPCOMUtils.jsm");
+// Everything resides within the MeekHTTPHelper namespace. MeekHTTPHelper is
+// also the type from which NSGetFactory is constructed, and it is the top-level
+// nsIServerSocketListener.
function MeekHTTPHelper() {
this.wrappedJSObject = this;
const LOCAL_PORT = 7000;
- // Create a "direct" nsIProxyInfo that bypasses the default proxy.
- // https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIProtocolProxyService
- var pps = Components.classes["@mozilla.org/network/protocol-proxy-service;1"]
- .getService(Components.interfaces.nsIProtocolProxyService);
- // https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIProxyInfo
- this.directProxyInfo = pps.newProxyInfo("direct", "", 0, 0, 0xffffffff, null);
-
- // https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIIOService
- this.ioService = Components.classes["@mozilla.org/network/io-service;1"]
- .getService(Components.interfaces.nsIIOService);
- this.httpProtocolHandler = this.ioService.getProtocolHandler("http")
- .QueryInterface(Components.interfaces.nsIHttpProtocolHandler);
-
// https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIServerSocket
var serverSocket = Components.classes["@mozilla.org/network/server-socket;1"]
.createInstance(Components.interfaces.nsIServerSocket);
serverSocket.init(LOCAL_PORT, true, -1);
serverSocket.asyncListen(this);
}
-
MeekHTTPHelper.prototype = {
classDescription: "meek HTTP helper component",
classID: Components.ID("{e7bc2b9c-f454-49f3-a19f-14848a4d871d}"),
@@ -39,51 +28,286 @@ MeekHTTPHelper.prototype = {
]),
// nsIServerSocketListener implementation.
- onSocketAccepted: function(aServ, aTransport) {
- dump("onSocketAccepted host " + aTransport.host + "\n");
+ onSocketAccepted: function(server, transport) {
+ dump("onSocketAccepted " + transport.host + ":" + transport.port + "\n");
+ new MeekHTTPHelper.LocalConnectionHandler(transport);
+ },
+ onStopListening: function(server, status) {
+ dump("onStopListening status " + status + "\n");
+ },
+};
- const FRONT_URL = "https://www.google.com/";
- const HOST = "meek-reflect.appspot.com";
+// Global variables and functions.
- var uri = this.ioService.newURI(FRONT_URL, null, null);
+MeekHTTPHelper.LOCAL_READ_TIMEOUT = 2.0;
+MeekHTTPHelper.LOCAL_WRITE_TIMEOUT = 2.0;
+
+// A "direct" nsIProxyInfo that bypasses the default proxy.
+// https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIProtocolProxyService
+// https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIProxyInfo
+MeekHTTPHelper.directProxyInfo = Components.classes["@mozilla.org/network/protocol-proxy-service;1"]
+ .getService(Components.interfaces.nsIProtocolProxyService)
+ .newProxyInfo("direct", "", 0, 0, 0xffffffff, null);
+
+// https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIIOService
+MeekHTTPHelper.ioService = Components.classes["@mozilla.org/network/io-service;1"]
+ .getService(Components.interfaces.nsIIOService);
+MeekHTTPHelper.httpProtocolHandler = MeekHTTPHelper.ioService.getProtocolHandler("http")
+ .QueryInterface(Components.interfaces.nsIHttpProtocolHandler);
+
+// Set the transport to time out at the given absolute deadline.
+MeekHTTPHelper.refreshDeadline = function(transport, deadline) {
+ var timeout;
+ if (deadline === null)
+ timeout = 0xffffffff;
+ else
+ timeout = Math.max(0.0, Math.ceil((deadline - Date.now()) / 1000.0));
+ transport.setTimeout(Components.interfaces.nsISocketTransport.TIMEOUT_READ_WRITE, timeout);
+};
+
+// Reverse-index the Components.results table.
+MeekHTTPHelper.lookupStatus = function(status) {
+ for (var name in Components.results) {
+ if (Components.results[name] === status)
+ return name;
+ }
+ return null;
+};
+
+// LocalConnectionHandler handles each new client connection received on the
+// socket opened by MeekHTTPHelper. It reads a JSON request, makes the request
+// on the Internet, and writes the result back to the socket. Error handling
+// happens within callbacks.
+MeekHTTPHelper.LocalConnectionHandler = function(transport) {
+ dump("LocalConnectionHandler\n");
+ this.transport = transport;
+ this.readRequest(this.makeRequest.bind(this));
+};
+MeekHTTPHelper.LocalConnectionHandler.prototype = {
+ readRequest: function(callback) {
+ dump("readRequest\n");
+ new MeekHTTPHelper.RequestReader(this.transport, this.makeRequest.bind(this));
+ },
+
+ makeRequest: function(req) {
+ dump("makeRequest " + JSON.stringify(req) + "\n");
+ if (!this.requestOk(req)) {
+ this.transport.close(0);
+ return;
+ }
+ var uri = MeekHTTPHelper.ioService.newURI(req.url, null, null);
// Construct an HTTP channel with the proxy bypass.
// https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIHttpChannel
- var channel = this.httpProtocolHandler.newProxiedChannel(uri, this.directProxyInfo, 0, null)
+ var channel = MeekHTTPHelper.httpProtocolHandler.newProxiedChannel(uri, MeekHTTPHelper.directProxyInfo, 0, null)
.QueryInterface(Components.interfaces.nsIHttpChannel);
- // Set the host we really want.
- channel.setRequestHeader("Host", HOST, false);
- channel.redirectionLimit = 0;
+ if (req.header !== undefined) {
+ for (var key in req.header) {
+ dump("setting header " + key + ": " + req.header[key] + "\n");
+ channel.setRequestHeader(key, req.header[key], false);
+ }
+ }
+ if (req.body !== undefined) {
+ let body = atob(req.body);
+ let inputStream = Components.classes["@mozilla.org/io/string-input-stream;1"]
+ .createInstance(Components.interfaces.nsIStringInputStream);
+ inputStream.setData(body, body.length);
+ let uploadChannel = channel.QueryInterface(Components.interfaces.nsIUploadChannel);
+ uploadChannel.setUploadStream(inputStream, "application/octet-stream", body.length);
+ }
// https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIUploadChannel
- // channel.requestMethod = "POST";
+ // says we must set requestMethod after calling setUploadStream.
+ channel.requestMethod = req.method;
+ channel.redirectionLimit = 0;
- channel.asyncOpen(new this.httpResponseStreamListener(), null);
+ channel.asyncOpen(new MeekHTTPHelper.HttpStreamListener(this.returnResponse.bind(this)), channel);
},
- onStopListening: function(aServ, aStatus) {
- dump("onStopListening status " + aStatus + "\n");
+
+ returnResponse: function(resp) {
+ dump("returnResponse " + JSON.stringify(resp) + "\n");
+ outputStream = this.transport.openOutputStream(
+ Components.interfaces.nsITransport.OPEN_BLOCKING | Components.interfaces.nsITransport.OPEN_UNBUFFERED, 0, 0);
+ var output = Components.classes["@mozilla.org/binaryoutputstream;1"]
+ .createInstance(Components.interfaces.nsIBinaryOutputStream);
+ output.setOutputStream(outputStream);
+
+ var converter = Components.classes["@mozilla.org/intl/scriptableunicodeconverter"]
+ .createInstance(Components.interfaces.nsIScriptableUnicodeConverter);
+ converter.charset = "UTF-8";
+ var s = JSON.stringify(resp);
+ var data = converter.convertToByteArray(s);
+
+ var deadline = Date.now() + MeekHTTPHelper.LOCAL_WRITE_TIMEOUT * 1000;
+ try {
+ MeekHTTPHelper.refreshDeadline(this.transport, deadline);
+ output.write32(data.length);
+ MeekHTTPHelper.refreshDeadline(this.transport, deadline);
+ output.writeByteArray(data, data.length);
+ } finally {
+ output.close();
+ }
+ },
+
+ // Enforce restrictions on what requests we are willing to make. These can
+ // probably be loosened up. Try and rule out anything unexpected until we
+ // know we need otherwise.
+ requestOk: function(req) {
+ if (req.method === undefined) {
+ dump("req missing \"method\".\n");
+ return false;
+ }
+ if (req.url === undefined) {
+ dump("req missing \"url\".\n");
+ return false;
+ }
+
+ if (req.method !== "POST") {
+ dump("req.method is " + JSON.stringify(req.method) + ", not \"POST\".\n");
+ return false;
+ }
+ if (!req.url.startsWith("https://")) {
+ dump("req.url doesn't start with \"https://\".\n");
+ return false;
+ }
+
+ return true;
},
};
+// RequestReader reads a JSON-encoded request from the given transport, and
+// calls the given callback with the request as an argument. In case of error,
+// the transport is closed and the callback is not called.
+MeekHTTPHelper.RequestReader = function(transport, callback) {
+ dump("RequestReader\n");
+ this.transport = transport;
+ this.callback = callback;
-// https://developer.mozilla.org/en-US/docs/Creating_Sandboxed_HTTP_Connections
-MeekHTTPHelper.prototype.httpResponseStreamListener = function() {
+ this.curThread = Components.classes["@mozilla.org/thread-manager;1"].getService().currentThread;
+ this.inputStream = this.transport.openInputStream(
+ Components.interfaces.nsITransport.OPEN_BLOCKING | Components.interfaces.nsITransport.OPEN_UNBUFFERED, 0, 0);
+
+ this.state = this.STATE_READING_LENGTH;
+ this.buf = new Uint8Array(4);
+ this.bytesToRead = this.buf.length;
+ this.deadline = Date.now() + MeekHTTPHelper.LOCAL_READ_TIMEOUT * 1000;
+ this.asyncWait();
};
-MeekHTTPHelper.prototype.httpResponseStreamListener.prototype = {
+MeekHTTPHelper.RequestReader.prototype = {
+ STATE_READING_LENGTH: 1,
+ STATE_READING_OBJECT: 2,
+ STATE_DONE: 3,
+
+ // Do an asyncWait and handle the result.
+ asyncWait: function() {
+ MeekHTTPHelper.refreshDeadline(this.transport, this.deadline);
+ this.inputStream.asyncWait(this, 0, 0, this.curThread);
+ },
+
+ // nsIInputStreamCallback implementation.
+ onInputStreamReady: function(inputStream) {
+ var input = Components.classes["@mozilla.org/binaryinputstream;1"]
+ .createInstance(Components.interfaces.nsIBinaryInputStream);
+ input.setInputStream(inputStream);
+ try {
+ switch (this.state) {
+ case this.STATE_READING_LENGTH:
+ this.doStateReadingLength(input);
+ this.asyncWait(inputStream);
+ break;
+ case this.STATE_READING_OBJECT:
+ this.doStateReadingObject(input);
+ if (this.bytesToRead > 0)
+ this.asyncWait(inputStream);
+ break;
+ }
+ } catch (e) {
+ dump("got exception " + e + "\n");
+ this.transport.close(0);
+ return;
+ }
+ },
+
+ // Read into this.buf (up to its capacity) and decrement this.bytesToRead.
+ readIntoBuf: function(input) {
+ var n = Math.min(input.available(), this.bytesToRead);
+ var data = input.readByteArray(n);
+ this.buf.subarray(this.buf.length - this.bytesToRead, n).set(data)
+ this.bytesToRead -= n;
+ },
+
+ doStateReadingLength: function(input) {
+ this.readIntoBuf(input);
+ if (this.bytesToRead > 0)
+ return;
+
+ this.state = this.STATE_READING_OBJECT;
+ var b = this.buf;
+ this.bytesToRead = (b[0] << 24) | (b[1] << 16) | (b[2] << 8) | b[3];
+ if (this.bytesToRead > 1000000) {
+ dump("Object length is too long (" + this.bytesToRead + "), ignoring.\n");
+ throw Components.results.NS_ERROR_FAILURE;
+ }
+ this.buf = new Uint8Array(this.bytesToRead);
+ },
+
+ doStateReadingObject: function(input) {
+ this.readIntoBuf(input);
+ if (this.bytesToRead > 0)
+ return;
+
+ this.state = this.STATE_DONE;
+ var converter = Components.classes["@mozilla.org/intl/scriptableunicodeconverter"]
+ .createInstance(Components.interfaces.nsIScriptableUnicodeConverter);
+ converter.charset = "UTF-8";
+ var s = converter.convertFromByteArray(this.buf, this.buf.length);
+ var req = JSON.parse(s);
+ this.callback(req);
+ },
+};
+
+// HttpStreamListener makes the requested HTTP request and calls the given
+// callback with a representation of the response. The "error" key of the return
+// value is defined if and only if there was an error.
+MeekHTTPHelper.HttpStreamListener = function(callback) {
+ this.callback = callback;
+ // This is a list of binary strings that is concatenated in onStopRequest.
+ this.body = [];
+ this.length = 0;
+};
+// https://developer.mozilla.org/en-US/docs/Creating_Sandboxed_HTTP_Connections
+MeekHTTPHelper.HttpStreamListener.prototype = {
// https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIRequestObserver
- onStartRequest: function(aRequest, aContext) {
+ onStartRequest: function(req, context) {
dump("onStartRequest\n");
},
- onStopRequest: function(aRequest, aContext, aStatus) {
- dump("onStopRequest\n");
+ onStopRequest: function(req, context, status) {
+ dump("onStopRequest " + status + "\n");
+ var resp = {
+ status: context.responseStatus,
+ };
+ if (Components.isSuccessCode(status)) {
+ resp.body = btoa(this.body.join(""));
+ } else {
+ let err = MeekHTTPHelper.lookupStatus(status);
+ if (err !== null)
+ resp.error = err;
+ else
+ resp.error = "error " + String(status);
+ }
+ this.callback(resp);
},
+
// https://developer.mozilla.org/en-US/docs/XPCOM_Interface_Reference/nsIStreamListener
- onDataAvailable: function(aRequest, aContext, aStream, aSourceOffset, aLength) {
- dump("onDataAvailable\n");
- var a = new Uint8Array(aLength);
+ onDataAvailable: function(request, context, stream, sourceOffset, length) {
+ dump("onDataAvailable " + length + " bytes\n");
+ if (this.length + length > 1000000) {
+ request.cancel(Components.results.NS_ERROR_ILLEGAL_VALUE);
+ return;
+ }
+ this.length += length;
var input = Components.classes["@mozilla.org/binaryinputstream;1"]
.createInstance(Components.interfaces.nsIBinaryInputStream);
- input.setInputStream(aStream);
- input.readByteArray(aLength, a);
- dump(aLength + ":" + a + "\n");
+ input.setInputStream(stream);
+ this.body.push(String.fromCharCode.apply(null, input.readByteArray(length)));
},
};
More information about the tor-commits
mailing list