/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 4 -*- * vim: sw=4 ts=4 sts=4 et filetype=javascript * ***** 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 Mozilla 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): * Boris Zbarsky (original author) * Shawn Wilsher * * 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 ***** */ let EXPORTED_SYMBOLS = [ "NetUtil", ]; /** * Necko utilities */ //////////////////////////////////////////////////////////////////////////////// //// Constants const Ci = Components.interfaces; const Cc = Components.classes; const Cr = Components.results; const Cu = Components.utils; const PR_UINT32_MAX = 0xffffffff; //////////////////////////////////////////////////////////////////////////////// //// NetUtil Object const NetUtil = { /** * Function to perform simple async copying from aSource (an input stream) * to aSink (an output stream). The copy will happen on some background * thread. Both streams will be closed when the copy completes. * * @param aSource * The input stream to read from * @param aSink * The output stream to write to * @param aCallback [optional] * A function that will be called at copy completion with a single * argument: the nsresult status code for the copy operation. * * @return An nsIRequest representing the copy operation (for example, this * can be used to cancel the copying). The consumer can ignore the * return value if desired. */ asyncCopy: function NetUtil_asyncCopy(aSource, aSink, aCallback) { if (!aSource || !aSink) { let exception = new Components.Exception( "Must have a source and a sink", Cr.NS_ERROR_INVALID_ARG, Components.stack.caller ); throw exception; } var sourceBuffered = ioUtil.inputStreamIsBuffered(aSource); var sinkBuffered = ioUtil.outputStreamIsBuffered(aSink); var ostream = aSink; if (!sourceBuffered && !sinkBuffered) { // wrap the sink in a buffered stream. ostream = Cc["@mozilla.org/network/buffered-output-stream;1"]. createInstance(Ci.nsIBufferedOutputStream); ostream.init(aSink, 0x8000); sinkBuffered = true; } // make a stream copier var copier = Cc["@mozilla.org/network/async-stream-copier;1"]. createInstance(Ci.nsIAsyncStreamCopier); // Initialize the copier. The 0x8000 should match the size of the // buffer our buffered stream is using, for best performance. If we're // not using our own buffered stream, that's ok too. But maybe we // should just use the default net segment size here? copier.init(aSource, ostream, null, sourceBuffered, sinkBuffered, 0x8000, true, true); var observer; if (aCallback) { observer = { onStartRequest: function(aRequest, aContext) {}, onStopRequest: function(aRequest, aContext, aStatusCode) { aCallback(aStatusCode); } } } else { observer = null; } // start the copying copier.asyncCopy(observer, null); return copier; }, /** * Asynchronously opens a source and fetches the response. A source can be * an nsIURI, nsIFile, string spec, or nsIChannel. The provided callback * will get an input stream containing the response, the result code, and a * reference to the request. * * @param aSource * The nsIURI, nsIFile, string spec, or nsIChannel to open. * Note: If passing an nsIChannel whose notificationCallbacks is * already set, callers are responsible for implementations * of nsIBadCertListener/nsISSLErrorListener. * @param aCallback * The callback function that will be notified upon completion. It * will get two arguments: * 1) An nsIInputStream containing the data from the channel, if any. * 2) The status code from opening the source. * 3) Reference to the channel (as an nsIRequest). */ asyncFetch: function NetUtil_asyncOpen(aSource, aCallback) { if (!aSource || !aCallback) { let exception = new Components.Exception( "Must have a source and a callback", Cr.NS_ERROR_INVALID_ARG, Components.stack.caller ); throw exception; } // Create a pipe that will create our output stream that we can use once // we have gotten all the data. let pipe = Cc["@mozilla.org/pipe;1"]. createInstance(Ci.nsIPipe); pipe.init(true, true, 0, PR_UINT32_MAX, null); // Create a listener that will give data to the pipe's output stream. let listener = Cc["@mozilla.org/network/simple-stream-listener;1"]. createInstance(Ci.nsISimpleStreamListener); listener.init(pipe.outputStream, { onStartRequest: function(aRequest, aContext) {}, onStopRequest: function(aRequest, aContext, aStatusCode) { pipe.outputStream.close(); aCallback(pipe.inputStream, aStatusCode, aRequest); } }); let channel = aSource; if (!(channel instanceof Ci.nsIChannel)) { channel = this.newChannel(aSource); } // Add a BadCertHandler to suppress SSL/cert error dialogs, but only if // the channel doesn't already have a notificationCallbacks. if (!channel.notificationCallbacks) { // Pass true to avoid optional redirect-cert-checking behavior. channel.notificationCallbacks = new BadCertHandler(true); } channel.asyncOpen(listener, null); }, /** * Constructs a new URI for the given spec, character set, and base URI, or * an nsIFile. * * @param aTarget * The string spec for the desired URI or an nsIFile. * @param aOriginCharset [optional] * The character set for the URI. Only used if aTarget is not an * nsIFile. * @param aBaseURI [optional] * The base URI for the spec. Only used if aTarget is not an * nsIFile. * * @return an nsIURI object. */ newURI: function NetUtil_newURI(aTarget, aOriginCharset, aBaseURI) { if (!aTarget) { let exception = new Components.Exception( "Must have a non-null string spec or nsIFile object", Cr.NS_ERROR_INVALID_ARG, Components.stack.caller ); throw exception; } if (aTarget instanceof Ci.nsIFile) { return this.ioService.newFileURI(aTarget); } return this.ioService.newURI(aTarget, aOriginCharset, aBaseURI); }, /** * Constructs a new channel for the given spec, character set, and base URI, * or nsIURI, or nsIFile. * * @param aWhatToLoad * The string spec for the desired URI, an nsIURI, or an nsIFile. * @param aOriginCharset [optional] * The character set for the URI. Only used if aWhatToLoad is a * string. * @param aBaseURI [optional] * The base URI for the spec. Only used if aWhatToLoad is a string. * * @return an nsIChannel object. */ newChannel: function NetUtil_newChannel(aWhatToLoad, aOriginCharset, aBaseURI) { if (!aWhatToLoad) { let exception = new Components.Exception( "Must have a non-null string spec, nsIURI, or nsIFile object", Cr.NS_ERROR_INVALID_ARG, Components.stack.caller ); throw exception; } let uri = aWhatToLoad; if (!(aWhatToLoad instanceof Ci.nsIURI)) { // We either have a string or an nsIFile that we'll need a URI for. uri = this.newURI(aWhatToLoad, aOriginCharset, aBaseURI); } return this.ioService.newChannelFromURI(uri); }, /** * Returns a reference to nsIIOService. * * @return a reference to nsIIOService. */ get ioService() { delete this.ioService; return this.ioService = Cc["@mozilla.org/network/io-service;1"]. getService(Ci.nsIIOService); }, }; //////////////////////////////////////////////////////////////////////////////// //// Initialization Components.utils.import("resource://gre/modules/XPCOMUtils.jsm"); // Define our lazy getters. XPCOMUtils.defineLazyServiceGetter(this, "ioUtil", "@mozilla.org/io-util;1", "nsIIOUtil"); XPCOMUtils.defineLazyGetter(this, "BadCertHandler", function () { var obj = {}; Cu.import("resource://gre/modules/CertUtils.jsm", obj); return obj.BadCertHandler; });