blob: 3eb0842c143a4b8c88609c05606a641aecfa089c [file] [log] [blame]
// Copyright (c) 2011 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "webkit/fileapi/file_system_url_request_job.h"
#include "base/compiler_specific.h"
#include "base/file_util_proxy.h"
#include "base/message_loop.h"
#include "base/platform_file.h"
#include "base/threading/thread_restrictions.h"
#include "build/build_config.h"
#include "googleurl/src/gurl.h"
#include "net/base/file_stream.h"
#include "net/base/io_buffer.h"
#include "net/base/mime_util.h"
#include "net/base/net_errors.h"
#include "net/base/net_util.h"
#include "net/http/http_util.h"
#include "net/url_request/url_request.h"
#include "webkit/fileapi/file_system_path_manager.h"
#include "webkit/fileapi/file_system_util.h"
using net::URLRequest;
using net::URLRequestJob;
using net::URLRequestStatus;
namespace fileapi {
static const int kFileFlags = base::PLATFORM_FILE_OPEN |
base::PLATFORM_FILE_READ |
base::PLATFORM_FILE_ASYNC;
FileSystemURLRequestJob::FileSystemURLRequestJob(
URLRequest* request, FileSystemPathManager* path_manager,
scoped_refptr<base::MessageLoopProxy> file_thread_proxy)
: URLRequestJob(request),
path_manager_(path_manager),
ALLOW_THIS_IN_INITIALIZER_LIST(
io_callback_(this, &FileSystemURLRequestJob::DidRead)),
stream_(NULL),
is_directory_(false),
remaining_bytes_(0),
ALLOW_THIS_IN_INITIALIZER_LIST(method_factory_(this)),
ALLOW_THIS_IN_INITIALIZER_LIST(callback_factory_(this)),
file_thread_proxy_(file_thread_proxy) {
}
FileSystemURLRequestJob::~FileSystemURLRequestJob() {
// Since we use the two-arg constructor of FileStream, we need to call Close()
// manually: ~FileStream won't call it for us.
if (stream_ != NULL)
stream_->Close();
}
void FileSystemURLRequestJob::Start() {
MessageLoop::current()->PostTask(FROM_HERE,
method_factory_.NewRunnableMethod(
&FileSystemURLRequestJob::StartAsync));
}
void FileSystemURLRequestJob::Kill() {
if (stream_ != NULL) {
stream_->Close();
stream_.reset(NULL);
}
URLRequestJob::Kill();
callback_factory_.RevokeAll();
}
bool FileSystemURLRequestJob::ReadRawData(net::IOBuffer* dest, int dest_size,
int *bytes_read) {
DCHECK_NE(dest_size, 0);
DCHECK(bytes_read);
DCHECK_GE(remaining_bytes_, 0);
if (stream_ == NULL)
return false;
if (remaining_bytes_ < dest_size)
dest_size = static_cast<int>(remaining_bytes_);
if (!dest_size) {
*bytes_read = 0;
return true;
}
int rv = stream_->Read(dest->data(), dest_size, &io_callback_);
if (rv >= 0) {
// Data is immediately available.
*bytes_read = rv;
remaining_bytes_ -= rv;
DCHECK_GE(remaining_bytes_, 0);
return true;
}
// Otherwise, a read error occured. We may just need to wait...
if (rv == net::ERR_IO_PENDING)
SetStatus(URLRequestStatus(URLRequestStatus::IO_PENDING, 0));
else
NotifyFailed(rv);
return false;
}
bool FileSystemURLRequestJob::GetMimeType(std::string* mime_type) const {
// URL requests should not block on the disk! On Windows this goes to the
// registry.
// http://code.google.com/p/chromium/issues/detail?id=59849
base::ThreadRestrictions::ScopedAllowIO allow_io;
DCHECK(request_);
return net::GetMimeTypeFromFile(absolute_file_path_, mime_type);
}
void FileSystemURLRequestJob::SetExtraRequestHeaders(
const net::HttpRequestHeaders& headers) {
std::string range_header;
if (headers.GetHeader(net::HttpRequestHeaders::kRange, &range_header)) {
std::vector<net::HttpByteRange> ranges;
if (net::HttpUtil::ParseRangeHeader(range_header, &ranges)) {
if (ranges.size() == 1)
byte_range_ = ranges[0];
else {
// We don't support multiple range requests in one single URL request.
// TODO(adamk): decide whether we want to support multiple range
// requests.
NotifyFailed(net::ERR_REQUEST_RANGE_NOT_SATISFIABLE);
}
}
}
}
void FileSystemURLRequestJob::StartAsync() {
GURL origin_url;
FileSystemType type;
if (!CrackFileSystemURL(request_->url(), &origin_url, &type,
&relative_file_path_)) {
NotifyFailed(net::ERR_INVALID_URL);
return;
}
path_manager_->GetFileSystemRootPath(
origin_url, type, false, // create
callback_factory_.NewCallback(&FileSystemURLRequestJob::DidGetRootPath));
}
void FileSystemURLRequestJob::DidGetRootPath(bool success,
const FilePath& root_path,
const std::string& name) {
if (!success) {
NotifyFailed(net::ERR_FILE_NOT_FOUND);
return;
}
absolute_file_path_ = root_path.Append(relative_file_path_);
base::FileUtilProxy::GetFileInfo(file_thread_proxy_, absolute_file_path_,
callback_factory_.NewCallback(&FileSystemURLRequestJob::DidResolve));
}
void FileSystemURLRequestJob::DidResolve(base::PlatformFileError error_code,
const base::PlatformFileInfo& file_info) {
// We may have been orphaned...
if (!request_)
return;
// We use FileSystemURLRequestJob to handle files as well as directories
// without trailing slash.
// If a directory does not exist, we return ERR_FILE_NOT_FOUND. Otherwise,
// we will append trailing slash and redirect to FileDirJob.
if (error_code != base::PLATFORM_FILE_OK) {
NotifyFailed(error_code);
return;
}
is_directory_ = file_info.is_directory;
if (!byte_range_.ComputeBounds(file_info.size)) {
NotifyFailed(net::ERR_REQUEST_RANGE_NOT_SATISFIABLE);
return;
}
if (!is_directory_) {
base::FileUtilProxy::CreateOrOpen(
file_thread_proxy_, absolute_file_path_, kFileFlags,
callback_factory_.NewCallback(&FileSystemURLRequestJob::DidOpen));
} else
NotifyHeadersComplete();
}
void FileSystemURLRequestJob::DidOpen(base::PlatformFileError error_code,
base::PassPlatformFile file,
bool created) {
if (error_code != base::PLATFORM_FILE_OK) {
NotifyFailed(error_code);
return;
}
stream_.reset(new net::FileStream(file.ReleaseValue(), kFileFlags));
remaining_bytes_ = byte_range_.last_byte_position() -
byte_range_.first_byte_position() + 1;
DCHECK_GE(remaining_bytes_, 0);
// Do the seek at the beginning of the request.
if (remaining_bytes_ > 0 &&
byte_range_.first_byte_position() != 0 &&
byte_range_.first_byte_position() !=
stream_->Seek(net::FROM_BEGIN, byte_range_.first_byte_position())) {
NotifyFailed(net::ERR_REQUEST_RANGE_NOT_SATISFIABLE);
return;
}
set_expected_content_size(remaining_bytes_);
NotifyHeadersComplete();
}
void FileSystemURLRequestJob::DidRead(int result) {
if (result > 0)
SetStatus(URLRequestStatus()); // Clear the IO_PENDING status
else if (result == 0)
NotifyDone(URLRequestStatus());
else
NotifyFailed(result);
remaining_bytes_ -= result;
DCHECK_GE(remaining_bytes_, 0);
NotifyReadComplete(result);
}
bool FileSystemURLRequestJob::IsRedirectResponse(GURL* location,
int* http_status_code) {
if (is_directory_) {
// This happens when we discovered the file is a directory, so needs a
// slash at the end of the path.
std::string new_path = request_->url().path();
new_path.push_back('/');
GURL::Replacements replacements;
replacements.SetPathStr(new_path);
*location = request_->url().ReplaceComponents(replacements);
*http_status_code = 301; // simulate a permanent redirect
return true;
}
return false;
}
void FileSystemURLRequestJob::NotifyFailed(int rv) {
NotifyDone(URLRequestStatus(URLRequestStatus::FAILED, rv));
}
} // namespace fileapi