// Copyright (c) 2012 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. // For loading files, we make use of overlapped i/o to ensure that reading from // the filesystem (e.g., a network filesystem) does not block the calling // thread. An alternative approach would be to use a background thread or pool // of threads, but it seems better to leverage the operating system's ability // to do background file reads for us. // // Since overlapped reads require a 'static' buffer for the duration of the // asynchronous read, the URLRequestFileJob keeps a buffer as a member var. In // URLRequestFileJob::Read, data is simply copied from the object's buffer into // the given buffer. If there is no data to copy, the URLRequestFileJob // attempts to read more from the file to fill its buffer. If reading from the // file does not complete synchronously, then the URLRequestFileJob waits for a // signal from the OS that the overlapped read has completed. It does so by // leveraging the MessageLoop::WatchObject API. #include "net/url_request/url_request_file_job.h" #include "base/bind.h" #include "base/compiler_specific.h" #include "base/files/file_util.h" #include "base/strings/string_util.h" #include "base/synchronization/lock.h" #include "base/task_runner.h" #include "base/threading/thread_restrictions.h" #include "build/build_config.h" #include "net/base/file_stream.h" #include "net/base/filename_util.h" #include "net/base/io_buffer.h" #include "net/base/load_flags.h" #include "net/base/mime_util.h" #include "net/filter/gzip_source_stream.h" #include "net/filter/source_stream.h" #include "net/http/http_util.h" #include "net/url_request/url_request_error_job.h" #include "net/url_request/url_request_file_dir_job.h" #include "url/gurl.h" #if defined(OS_WIN) #include "base/win/shortcut.h" #endif namespace net { URLRequestFileJob::FileMetaInfo::FileMetaInfo() : file_size(0), mime_type_result(false), file_exists(false), is_directory(false) { } URLRequestFileJob::URLRequestFileJob( URLRequest* request, NetworkDelegate* network_delegate, const base::FilePath& file_path, const scoped_refptr& file_task_runner) : URLRequestJob(request, network_delegate), file_path_(file_path), stream_(new FileStream(file_task_runner)), file_task_runner_(file_task_runner), remaining_bytes_(0), range_parse_result_(OK), weak_ptr_factory_(this) {} void URLRequestFileJob::Start() { FileMetaInfo* meta_info = new FileMetaInfo(); file_task_runner_->PostTaskAndReply( FROM_HERE, base::Bind(&URLRequestFileJob::FetchMetaInfo, file_path_, base::Unretained(meta_info)), base::Bind(&URLRequestFileJob::DidFetchMetaInfo, weak_ptr_factory_.GetWeakPtr(), base::Owned(meta_info))); } void URLRequestFileJob::Kill() { stream_.reset(); weak_ptr_factory_.InvalidateWeakPtrs(); URLRequestJob::Kill(); } int URLRequestFileJob::ReadRawData(IOBuffer* dest, int dest_size) { DCHECK_NE(dest_size, 0); DCHECK_GE(remaining_bytes_, 0); if (remaining_bytes_ < dest_size) dest_size = remaining_bytes_; // If we should copy zero bytes because |remaining_bytes_| is zero, short // circuit here. if (!dest_size) return 0; int rv = stream_->Read( dest, dest_size, base::Bind(&URLRequestFileJob::DidRead, weak_ptr_factory_.GetWeakPtr(), base::WrapRefCounted(dest))); if (rv >= 0) { remaining_bytes_ -= rv; DCHECK_GE(remaining_bytes_, 0); } return rv; } bool URLRequestFileJob::IsRedirectResponse(GURL* location, int* http_status_code) { if (meta_info_.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; } #if defined(OS_WIN) // Follow a Windows shortcut. // We just resolve .lnk file, ignore others. if (!base::LowerCaseEqualsASCII(file_path_.Extension(), ".lnk")) return false; base::FilePath new_path = file_path_; bool resolved; resolved = base::win::ResolveShortcut(new_path, &new_path, NULL); // If shortcut is not resolved successfully, do not redirect. if (!resolved) return false; *location = FilePathToFileURL(new_path); *http_status_code = 301; return true; #else return false; #endif } bool URLRequestFileJob::GetMimeType(std::string* mime_type) const { DCHECK(request_); if (meta_info_.mime_type_result) { *mime_type = meta_info_.mime_type; return true; } return false; } void URLRequestFileJob::SetExtraRequestHeaders( const HttpRequestHeaders& headers) { std::string range_header; if (headers.GetHeader(HttpRequestHeaders::kRange, &range_header)) { // This job only cares about the Range header. This method stashes the value // for later use in DidOpen(), which is responsible for some of the range // validation as well. NotifyStartError is not legal to call here since // the job has not started. std::vector ranges; if (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, // because we need to do multipart encoding here. // TODO(hclam): decide whether we want to support multiple range // requests. range_parse_result_ = ERR_REQUEST_RANGE_NOT_SATISFIABLE; } } } } void URLRequestFileJob::OnOpenComplete(int result) {} void URLRequestFileJob::OnSeekComplete(int64_t result) {} void URLRequestFileJob::OnReadComplete(IOBuffer* buf, int result) { } URLRequestFileJob::~URLRequestFileJob() = default; std::unique_ptr URLRequestFileJob::SetUpSourceStream() { std::unique_ptr source = URLRequestJob::SetUpSourceStream(); if (!base::LowerCaseEqualsASCII(file_path_.Extension(), ".svgz")) return source; return GzipSourceStream::Create(std::move(source), SourceStream::TYPE_GZIP); } bool URLRequestFileJob::CanAccessFile(const base::FilePath& original_path, const base::FilePath& absolute_path) { return !network_delegate() || network_delegate()->CanAccessFile( *request(), original_path, absolute_path); } void URLRequestFileJob::FetchMetaInfo(const base::FilePath& file_path, FileMetaInfo* meta_info) { base::File::Info file_info; meta_info->file_exists = base::GetFileInfo(file_path, &file_info); if (meta_info->file_exists) { meta_info->file_size = file_info.size; meta_info->is_directory = file_info.is_directory; } // On Windows GetMimeTypeFromFile() goes to the registry. Thus it should be // done in WorkerPool. meta_info->mime_type_result = GetMimeTypeFromFile(file_path, &meta_info->mime_type); meta_info->absolute_path = base::MakeAbsoluteFilePath(file_path); } void URLRequestFileJob::DidFetchMetaInfo(const FileMetaInfo* meta_info) { meta_info_ = *meta_info; // We use URLRequestFileJob 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. // A special case is "\" on Windows. We should resolve as invalid. // However, Windows resolves "\" to "C:\", thus reports it as existent. // So what happens is we append it with trailing slash and redirect it to // FileDirJob where it is resolved as invalid. if (!meta_info_.file_exists) { DidOpen(ERR_FILE_NOT_FOUND); return; } if (meta_info_.is_directory) { DidOpen(OK); return; } if (!CanAccessFile(file_path_, meta_info->absolute_path)) { DidOpen(ERR_ACCESS_DENIED); return; } int flags = base::File::FLAG_OPEN | base::File::FLAG_READ | base::File::FLAG_ASYNC; int rv = stream_->Open(file_path_, flags, base::Bind(&URLRequestFileJob::DidOpen, weak_ptr_factory_.GetWeakPtr())); if (rv != ERR_IO_PENDING) DidOpen(rv); } void URLRequestFileJob::DidOpen(int result) { OnOpenComplete(result); if (result != OK) { NotifyStartError(URLRequestStatus(URLRequestStatus::FAILED, result)); return; } if (range_parse_result_ != OK || !byte_range_.ComputeBounds(meta_info_.file_size)) { DidSeek(ERR_REQUEST_RANGE_NOT_SATISFIABLE); return; } remaining_bytes_ = byte_range_.last_byte_position() - byte_range_.first_byte_position() + 1; DCHECK_GE(remaining_bytes_, 0); if (remaining_bytes_ > 0 && byte_range_.first_byte_position() != 0) { int rv = stream_->Seek(byte_range_.first_byte_position(), base::Bind(&URLRequestFileJob::DidSeek, weak_ptr_factory_.GetWeakPtr())); if (rv != ERR_IO_PENDING) DidSeek(ERR_REQUEST_RANGE_NOT_SATISFIABLE); } else { // We didn't need to call stream_->Seek() at all, so we pass to DidSeek() // the value that would mean seek success. This way we skip the code // handling seek failure. DidSeek(byte_range_.first_byte_position()); } } void URLRequestFileJob::DidSeek(int64_t result) { DCHECK(result < 0 || result == byte_range_.first_byte_position()); OnSeekComplete(result); if (result < 0) { NotifyStartError(URLRequestStatus(URLRequestStatus::FAILED, ERR_REQUEST_RANGE_NOT_SATISFIABLE)); return; } set_expected_content_size(remaining_bytes_); NotifyHeadersComplete(); } void URLRequestFileJob::DidRead(scoped_refptr buf, int result) { if (result >= 0) { remaining_bytes_ -= result; DCHECK_GE(remaining_bytes_, 0); } OnReadComplete(buf.get(), result); buf = NULL; ReadRawDataComplete(result); } } // namespace net