зеркало из https://github.com/mozilla/gecko-dev.git
446 строки
14 KiB
C++
446 строки
14 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim:set ts=2 sw=2 sts=2 et cin: */
|
|
/* ***** 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.org code.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Netscape Communications Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 1998
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Darin Fisher <darin@meer.net>
|
|
*
|
|
* 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 ***** */
|
|
|
|
#include "nsIOService.h"
|
|
#include "nsFileChannel.h"
|
|
#include "nsBaseContentStream.h"
|
|
#include "nsDirectoryIndexStream.h"
|
|
#include "nsThreadUtils.h"
|
|
#include "nsTransportUtils.h"
|
|
#include "nsStreamUtils.h"
|
|
#include "nsURLHelper.h"
|
|
#include "nsMimeTypes.h"
|
|
#include "nsNetUtil.h"
|
|
#include "nsProxyRelease.h"
|
|
#include "nsAutoPtr.h"
|
|
#include "nsStandardURL.h"
|
|
|
|
#include "nsIFileURL.h"
|
|
#include "nsIMIMEService.h"
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
class nsFileCopyEvent : public nsRunnable {
|
|
public:
|
|
nsFileCopyEvent(nsIOutputStream *dest, nsIInputStream *source, PRInt64 len)
|
|
: mDest(dest)
|
|
, mSource(source)
|
|
, mLen(len)
|
|
, mStatus(NS_OK)
|
|
, mInterruptStatus(NS_OK) {
|
|
}
|
|
|
|
// Read the current status of the file copy operation.
|
|
nsresult Status() { return mStatus; }
|
|
|
|
// Call this method to perform the file copy synchronously.
|
|
void DoCopy();
|
|
|
|
// Call this method to perform the file copy on a background thread. The
|
|
// callback is dispatched when the file copy completes.
|
|
nsresult Dispatch(nsIRunnable *callback,
|
|
nsITransportEventSink *sink,
|
|
nsIEventTarget *target);
|
|
|
|
// Call this method to interrupt a file copy operation that is occuring on
|
|
// a background thread. The status parameter passed to this function must
|
|
// be a failure code and is set as the status of this file copy operation.
|
|
void Interrupt(nsresult status) {
|
|
NS_ASSERTION(NS_FAILED(status), "must be a failure code");
|
|
mInterruptStatus = status;
|
|
}
|
|
|
|
NS_IMETHOD Run() {
|
|
DoCopy();
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
nsCOMPtr<nsIEventTarget> mCallbackTarget;
|
|
nsCOMPtr<nsIRunnable> mCallback;
|
|
nsCOMPtr<nsITransportEventSink> mSink;
|
|
nsCOMPtr<nsIOutputStream> mDest;
|
|
nsCOMPtr<nsIInputStream> mSource;
|
|
PRInt64 mLen;
|
|
nsresult mStatus; // modified on i/o thread only
|
|
nsresult mInterruptStatus; // modified on main thread only
|
|
};
|
|
|
|
void
|
|
nsFileCopyEvent::DoCopy()
|
|
{
|
|
// We'll copy in chunks this large by default. This size affects how
|
|
// frequently we'll check for interrupts.
|
|
const PRInt32 chunk = nsIOService::gDefaultSegmentSize * nsIOService::gDefaultSegmentCount;
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
PRInt64 len = mLen, progress = 0;
|
|
while (len) {
|
|
// If we've been interrupted, then stop copying.
|
|
rv = mInterruptStatus;
|
|
if (NS_FAILED(rv))
|
|
break;
|
|
|
|
PRInt32 num = NS_MIN((PRInt32) len, chunk);
|
|
|
|
PRUint32 result;
|
|
rv = mSource->ReadSegments(NS_CopySegmentToStream, mDest, num, &result);
|
|
if (NS_FAILED(rv))
|
|
break;
|
|
if (result != (PRUint32) num) {
|
|
rv = NS_ERROR_FILE_DISK_FULL; // stopped prematurely (out of disk space)
|
|
break;
|
|
}
|
|
|
|
// Dispatch progress notification
|
|
if (mSink) {
|
|
progress += num;
|
|
mSink->OnTransportStatus(nsnull, nsITransport::STATUS_WRITING, progress,
|
|
mLen);
|
|
}
|
|
|
|
len -= num;
|
|
}
|
|
|
|
if (NS_FAILED(rv))
|
|
mStatus = rv;
|
|
|
|
// Close the output stream before notifying our callback so that others may
|
|
// freely "play" with the file.
|
|
mDest->Close();
|
|
|
|
// Notify completion
|
|
if (mCallback) {
|
|
mCallbackTarget->Dispatch(mCallback, NS_DISPATCH_NORMAL);
|
|
|
|
// Release the callback on the target thread to avoid destroying stuff on
|
|
// the wrong thread.
|
|
nsIRunnable *doomed = nsnull;
|
|
mCallback.swap(doomed);
|
|
NS_ProxyRelease(mCallbackTarget, doomed);
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
nsFileCopyEvent::Dispatch(nsIRunnable *callback,
|
|
nsITransportEventSink *sink,
|
|
nsIEventTarget *target)
|
|
{
|
|
// Use the supplied event target for all asynchronous operations.
|
|
|
|
mCallback = callback;
|
|
mCallbackTarget = target;
|
|
|
|
// Build a coalescing proxy for progress events
|
|
nsresult rv = net_NewTransportEventSinkProxy(getter_AddRefs(mSink), sink,
|
|
target, PR_TRUE);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
// Dispatch ourselves to I/O thread pool...
|
|
nsCOMPtr<nsIEventTarget> pool =
|
|
do_GetService(NS_STREAMTRANSPORTSERVICE_CONTRACTID, &rv);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
return pool->Dispatch(this, NS_DISPATCH_NORMAL);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// This is a dummy input stream that when read, performs the file copy. The
|
|
// copy happens on a background thread via mCopyEvent.
|
|
|
|
class nsFileUploadContentStream : public nsBaseContentStream {
|
|
public:
|
|
NS_DECL_ISUPPORTS_INHERITED
|
|
|
|
nsFileUploadContentStream(PRBool nonBlocking,
|
|
nsIOutputStream *dest,
|
|
nsIInputStream *source,
|
|
PRInt64 len,
|
|
nsITransportEventSink *sink)
|
|
: nsBaseContentStream(nonBlocking)
|
|
, mCopyEvent(new nsFileCopyEvent(dest, source, len))
|
|
, mSink(sink) {
|
|
}
|
|
|
|
PRBool IsInitialized() {
|
|
return mCopyEvent != nsnull;
|
|
}
|
|
|
|
NS_IMETHODIMP ReadSegments(nsWriteSegmentFun fun, void *closure,
|
|
PRUint32 count, PRUint32 *result);
|
|
NS_IMETHODIMP AsyncWait(nsIInputStreamCallback *callback, PRUint32 flags,
|
|
PRUint32 count, nsIEventTarget *target);
|
|
|
|
private:
|
|
void OnCopyComplete();
|
|
|
|
nsRefPtr<nsFileCopyEvent> mCopyEvent;
|
|
nsCOMPtr<nsITransportEventSink> mSink;
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS_INHERITED0(nsFileUploadContentStream,
|
|
nsBaseContentStream)
|
|
|
|
NS_IMETHODIMP
|
|
nsFileUploadContentStream::ReadSegments(nsWriteSegmentFun fun, void *closure,
|
|
PRUint32 count, PRUint32 *result)
|
|
{
|
|
*result = 0; // nothing is ever actually read from this stream
|
|
|
|
if (IsClosed())
|
|
return NS_OK;
|
|
|
|
if (IsNonBlocking()) {
|
|
// Inform the caller that they will have to wait for the copy operation to
|
|
// complete asynchronously. We'll kick of the copy operation once they
|
|
// call AsyncWait.
|
|
return NS_BASE_STREAM_WOULD_BLOCK;
|
|
}
|
|
|
|
// Perform copy synchronously, and then close out the stream.
|
|
mCopyEvent->DoCopy();
|
|
nsresult status = mCopyEvent->Status();
|
|
CloseWithStatus(NS_FAILED(status) ? status : NS_BASE_STREAM_CLOSED);
|
|
return status;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsFileUploadContentStream::AsyncWait(nsIInputStreamCallback *callback,
|
|
PRUint32 flags, PRUint32 count,
|
|
nsIEventTarget *target)
|
|
{
|
|
nsresult rv = nsBaseContentStream::AsyncWait(callback, flags, count, target);
|
|
if (NS_FAILED(rv) || IsClosed())
|
|
return rv;
|
|
|
|
if (IsNonBlocking()) {
|
|
nsCOMPtr<nsIRunnable> callback =
|
|
NS_NewRunnableMethod(this, &nsFileUploadContentStream::OnCopyComplete);
|
|
mCopyEvent->Dispatch(callback, mSink, target);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
nsFileUploadContentStream::OnCopyComplete()
|
|
{
|
|
// This method is being called to indicate that we are done copying.
|
|
nsresult status = mCopyEvent->Status();
|
|
|
|
CloseWithStatus(NS_FAILED(status) ? status : NS_BASE_STREAM_CLOSED);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
nsresult
|
|
nsFileChannel::MakeFileInputStream(nsIFile *file,
|
|
nsCOMPtr<nsIInputStream> &stream,
|
|
nsCString &contentType)
|
|
{
|
|
// we accept that this might result in a disk hit to stat the file
|
|
PRBool isDir;
|
|
nsresult rv = file->IsDirectory(&isDir);
|
|
if (NS_FAILED(rv)) {
|
|
// canonicalize error message
|
|
if (rv == NS_ERROR_FILE_TARGET_DOES_NOT_EXIST)
|
|
rv = NS_ERROR_FILE_NOT_FOUND;
|
|
return rv;
|
|
}
|
|
|
|
if (isDir) {
|
|
rv = nsDirectoryIndexStream::Create(file, getter_AddRefs(stream));
|
|
if (NS_SUCCEEDED(rv) && !HasContentTypeHint())
|
|
contentType.AssignLiteral(APPLICATION_HTTP_INDEX_FORMAT);
|
|
} else {
|
|
rv = NS_NewLocalFileInputStream(getter_AddRefs(stream), file);
|
|
if (NS_SUCCEEDED(rv) && !HasContentTypeHint()) {
|
|
// Use file extension to infer content type
|
|
nsCOMPtr<nsIMIMEService> mime = do_GetService("@mozilla.org/mime;1", &rv);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
mime->GetTypeFromFile(file, contentType);
|
|
}
|
|
}
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
nsresult
|
|
nsFileChannel::OpenContentStream(PRBool async, nsIInputStream **result,
|
|
nsIChannel** channel)
|
|
{
|
|
// NOTE: the resulting file is a clone, so it is safe to pass it to the
|
|
// file input stream which will be read on a background thread.
|
|
nsCOMPtr<nsIFile> file;
|
|
nsresult rv = GetFile(getter_AddRefs(file));
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
nsCOMPtr<nsIFileProtocolHandler> fileHandler;
|
|
rv = NS_GetFileProtocolHandler(getter_AddRefs(fileHandler));
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
nsCOMPtr<nsIURI> newURI;
|
|
rv = fileHandler->ReadURLFile(file, getter_AddRefs(newURI));
|
|
if (NS_SUCCEEDED(rv)) {
|
|
nsCOMPtr<nsIChannel> newChannel;
|
|
rv = NS_NewChannel(getter_AddRefs(newChannel), newURI);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
*result = nsnull;
|
|
newChannel.forget(channel);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIInputStream> stream;
|
|
|
|
if (mUploadStream) {
|
|
// Pass back a nsFileUploadContentStream instance that knows how to perform
|
|
// the file copy when "read" (the resulting stream in this case does not
|
|
// actually return any data).
|
|
|
|
nsCOMPtr<nsIOutputStream> fileStream;
|
|
rv = NS_NewLocalFileOutputStream(getter_AddRefs(fileStream), file,
|
|
PR_WRONLY | PR_CREATE_FILE | PR_TRUNCATE,
|
|
PR_IRUSR | PR_IWUSR);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
nsFileUploadContentStream *uploadStream =
|
|
new nsFileUploadContentStream(async, fileStream, mUploadStream,
|
|
mUploadLength, this);
|
|
if (!uploadStream || !uploadStream->IsInitialized()) {
|
|
delete uploadStream;
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
stream = uploadStream;
|
|
|
|
SetContentLength64(0);
|
|
|
|
// Since there isn't any content to speak of we just set the content-type
|
|
// to something other than "unknown" to avoid triggering the content-type
|
|
// sniffer code in nsBaseChannel.
|
|
// However, don't override explicitly set types.
|
|
if (!HasContentTypeHint())
|
|
SetContentType(NS_LITERAL_CSTRING(APPLICATION_OCTET_STREAM));
|
|
} else {
|
|
nsCAutoString contentType;
|
|
nsresult rv = MakeFileInputStream(file, stream, contentType);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
EnableSynthesizedProgressEvents(PR_TRUE);
|
|
|
|
// fixup content length and type
|
|
if (ContentLength64() < 0) {
|
|
PRInt64 size;
|
|
rv = file->GetFileSize(&size);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
SetContentLength64(size);
|
|
}
|
|
if (!contentType.IsEmpty())
|
|
SetContentType(contentType);
|
|
}
|
|
|
|
*result = nsnull;
|
|
stream.swap(*result);
|
|
return NS_OK;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// nsFileChannel::nsISupports
|
|
|
|
NS_IMPL_ISUPPORTS_INHERITED2(nsFileChannel,
|
|
nsBaseChannel,
|
|
nsIUploadChannel,
|
|
nsIFileChannel)
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// nsFileChannel::nsIFileChannel
|
|
|
|
NS_IMETHODIMP
|
|
nsFileChannel::GetFile(nsIFile **file)
|
|
{
|
|
nsCOMPtr<nsIFileURL> fileURL = do_QueryInterface(URI());
|
|
NS_ENSURE_STATE(fileURL);
|
|
|
|
// This returns a cloned nsIFile
|
|
return fileURL->GetFile(file);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// nsFileChannel::nsIUploadChannel
|
|
|
|
NS_IMETHODIMP
|
|
nsFileChannel::SetUploadStream(nsIInputStream *stream,
|
|
const nsACString &contentType,
|
|
PRInt32 contentLength)
|
|
{
|
|
NS_ENSURE_TRUE(!IsPending(), NS_ERROR_IN_PROGRESS);
|
|
|
|
if ((mUploadStream = stream)) {
|
|
mUploadLength = contentLength;
|
|
if (mUploadLength < 0) {
|
|
// Make sure we know how much data we are uploading.
|
|
PRUint32 avail;
|
|
nsresult rv = mUploadStream->Available(&avail);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
mUploadLength = avail;
|
|
}
|
|
} else {
|
|
mUploadLength = -1;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsFileChannel::GetUploadStream(nsIInputStream **result)
|
|
{
|
|
NS_IF_ADDREF(*result = mUploadStream);
|
|
return NS_OK;
|
|
}
|