2012-02-29 03:19:00 -08:00
|
|
|
const Cc = Components.classes;
|
|
|
|
const Ci = Components.interfaces;
|
|
|
|
const Cr = Components.results;
|
|
|
|
const Cu = Components.utils;
|
|
|
|
const CC = Components.Constructor;
|
|
|
|
|
|
|
|
const ServerSocket = CC("@mozilla.org/network/server-socket;1",
|
|
|
|
"nsIServerSocket",
|
|
|
|
"init");
|
|
|
|
|
|
|
|
/**
|
|
|
|
* TestServer: A single instance of this is created as |serv|. When created,
|
|
|
|
* it starts listening on the loopback address on port |serv.port|. Tests will
|
|
|
|
* connect to it after setting |serv.acceptCallback|, which is invoked after it
|
|
|
|
* accepts a connection.
|
|
|
|
*
|
|
|
|
* Within |serv.acceptCallback|, various properties of |serv| can be used to
|
|
|
|
* run checks. After the callback, the connection is closed, but the server
|
|
|
|
* remains listening until |serv.stop|
|
|
|
|
*
|
|
|
|
* Note: TestServer can only handle a single connection at a time. Tests
|
|
|
|
* should use run_next_test at the end of |serv.acceptCallback| to start the
|
|
|
|
* following test which creates a connection.
|
|
|
|
*/
|
|
|
|
function TestServer() {
|
|
|
|
this.reset();
|
|
|
|
|
|
|
|
// start server.
|
|
|
|
// any port (-1), loopback only (true), default backlog (-1)
|
|
|
|
this.listener = ServerSocket(-1, true, -1);
|
|
|
|
this.port = this.listener.port;
|
|
|
|
do_print('server: listening on', this.port);
|
|
|
|
this.listener.asyncListen(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
TestServer.prototype = {
|
|
|
|
onSocketAccepted: function(socket, trans) {
|
|
|
|
do_print('server: got client connection');
|
|
|
|
|
|
|
|
// one connection at a time.
|
|
|
|
if (this.input !== null) {
|
|
|
|
try { socket.close(); } catch(ignore) {}
|
|
|
|
do_throw("Test written to handle one connection at a time.");
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
this.input = trans.openInputStream(0, 0, 0);
|
|
|
|
this.output = trans.openOutputStream(0, 0, 0);
|
|
|
|
this.selfAddr = trans.getScriptableSelfAddr();
|
|
|
|
this.peerAddr = trans.getScriptablePeerAddr();
|
|
|
|
|
|
|
|
this.acceptCallback();
|
|
|
|
} catch(e) {
|
|
|
|
/* In a native callback such as onSocketAccepted, exceptions might not
|
|
|
|
* get output correctly or logged to test output. Send them through
|
|
|
|
* do_throw, which fails the test immediately. */
|
|
|
|
do_report_unexpected_exception(e, "in TestServer.onSocketAccepted");
|
|
|
|
}
|
|
|
|
|
|
|
|
this.reset();
|
|
|
|
} ,
|
|
|
|
|
|
|
|
onStopListening: function(socket) {} ,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called to close a connection and clean up properties.
|
|
|
|
*/
|
|
|
|
reset: function() {
|
|
|
|
if (this.input)
|
|
|
|
try { this.input.close(); } catch(ignore) {}
|
|
|
|
if (this.output)
|
|
|
|
try { this.output.close(); } catch(ignore) {}
|
|
|
|
|
|
|
|
this.input = null;
|
|
|
|
this.output = null;
|
|
|
|
this.acceptCallback = null;
|
|
|
|
this.selfAddr = null;
|
|
|
|
this.peerAddr = null;
|
|
|
|
} ,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Cleanup for TestServer and this test case.
|
|
|
|
*/
|
|
|
|
stop: function() {
|
|
|
|
this.reset();
|
|
|
|
try { this.listener.close(); } catch(ignore) {}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper function.
|
|
|
|
* Compares two nsINetAddr objects and ensures they are logically equivalent.
|
|
|
|
*/
|
|
|
|
function checkAddrEqual(lhs, rhs) {
|
|
|
|
do_check_eq(lhs.family, rhs.family);
|
|
|
|
|
|
|
|
if (lhs.family === Ci.nsINetAddr.FAMILY_INET) {
|
|
|
|
do_check_eq(lhs.address, rhs.address);
|
|
|
|
do_check_eq(lhs.port, rhs.port);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: fully support ipv6 and local */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* An instance of SocketTransportService, used to create connections.
|
|
|
|
*/
|
|
|
|
var sts;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Single instance of TestServer
|
|
|
|
*/
|
|
|
|
var serv;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Connections have 5 seconds to be made, or a timeout function fails this
|
|
|
|
* test. This prevents the test from hanging and bringing down the entire
|
|
|
|
* xpcshell test chain.
|
|
|
|
*/
|
|
|
|
var connectTimeout = 5*1000;
|
|
|
|
|
2012-03-06 02:02:00 -08:00
|
|
|
/**
|
|
|
|
* A place for individual tests to place Objects of importance for access
|
|
|
|
* throughout asynchronous testing. Particularly important for any output or
|
|
|
|
* input streams opened, as cleanup of those objects (by the garbage collector)
|
|
|
|
* causes the stream to close and may have other side effects.
|
|
|
|
*/
|
|
|
|
var testDataStore = null;
|
|
|
|
|
2012-02-29 03:19:00 -08:00
|
|
|
/**
|
|
|
|
* IPv4 test.
|
|
|
|
*/
|
|
|
|
function testIpv4() {
|
2012-03-06 02:02:00 -08:00
|
|
|
testDataStore = {
|
|
|
|
transport : null ,
|
|
|
|
ouput : null
|
|
|
|
}
|
2012-02-29 03:19:00 -08:00
|
|
|
|
|
|
|
serv.acceptCallback = function() {
|
|
|
|
// disable the timeoutCallback
|
|
|
|
serv.timeoutCallback = function(){};
|
|
|
|
|
2012-03-06 02:02:00 -08:00
|
|
|
var selfAddr = testDataStore.transport.getScriptableSelfAddr();
|
|
|
|
var peerAddr = testDataStore.transport.getScriptablePeerAddr();
|
2012-02-29 03:19:00 -08:00
|
|
|
|
|
|
|
// check peerAddr against expected values
|
|
|
|
do_check_eq(peerAddr.family, Ci.nsINetAddr.FAMILY_INET);
|
2012-03-06 02:02:00 -08:00
|
|
|
do_check_eq(peerAddr.port, testDataStore.transport.port);
|
2012-02-29 03:19:00 -08:00
|
|
|
do_check_eq(peerAddr.port, serv.port);
|
|
|
|
do_check_eq(peerAddr.address, "127.0.0.1");
|
|
|
|
|
|
|
|
// check selfAddr against expected values
|
|
|
|
do_check_eq(selfAddr.family, Ci.nsINetAddr.FAMILY_INET);
|
|
|
|
do_check_eq(selfAddr.address, "127.0.0.1");
|
|
|
|
|
|
|
|
// check that selfAddr = server.peerAddr and vice versa.
|
|
|
|
checkAddrEqual(selfAddr, serv.peerAddr);
|
|
|
|
checkAddrEqual(peerAddr, serv.selfAddr);
|
|
|
|
|
2012-03-06 02:02:00 -08:00
|
|
|
testDataStore = null;
|
2012-02-29 03:19:00 -08:00
|
|
|
do_execute_soon(run_next_test);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Useful timeout for debugging test hangs
|
|
|
|
/*serv.timeoutCallback = function(tname) {
|
|
|
|
if (tname === 'testIpv4')
|
|
|
|
do_throw('testIpv4 never completed a connection to TestServ');
|
|
|
|
};
|
|
|
|
do_timeout(connectTimeout, function(){ serv.timeoutCallback('testIpv4'); });*/
|
|
|
|
|
2012-03-06 02:02:00 -08:00
|
|
|
testDataStore.transport = sts.createTransport(null, 0, '127.0.0.1', serv.port, null);
|
|
|
|
/*
|
|
|
|
* Need to hold |output| so that the output stream doesn't close itself and
|
|
|
|
* the associated connection.
|
|
|
|
*/
|
|
|
|
testDataStore.output = testDataStore.transport.openOutputStream(Ci.nsITransport.OPEN_BLOCKING,0,0);
|
|
|
|
|
2012-02-29 03:19:00 -08:00
|
|
|
/* NEXT:
|
|
|
|
* openOutputStream -> onSocketAccepted -> acceptedCallback -> run_next_test
|
2012-03-06 02:02:00 -08:00
|
|
|
* OR (if the above timeout is uncommented)
|
2012-02-29 03:19:00 -08:00
|
|
|
* <connectTimeout lapses> -> timeoutCallback -> do_throw
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Running the tests.
|
|
|
|
*/
|
|
|
|
function run_test() {
|
|
|
|
sts = Cc["@mozilla.org/network/socket-transport-service;1"]
|
|
|
|
.getService(Ci.nsISocketTransportService);
|
|
|
|
serv = new TestServer();
|
|
|
|
|
|
|
|
do_register_cleanup(function(){ serv.stop(); });
|
|
|
|
|
|
|
|
add_test(testIpv4);
|
|
|
|
/* TODO: testIpv6 */
|
|
|
|
/* TODO: testLocal */
|
|
|
|
|
|
|
|
run_next_test();
|
|
|
|
}
|