mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
b3b7f224fd
Adds a new method to NetUtil (readInputStreamToString) that will read a string with or without embedded NULLs from an input stream. Also adds the needed API on nsIScriptableInputStream to make this happen. r=bz sr=biesi a=blocking2.0
582 lines
17 KiB
JavaScript
582 lines
17 KiB
JavaScript
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
* vim: sw=2 ts=2 sts=2 et
|
|
* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is Necko Test Code.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Mozilla Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 2009
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Shawn Wilsher <me@shawnwilsher.com> (Original Author)
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
/**
|
|
* This file tests the methods on NetUtil.jsm.
|
|
*/
|
|
|
|
do_load_httpd_js();
|
|
|
|
Components.utils.import("resource://gre/modules/NetUtil.jsm");
|
|
|
|
// We need the profile directory so the test harness will clean up our test
|
|
// files.
|
|
do_get_profile();
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
//// Helper Methods
|
|
|
|
/**
|
|
* Reads the contents of a file and returns it as a string.
|
|
*
|
|
* @param aFile
|
|
* The file to return from.
|
|
* @return the contents of the file in the form of a string.
|
|
*/
|
|
function getFileContents(aFile)
|
|
{
|
|
let fstream = Cc["@mozilla.org/network/file-input-stream;1"].
|
|
createInstance(Ci.nsIFileInputStream);
|
|
fstream.init(aFile, -1, 0, 0);
|
|
|
|
let cstream = Cc["@mozilla.org/intl/converter-input-stream;1"].
|
|
createInstance(Ci.nsIConverterInputStream);
|
|
cstream.init(fstream, "UTF-8", 0, 0);
|
|
|
|
let string = {};
|
|
cstream.readString(-1, string);
|
|
cstream.close();
|
|
return string.value;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
//// Tests
|
|
|
|
function test_async_write_file()
|
|
{
|
|
do_test_pending();
|
|
|
|
// First, we need an output file to write to.
|
|
let file = Cc["@mozilla.org/file/directory_service;1"].
|
|
getService(Ci.nsIProperties).
|
|
get("ProfD", Ci.nsIFile);
|
|
file.append("NetUtil-async-test-file.tmp");
|
|
file.createUnique(Ci.nsIFile.NORMAL_FILE_TYPE, 0666);
|
|
|
|
// Then, we need an output stream to our output file.
|
|
let ostream = Cc["@mozilla.org/network/file-output-stream;1"].
|
|
createInstance(Ci.nsIFileOutputStream);
|
|
ostream.init(file, -1, -1, 0);
|
|
|
|
// Finally, we need an input stream to take data from.
|
|
const TEST_DATA = "this is a test string";
|
|
let istream = Cc["@mozilla.org/io/string-input-stream;1"].
|
|
createInstance(Ci.nsIStringInputStream);
|
|
istream.setData(TEST_DATA, TEST_DATA.length);
|
|
|
|
NetUtil.asyncCopy(istream, ostream, function(aResult) {
|
|
// Make sure the copy was successful!
|
|
do_check_true(Components.isSuccessCode(aResult));
|
|
|
|
// Check the file contents.
|
|
do_check_eq(TEST_DATA, getFileContents(file));
|
|
|
|
// Finish the test.
|
|
do_test_finished();
|
|
run_next_test();
|
|
});
|
|
}
|
|
|
|
function test_async_write_file_nsISafeOutputStream()
|
|
{
|
|
do_test_pending();
|
|
|
|
// First, we need an output file to write to.
|
|
let file = Cc["@mozilla.org/file/directory_service;1"].
|
|
getService(Ci.nsIProperties).
|
|
get("ProfD", Ci.nsIFile);
|
|
file.append("NetUtil-async-test-file.tmp");
|
|
file.createUnique(Ci.nsIFile.NORMAL_FILE_TYPE, 0666);
|
|
|
|
// Then, we need an output stream to our output file.
|
|
let ostream = Cc["@mozilla.org/network/safe-file-output-stream;1"].
|
|
createInstance(Ci.nsIFileOutputStream);
|
|
ostream.init(file, -1, -1, 0);
|
|
|
|
// Finally, we need an input stream to take data from.
|
|
const TEST_DATA = "this is a test string";
|
|
let istream = Cc["@mozilla.org/io/string-input-stream;1"].
|
|
createInstance(Ci.nsIStringInputStream);
|
|
istream.setData(TEST_DATA, TEST_DATA.length);
|
|
|
|
NetUtil.asyncCopy(istream, ostream, function(aResult) {
|
|
// Make sure the copy was successful!
|
|
do_check_true(Components.isSuccessCode(aResult));
|
|
|
|
// Check the file contents.
|
|
do_check_eq(TEST_DATA, getFileContents(file));
|
|
|
|
// Finish the test.
|
|
do_test_finished();
|
|
run_next_test();
|
|
});
|
|
}
|
|
|
|
function test_newURI_no_spec_throws()
|
|
{
|
|
try {
|
|
NetUtil.newURI();
|
|
do_throw("should throw!");
|
|
}
|
|
catch (e) {
|
|
do_check_eq(e.result, Cr.NS_ERROR_INVALID_ARG);
|
|
}
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_newURI()
|
|
{
|
|
let ios = Cc["@mozilla.org/network/io-service;1"].getService(Ci.nsIIOService);
|
|
|
|
// Check that we get the same URI back from the IO service and the utility
|
|
// method.
|
|
const TEST_URI = "http://mozilla.org";
|
|
let iosURI = ios.newURI(TEST_URI, null, null);
|
|
let NetUtilURI = NetUtil.newURI(TEST_URI);
|
|
do_check_true(iosURI.equals(NetUtilURI));
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_newURI_takes_nsIFile()
|
|
{
|
|
let ios = Cc["@mozilla.org/network/io-service;1"].getService(Ci.nsIIOService);
|
|
|
|
// Create a test file that we can pass into NetUtil.newURI
|
|
let file = Cc["@mozilla.org/file/directory_service;1"].
|
|
getService(Ci.nsIProperties).
|
|
get("ProfD", Ci.nsIFile);
|
|
file.append("NetUtil-test-file.tmp");
|
|
|
|
// Check that we get the same URI back from the IO service and the utility
|
|
// method.
|
|
let iosURI = ios.newFileURI(file);
|
|
let NetUtilURI = NetUtil.newURI(file);
|
|
do_check_true(iosURI.equals(NetUtilURI));
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_ioService()
|
|
{
|
|
do_check_true(NetUtil.ioService instanceof Ci.nsIIOService);
|
|
run_next_test();
|
|
}
|
|
|
|
function test_asyncFetch_no_channel()
|
|
{
|
|
try {
|
|
NetUtil.asyncFetch(null, function() { });
|
|
do_throw("should throw!");
|
|
}
|
|
catch (e) {
|
|
do_check_eq(e.result, Cr.NS_ERROR_INVALID_ARG);
|
|
}
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_asyncFetch_no_callback()
|
|
{
|
|
try {
|
|
NetUtil.asyncFetch({ });
|
|
do_throw("should throw!");
|
|
}
|
|
catch (e) {
|
|
do_check_eq(e.result, Cr.NS_ERROR_INVALID_ARG);
|
|
}
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_asyncFetch_with_nsIChannel()
|
|
{
|
|
const TEST_DATA = "this is a test string";
|
|
|
|
// Start the http server, and register our handler.
|
|
let server = new nsHttpServer();
|
|
server.registerPathHandler("/test", function(aRequest, aResponse) {
|
|
aResponse.setStatusLine(aRequest.httpVersion, 200, "OK");
|
|
aResponse.setHeader("Content-Type", "text/plain", false);
|
|
aResponse.write(TEST_DATA);
|
|
});
|
|
server.start(4444);
|
|
|
|
// Create our channel.
|
|
let channel = NetUtil.ioService.
|
|
newChannel("http://localhost:4444/test", null, null);
|
|
|
|
// Open our channel asynchronously.
|
|
NetUtil.asyncFetch(channel, function(aInputStream, aResult) {
|
|
// Check that we had success.
|
|
do_check_true(Components.isSuccessCode(aResult));
|
|
|
|
// Check that we got the right data.
|
|
do_check_eq(aInputStream.available(), TEST_DATA.length);
|
|
let is = Cc["@mozilla.org/scriptableinputstream;1"].
|
|
createInstance(Ci.nsIScriptableInputStream);
|
|
is.init(aInputStream);
|
|
let result = is.read(TEST_DATA.length);
|
|
do_check_eq(TEST_DATA, result);
|
|
|
|
server.stop(run_next_test);
|
|
});
|
|
}
|
|
|
|
function test_asyncFetch_with_nsIURI()
|
|
{
|
|
const TEST_DATA = "this is a test string";
|
|
|
|
// Start the http server, and register our handler.
|
|
let server = new nsHttpServer();
|
|
server.registerPathHandler("/test", function(aRequest, aResponse) {
|
|
aResponse.setStatusLine(aRequest.httpVersion, 200, "OK");
|
|
aResponse.setHeader("Content-Type", "text/plain", false);
|
|
aResponse.write(TEST_DATA);
|
|
});
|
|
server.start(4444);
|
|
|
|
// Create our URI.
|
|
let uri = NetUtil.newURI("http://localhost:4444/test");
|
|
|
|
// Open our URI asynchronously.
|
|
NetUtil.asyncFetch(uri, function(aInputStream, aResult) {
|
|
// Check that we had success.
|
|
do_check_true(Components.isSuccessCode(aResult));
|
|
|
|
// Check that we got the right data.
|
|
do_check_eq(aInputStream.available(), TEST_DATA.length);
|
|
let is = Cc["@mozilla.org/scriptableinputstream;1"].
|
|
createInstance(Ci.nsIScriptableInputStream);
|
|
is.init(aInputStream);
|
|
let result = is.read(TEST_DATA.length);
|
|
do_check_eq(TEST_DATA, result);
|
|
|
|
server.stop(run_next_test);
|
|
});
|
|
}
|
|
|
|
function test_asyncFetch_with_string()
|
|
{
|
|
const TEST_DATA = "this is a test string";
|
|
|
|
// Start the http server, and register our handler.
|
|
let server = new nsHttpServer();
|
|
server.registerPathHandler("/test", function(aRequest, aResponse) {
|
|
aResponse.setStatusLine(aRequest.httpVersion, 200, "OK");
|
|
aResponse.setHeader("Content-Type", "text/plain", false);
|
|
aResponse.write(TEST_DATA);
|
|
});
|
|
server.start(4444);
|
|
|
|
// Open our location asynchronously.
|
|
NetUtil.asyncFetch("http://localhost:4444/test", function(aInputStream,
|
|
aResult) {
|
|
// Check that we had success.
|
|
do_check_true(Components.isSuccessCode(aResult));
|
|
|
|
// Check that we got the right data.
|
|
do_check_eq(aInputStream.available(), TEST_DATA.length);
|
|
let is = Cc["@mozilla.org/scriptableinputstream;1"].
|
|
createInstance(Ci.nsIScriptableInputStream);
|
|
is.init(aInputStream);
|
|
let result = is.read(TEST_DATA.length);
|
|
do_check_eq(TEST_DATA, result);
|
|
|
|
server.stop(run_next_test);
|
|
});
|
|
}
|
|
|
|
function test_asyncFetch_with_nsIFile()
|
|
{
|
|
const TEST_DATA = "this is a test string";
|
|
|
|
// First we need a file to read from.
|
|
let file = Cc["@mozilla.org/file/directory_service;1"].
|
|
getService(Ci.nsIProperties).
|
|
get("ProfD", Ci.nsIFile);
|
|
file.append("NetUtil-asyncFetch-test-file.tmp");
|
|
file.createUnique(Ci.nsIFile.NORMAL_FILE_TYPE, 0666);
|
|
|
|
// Write the test data to the file.
|
|
let ostream = Cc["@mozilla.org/network/file-output-stream;1"].
|
|
createInstance(Ci.nsIFileOutputStream);
|
|
ostream.init(file, -1, -1, 0);
|
|
ostream.write(TEST_DATA, TEST_DATA.length);
|
|
|
|
// Sanity check to make sure the data was written.
|
|
do_check_eq(TEST_DATA, getFileContents(file));
|
|
|
|
// Open our file asynchronously.
|
|
NetUtil.asyncFetch(file, function(aInputStream, aResult) {
|
|
// Check that we had success.
|
|
do_check_true(Components.isSuccessCode(aResult));
|
|
|
|
// Check that we got the right data.
|
|
do_check_eq(aInputStream.available(), TEST_DATA.length);
|
|
let is = Cc["@mozilla.org/scriptableinputstream;1"].
|
|
createInstance(Ci.nsIScriptableInputStream);
|
|
is.init(aInputStream);
|
|
let result = is.read(TEST_DATA.length);
|
|
do_check_eq(TEST_DATA, result);
|
|
|
|
run_next_test();
|
|
});
|
|
}
|
|
|
|
function test_asyncFetch_does_not_block()
|
|
{
|
|
// Create our channel that has no data.
|
|
let channel = NetUtil.ioService.
|
|
newChannel("data:text/plain,", null, null);
|
|
|
|
// Open our channel asynchronously.
|
|
NetUtil.asyncFetch(channel, function(aInputStream, aResult) {
|
|
// Check that we had success.
|
|
do_check_true(Components.isSuccessCode(aResult));
|
|
|
|
// Check that reading a byte throws that the stream was closed (as opposed
|
|
// saying it would block).
|
|
let is = Cc["@mozilla.org/scriptableinputstream;1"].
|
|
createInstance(Ci.nsIScriptableInputStream);
|
|
is.init(aInputStream);
|
|
try {
|
|
is.read(1);
|
|
do_throw("should throw!");
|
|
}
|
|
catch (e) {
|
|
do_check_eq(e.result, Cr.NS_BASE_STREAM_CLOSED);
|
|
}
|
|
|
|
run_next_test();
|
|
});
|
|
}
|
|
|
|
function test_newChannel_no_specifier()
|
|
{
|
|
try {
|
|
NetUtil.newChannel();
|
|
do_throw("should throw!");
|
|
}
|
|
catch (e) {
|
|
do_check_eq(e.result, Cr.NS_ERROR_INVALID_ARG);
|
|
}
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_newChannel_with_string()
|
|
{
|
|
const TEST_SPEC = "http://mozilla.org";
|
|
|
|
// Check that we get the same URI back from channel the IO service creates and
|
|
// the channel the utility method creates.
|
|
let ios = NetUtil.ioService;
|
|
let iosChannel = ios.newChannel(TEST_SPEC, null, null);
|
|
let NetUtilChannel = NetUtil.newChannel(TEST_SPEC);
|
|
do_check_true(iosChannel.URI.equals(NetUtilChannel.URI));
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_newChannel_with_nsIURI()
|
|
{
|
|
const TEST_SPEC = "http://mozilla.org";
|
|
|
|
// Check that we get the same URI back from channel the IO service creates and
|
|
// the channel the utility method creates.
|
|
let uri = NetUtil.newURI(TEST_SPEC);
|
|
let iosChannel = NetUtil.ioService.newChannelFromURI(uri);
|
|
let NetUtilChannel = NetUtil.newChannel(uri);
|
|
do_check_true(iosChannel.URI.equals(NetUtilChannel.URI));
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_newChannel_with_nsIFile()
|
|
{
|
|
let file = Cc["@mozilla.org/file/directory_service;1"].
|
|
getService(Ci.nsIProperties).
|
|
get("ProfD", Ci.nsIFile);
|
|
file.append("NetUtil-test-file.tmp");
|
|
|
|
// Check that we get the same URI back from channel the IO service creates and
|
|
// the channel the utility method creates.
|
|
let uri = NetUtil.newURI(file);
|
|
let iosChannel = NetUtil.ioService.newChannelFromURI(uri);
|
|
let NetUtilChannel = NetUtil.newChannel(uri);
|
|
do_check_true(iosChannel.URI.equals(NetUtilChannel.URI));
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_readInputStreamToString()
|
|
{
|
|
const TEST_DATA = "this is a test string\0 with an embedded null";
|
|
let istream = Cc["@mozilla.org/io/string-input-stream;1"].
|
|
createInstance(Ci.nsISupportsCString);
|
|
istream.data = TEST_DATA;
|
|
|
|
do_check_eq(NetUtil.readInputStreamToString(istream, TEST_DATA.length),
|
|
TEST_DATA);
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_readInputStreamToString_no_input_stream()
|
|
{
|
|
try {
|
|
NetUtil.readInputStreamToString("hi", 2);
|
|
do_throw("should throw!");
|
|
}
|
|
catch (e) {
|
|
do_check_eq(e.result, Cr.NS_ERROR_INVALID_ARG);
|
|
}
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_readInputStreamToString_no_bytes_arg()
|
|
{
|
|
const TEST_DATA = "this is a test string";
|
|
let istream = Cc["@mozilla.org/io/string-input-stream;1"].
|
|
createInstance(Ci.nsIStringInputStream);
|
|
istream.setData(TEST_DATA, TEST_DATA.length);
|
|
|
|
try {
|
|
NetUtil.readInputStreamToString(istream);
|
|
do_throw("should throw!");
|
|
}
|
|
catch (e) {
|
|
do_check_eq(e.result, Cr.NS_ERROR_INVALID_ARG);
|
|
}
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
function test_readInputStreamToString_blocking_stream()
|
|
{
|
|
let pipe = Cc["@mozilla.org/pipe;1"].createInstance(Ci.nsIPipe);
|
|
pipe.init(true, true, 0, 0, null);
|
|
|
|
try {
|
|
NetUtil.readInputStreamToString(pipe.inputStream, 10);
|
|
do_throw("should throw!");
|
|
}
|
|
catch (e) {
|
|
do_check_eq(e.result, Cr.NS_BASE_STREAM_WOULD_BLOCK);
|
|
}
|
|
run_next_test();
|
|
}
|
|
|
|
function test_readInputStreamToString_too_many_bytes()
|
|
{
|
|
const TEST_DATA = "this is a test string";
|
|
let istream = Cc["@mozilla.org/io/string-input-stream;1"].
|
|
createInstance(Ci.nsIStringInputStream);
|
|
istream.setData(TEST_DATA, TEST_DATA.length);
|
|
|
|
try {
|
|
NetUtil.readInputStreamToString(istream, TEST_DATA.length + 10);
|
|
do_throw("should throw!");
|
|
}
|
|
catch (e) {
|
|
do_check_eq(e.result, Cr.NS_ERROR_FAILURE);
|
|
}
|
|
|
|
run_next_test();
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
//// Test Runner
|
|
|
|
let tests = [
|
|
test_async_write_file,
|
|
test_async_write_file_nsISafeOutputStream,
|
|
test_newURI_no_spec_throws,
|
|
test_newURI,
|
|
test_newURI_takes_nsIFile,
|
|
test_ioService,
|
|
test_asyncFetch_no_channel,
|
|
test_asyncFetch_no_callback,
|
|
test_asyncFetch_with_nsIChannel,
|
|
test_asyncFetch_with_nsIURI,
|
|
test_asyncFetch_with_string,
|
|
test_asyncFetch_with_nsIFile,
|
|
test_asyncFetch_does_not_block,
|
|
test_newChannel_no_specifier,
|
|
test_newChannel_with_string,
|
|
test_newChannel_with_nsIURI,
|
|
test_newChannel_with_nsIFile,
|
|
test_readInputStreamToString,
|
|
test_readInputStreamToString_no_input_stream,
|
|
test_readInputStreamToString_no_bytes_arg,
|
|
test_readInputStreamToString_blocking_stream,
|
|
test_readInputStreamToString_too_many_bytes,
|
|
];
|
|
let index = 0;
|
|
|
|
function run_next_test()
|
|
{
|
|
if (index < tests.length) {
|
|
do_test_pending();
|
|
|
|
// Asynchronous test exceptions do not kill the test...
|
|
do_execute_soon(function() {
|
|
try {
|
|
print("Running the next test: " + tests[index].name);
|
|
tests[index++]();
|
|
}
|
|
catch (e) {
|
|
do_throw(e);
|
|
}
|
|
});
|
|
}
|
|
|
|
do_test_finished();
|
|
}
|
|
|
|
function run_test()
|
|
{
|
|
do_test_pending();
|
|
run_next_test();
|
|
}
|
|
|