blob: 23f87ce5b7e574946f9e6ff7de3e1aa4b41264a0 [file] [log] [blame]
// 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.
#include "webkit/fileapi/file_system_operation.h"
#include "base/bind.h"
#include "base/time.h"
#include "base/utf_string_conversions.h"
#include "net/base/escape.h"
#include "net/url_request/url_request_context.h"
#include "webkit/blob/shareable_file_reference.h"
#include "webkit/fileapi/file_system_context.h"
#include "webkit/fileapi/file_system_file_util_proxy.h"
#include "webkit/fileapi/file_system_mount_point_provider.h"
#include "webkit/fileapi/file_system_operation_context.h"
#include "webkit/fileapi/file_system_quota_util.h"
#include "webkit/fileapi/file_system_types.h"
#include "webkit/fileapi/file_system_util.h"
#include "webkit/fileapi/file_writer_delegate.h"
#include "webkit/quota/quota_manager.h"
#include "webkit/quota/quota_types.h"
namespace fileapi {
namespace {
void GetMetadataForSnapshot(
const FileSystemOperationInterface::SnapshotFileCallback& callback,
base::PlatformFileError result,
const base::PlatformFileInfo& file_info,
const FilePath& platform_path) {
// We don't want the third party to delete our local file, so just returning
// NULL as |file_reference|.
callback.Run(result, file_info, platform_path, NULL);
}
} // anonymous namespace
class FileSystemOperation::ScopedQuotaNotifier {
public:
ScopedQuotaNotifier(FileSystemContext* context,
const GURL& origin_url,
FileSystemType type);
~ScopedQuotaNotifier();
private:
// Not owned; owned by the owner of this instance (i.e. FileSystemOperation).
FileSystemQuotaUtil* quota_util_;
const GURL origin_url_;
FileSystemType type_;
DISALLOW_COPY_AND_ASSIGN(ScopedQuotaNotifier);
};
FileSystemOperation::ScopedQuotaNotifier::ScopedQuotaNotifier(
FileSystemContext* context, const GURL& origin_url, FileSystemType type)
: origin_url_(origin_url), type_(type) {
DCHECK(context);
DCHECK(type_ != kFileSystemTypeUnknown);
quota_util_ = context->GetQuotaUtil(type_);
if (quota_util_) {
DCHECK(quota_util_->proxy());
quota_util_->proxy()->StartUpdateOrigin(origin_url_, type_);
}
}
FileSystemOperation::ScopedQuotaNotifier::~ScopedQuotaNotifier() {
if (quota_util_) {
DCHECK(quota_util_->proxy());
quota_util_->proxy()->EndUpdateOrigin(origin_url_, type_);
}
}
FileSystemOperation::TaskParamsForDidGetQuota::TaskParamsForDidGetQuota()
: type(kFileSystemTypeUnknown) {
}
FileSystemOperation::TaskParamsForDidGetQuota::~TaskParamsForDidGetQuota() {}
FileSystemOperation::~FileSystemOperation() {
if (file_writer_delegate_.get()) {
FileSystemOperationContext* c =
new FileSystemOperationContext(operation_context_);
base::FileUtilProxy::RelayClose(
proxy_,
base::Bind(&FileSystemFileUtil::Close,
base::Unretained(src_util_),
base::Owned(c)),
file_writer_delegate_->file(),
base::FileUtilProxy::StatusCallback());
}
}
void FileSystemOperation::CreateFile(const GURL& path_url,
bool exclusive,
const StatusCallback& callback) {
DCHECK(SetPendingOperationType(kOperationCreateFile));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_CREATE);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result);
delete this;
return;
}
GetUsageAndQuotaThenRunTask(
src_path_.origin(), src_path_.type(),
base::Bind(&FileSystemOperation::DoCreateFile,
base::Unretained(this), callback, exclusive),
base::Bind(callback, base::PLATFORM_FILE_ERROR_FAILED));
}
void FileSystemOperation::CreateDirectory(const GURL& path_url,
bool exclusive,
bool recursive,
const StatusCallback& callback) {
DCHECK(SetPendingOperationType(kOperationCreateDirectory));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_CREATE);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result);
delete this;
return;
}
GetUsageAndQuotaThenRunTask(
src_path_.origin(), src_path_.type(),
base::Bind(&FileSystemOperation::DoCreateDirectory,
base::Unretained(this), callback, exclusive, recursive),
base::Bind(callback, base::PLATFORM_FILE_ERROR_FAILED));
}
void FileSystemOperation::Copy(const GURL& src_path_url,
const GURL& dest_path_url,
const StatusCallback& callback) {
DCHECK(SetPendingOperationType(kOperationCopy));
base::PlatformFileError result = SetUpFileSystemPath(
src_path_url, &src_path_, &src_util_, PATH_FOR_READ);
if (result == base::PLATFORM_FILE_OK)
result = SetUpFileSystemPath(
dest_path_url, &dest_path_, &dest_util_, PATH_FOR_CREATE);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result);
delete this;
return;
}
GetUsageAndQuotaThenRunTask(
dest_path_.origin(), dest_path_.type(),
base::Bind(&FileSystemOperation::DoCopy,
base::Unretained(this), callback),
base::Bind(callback, base::PLATFORM_FILE_ERROR_FAILED));
}
void FileSystemOperation::Move(const GURL& src_path_url,
const GURL& dest_path_url,
const StatusCallback& callback) {
DCHECK(SetPendingOperationType(kOperationMove));
base::PlatformFileError result = SetUpFileSystemPath(
src_path_url, &src_path_, &src_util_, PATH_FOR_WRITE);
if (result == base::PLATFORM_FILE_OK)
result = SetUpFileSystemPath(
dest_path_url, &dest_path_, &dest_util_, PATH_FOR_CREATE);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result);
delete this;
return;
}
GetUsageAndQuotaThenRunTask(
dest_path_.origin(), dest_path_.type(),
base::Bind(&FileSystemOperation::DoMove,
base::Unretained(this), callback),
base::Bind(callback, base::PLATFORM_FILE_ERROR_FAILED));
}
void FileSystemOperation::DirectoryExists(const GURL& path_url,
const StatusCallback& callback) {
DCHECK(SetPendingOperationType(kOperationDirectoryExists));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_READ);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result);
delete this;
return;
}
FileSystemFileUtilProxy::GetFileInfo(
proxy_, &operation_context_, src_util_, src_path_,
base::Bind(&FileSystemOperation::DidDirectoryExists,
base::Owned(this), callback));
}
void FileSystemOperation::FileExists(const GURL& path_url,
const StatusCallback& callback) {
DCHECK(SetPendingOperationType(kOperationFileExists));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_READ);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result);
delete this;
return;
}
FileSystemFileUtilProxy::GetFileInfo(
proxy_, &operation_context_, src_util_, src_path_,
base::Bind(&FileSystemOperation::DidFileExists,
base::Owned(this), callback));
}
void FileSystemOperation::GetMetadata(const GURL& path_url,
const GetMetadataCallback& callback) {
DCHECK(SetPendingOperationType(kOperationGetMetadata));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_READ);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result, base::PlatformFileInfo(), FilePath());
delete this;
return;
}
FileSystemFileUtilProxy::GetFileInfo(
proxy_, &operation_context_, src_util_, src_path_,
base::Bind(&FileSystemOperation::DidGetMetadata,
base::Owned(this), callback));
}
void FileSystemOperation::ReadDirectory(const GURL& path_url,
const ReadDirectoryCallback& callback) {
DCHECK(SetPendingOperationType(kOperationReadDirectory));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_READ);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result, std::vector<base::FileUtilProxy::Entry>(), false);
delete this;
return;
}
FileSystemFileUtilProxy::ReadDirectory(
proxy_, &operation_context_, src_util_, src_path_,
base::Bind(&FileSystemOperation::DidReadDirectory,
base::Owned(this), callback));
}
void FileSystemOperation::Remove(const GURL& path_url, bool recursive,
const StatusCallback& callback) {
DCHECK(SetPendingOperationType(kOperationRemove));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_WRITE);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result);
delete this;
return;
}
scoped_quota_notifier_.reset(new ScopedQuotaNotifier(
file_system_context(), src_path_.origin(), src_path_.type()));
FileSystemFileUtilProxy::Delete(
proxy_, &operation_context_, src_util_, src_path_, recursive,
base::Bind(&FileSystemOperation::DidFinishFileOperation,
base::Owned(this), callback));
}
void FileSystemOperation::Write(
const net::URLRequestContext* url_request_context,
const GURL& path_url,
const GURL& blob_url,
int64 offset,
const WriteCallback& callback) {
DCHECK(SetPendingOperationType(kOperationWrite));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_WRITE);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result, 0, false);
delete this;
return;
}
DCHECK(blob_url.is_valid());
file_writer_delegate_.reset(new FileWriterDelegate(
this, src_path_, offset, proxy_));
set_write_callback(callback);
blob_request_.reset(
new net::URLRequest(blob_url, file_writer_delegate_.get()));
blob_request_->set_context(url_request_context);
GetUsageAndQuotaThenRunTask(
src_path_.origin(), src_path_.type(),
base::Bind(&FileSystemOperation::DoWrite, base::Unretained(this)),
base::Bind(callback, base::PLATFORM_FILE_ERROR_FAILED, 0, true));
}
void FileSystemOperation::Truncate(const GURL& path_url, int64 length,
const StatusCallback& callback) {
DCHECK(SetPendingOperationType(kOperationTruncate));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_WRITE);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result);
delete this;
return;
}
GetUsageAndQuotaThenRunTask(
src_path_.origin(), src_path_.type(),
base::Bind(&FileSystemOperation::DoTruncate,
base::Unretained(this), callback, length),
base::Bind(callback, base::PLATFORM_FILE_ERROR_FAILED));
}
void FileSystemOperation::TouchFile(const GURL& path_url,
const base::Time& last_access_time,
const base::Time& last_modified_time,
const StatusCallback& callback) {
DCHECK(SetPendingOperationType(kOperationTouchFile));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_WRITE);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result);
delete this;
return;
}
FileSystemFileUtilProxy::Touch(
proxy_, &operation_context_, src_util_, src_path_,
last_access_time, last_modified_time,
base::Bind(&FileSystemOperation::DidTouchFile,
base::Owned(this), callback));
}
void FileSystemOperation::OpenFile(const GURL& path_url,
int file_flags,
base::ProcessHandle peer_handle,
const OpenFileCallback& callback) {
DCHECK(SetPendingOperationType(kOperationOpenFile));
scoped_ptr<FileSystemOperation> deleter(this);
peer_handle_ = peer_handle;
if (file_flags & (
(base::PLATFORM_FILE_ENUMERATE | base::PLATFORM_FILE_TEMPORARY |
base::PLATFORM_FILE_HIDDEN))) {
callback.Run(base::PLATFORM_FILE_ERROR_FAILED,
base::PlatformFile(), base::ProcessHandle());
return;
}
if (file_flags &
(base::PLATFORM_FILE_CREATE | base::PLATFORM_FILE_OPEN_ALWAYS |
base::PLATFORM_FILE_CREATE_ALWAYS | base::PLATFORM_FILE_OPEN_TRUNCATED |
base::PLATFORM_FILE_WRITE | base::PLATFORM_FILE_EXCLUSIVE_WRITE |
base::PLATFORM_FILE_DELETE_ON_CLOSE |
base::PLATFORM_FILE_WRITE_ATTRIBUTES)) {
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_CREATE);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result, base::PlatformFile(), base::ProcessHandle());
return;
}
} else {
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_READ);
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result, base::PlatformFile(), base::ProcessHandle());
return;
}
}
GetUsageAndQuotaThenRunTask(
src_path_.origin(), src_path_.type(),
base::Bind(&FileSystemOperation::DoOpenFile,
base::Unretained(deleter.release()), callback, file_flags),
base::Bind(callback, base::PLATFORM_FILE_ERROR_FAILED,
base::kInvalidPlatformFileValue,
base::kNullProcessHandle));
}
// We can only get here on a write or truncate that's not yet completed.
// We don't support cancelling any other operation at this time.
void FileSystemOperation::Cancel(const StatusCallback& cancel_callback) {
if (file_writer_delegate_.get()) {
DCHECK_EQ(kOperationWrite, pending_operation_);
// Writes are done without proxying through FileUtilProxy after the initial
// opening of the PlatformFile. All state changes are done on this thread,
// so we're guaranteed to be able to shut down atomically. We do need to
// check that the file has been opened [which means the blob_request_ has
// been created], so we know how much we need to do.
if (blob_request_.get())
// This halts any calls to file_writer_delegate_ from blob_request_.
blob_request_->Cancel();
if (!write_callback_.is_null()) {
// Notify the failure status to the ongoing operation's callback.
write_callback_.Run(base::PLATFORM_FILE_ERROR_ABORT, 0, false);
// Do not delete this FileSystemOperation object yet. As a result of
// abort, this->DidWrite is called later and there we delete this object.
}
cancel_callback.Run(base::PLATFORM_FILE_OK);
write_callback_.Reset();
} else {
DCHECK_EQ(kOperationTruncate, pending_operation_);
// We're cancelling a truncate operation, but we can't actually stop it
// since it's been proxied to another thread. We need to save the
// cancel_callback so that when the truncate returns, it can see that it's
// been cancelled, report it, and report that the cancel has succeeded.
DCHECK(cancel_callback_.is_null());
cancel_callback_ = cancel_callback;
}
}
FileSystemOperation* FileSystemOperation::AsFileSystemOperation() {
return this;
}
void FileSystemOperation::SyncGetPlatformPath(const GURL& path_url,
FilePath* platform_path) {
DCHECK(SetPendingOperationType(kOperationGetLocalPath));
base::PlatformFileError result = SetUpFileSystemPath(
path_url, &src_path_, &src_util_, PATH_FOR_READ);
if (result != base::PLATFORM_FILE_OK) {
delete this;
return;
}
src_util_->GetLocalFilePath(
&operation_context_, src_path_, platform_path);
delete this;
}
void FileSystemOperation::CreateSnapshotFile(
const GURL& path_url,
const SnapshotFileCallback& callback) {
GetMetadata(path_url, base::Bind(&GetMetadataForSnapshot, callback));
}
FileSystemOperation::FileSystemOperation(
scoped_refptr<base::MessageLoopProxy> proxy,
FileSystemContext* file_system_context)
: proxy_(proxy),
operation_context_(file_system_context),
src_util_(NULL),
dest_util_(NULL),
peer_handle_(base::kNullProcessHandle),
pending_operation_(kOperationNone) {
}
void FileSystemOperation::GetUsageAndQuotaThenRunTask(
const GURL& origin, FileSystemType type,
const base::Closure& task,
const base::Closure& error_callback) {
quota::QuotaManagerProxy* quota_manager_proxy =
file_system_context()->quota_manager_proxy();
if (!quota_manager_proxy ||
!file_system_context()->GetQuotaUtil(type)) {
// If we don't have the quota manager or the requested filesystem type
// does not support quota, we should be able to let it go.
operation_context_.set_allowed_bytes_growth(kint64max);
task.Run();
return;
}
TaskParamsForDidGetQuota params;
params.origin = origin;
params.type = type;
params.task = task;
params.error_callback = error_callback;
DCHECK(quota_manager_proxy);
DCHECK(quota_manager_proxy->quota_manager());
quota_manager_proxy->quota_manager()->GetUsageAndQuota(
origin,
FileSystemTypeToQuotaStorageType(type),
base::Bind(&FileSystemOperation::DidGetUsageAndQuotaAndRunTask,
base::Unretained(this), params));
}
void FileSystemOperation::DidGetUsageAndQuotaAndRunTask(
const TaskParamsForDidGetQuota& params,
quota::QuotaStatusCode status,
int64 usage, int64 quota) {
if (status != quota::kQuotaStatusOk) {
LOG(WARNING) << "Got unexpected quota error : " << status;
params.error_callback.Run();
return;
}
operation_context_.set_allowed_bytes_growth(quota - usage);
scoped_quota_notifier_.reset(new ScopedQuotaNotifier(
file_system_context(), params.origin, params.type));
params.task.Run();
}
void FileSystemOperation::DoCreateFile(
const StatusCallback& callback,
bool exclusive) {
FileSystemFileUtilProxy::EnsureFileExists(
proxy_, &operation_context_, src_util_, src_path_,
base::Bind(
exclusive ? &FileSystemOperation::DidEnsureFileExistsExclusive
: &FileSystemOperation::DidEnsureFileExistsNonExclusive,
base::Owned(this), callback));
}
void FileSystemOperation::DoCreateDirectory(
const StatusCallback& callback,
bool exclusive, bool recursive) {
FileSystemFileUtilProxy::CreateDirectory(
proxy_, &operation_context_, src_util_, src_path_, exclusive, recursive,
base::Bind(&FileSystemOperation::DidFinishFileOperation,
base::Owned(this), callback));
}
void FileSystemOperation::DoCopy(const StatusCallback& callback) {
FileSystemFileUtilProxy::Copy(
proxy_, &operation_context_,
src_util_, dest_util_,
src_path_, dest_path_,
base::Bind(&FileSystemOperation::DidFinishFileOperation,
base::Owned(this), callback));
}
void FileSystemOperation::DoMove(const StatusCallback& callback) {
FileSystemFileUtilProxy::Move(
proxy_, &operation_context_,
src_util_, dest_util_,
src_path_, dest_path_,
base::Bind(&FileSystemOperation::DidFinishFileOperation,
base::Owned(this), callback));
}
void FileSystemOperation::DoWrite() {
int file_flags = base::PLATFORM_FILE_OPEN |
base::PLATFORM_FILE_WRITE |
base::PLATFORM_FILE_ASYNC;
FileSystemFileUtilProxy::CreateOrOpen(
proxy_, &operation_context_, src_util_, src_path_, file_flags,
base::Bind(&FileSystemOperation::OnFileOpenedForWrite,
base::Unretained(this)));
}
void FileSystemOperation::DoTruncate(const StatusCallback& callback,
int64 length) {
FileSystemFileUtilProxy::Truncate(
proxy_, &operation_context_, src_util_, src_path_, length,
base::Bind(&FileSystemOperation::DidFinishFileOperation,
base::Owned(this), callback));
}
void FileSystemOperation::DoOpenFile(const OpenFileCallback& callback,
int file_flags) {
FileSystemFileUtilProxy::CreateOrOpen(
proxy_, &operation_context_, src_util_, src_path_, file_flags,
base::Bind(&FileSystemOperation::DidOpenFile,
base::Owned(this), callback));
}
void FileSystemOperation::DidEnsureFileExistsExclusive(
const StatusCallback& callback,
base::PlatformFileError rv, bool created) {
if (rv == base::PLATFORM_FILE_OK && !created) {
callback.Run(base::PLATFORM_FILE_ERROR_EXISTS);
} else {
DidFinishFileOperation(callback, rv);
}
}
void FileSystemOperation::DidEnsureFileExistsNonExclusive(
const StatusCallback& callback,
base::PlatformFileError rv, bool /* created */) {
DidFinishFileOperation(callback, rv);
}
void FileSystemOperation::DidFinishFileOperation(
const StatusCallback& callback,
base::PlatformFileError rv) {
if (!cancel_callback_.is_null()) {
DCHECK_EQ(kOperationTruncate, pending_operation_);
callback.Run(base::PLATFORM_FILE_ERROR_ABORT);
cancel_callback_.Run(base::PLATFORM_FILE_OK);
cancel_callback_.Reset();
} else {
callback.Run(rv);
}
}
void FileSystemOperation::DidDirectoryExists(
const StatusCallback& callback,
base::PlatformFileError rv,
const base::PlatformFileInfo& file_info,
const FilePath& unused) {
if (rv == base::PLATFORM_FILE_OK && !file_info.is_directory)
rv = base::PLATFORM_FILE_ERROR_NOT_A_DIRECTORY;
callback.Run(rv);
}
void FileSystemOperation::DidFileExists(
const StatusCallback& callback,
base::PlatformFileError rv,
const base::PlatformFileInfo& file_info,
const FilePath& unused) {
if (rv == base::PLATFORM_FILE_OK && file_info.is_directory)
rv = base::PLATFORM_FILE_ERROR_NOT_A_FILE;
callback.Run(rv);
}
void FileSystemOperation::DidGetMetadata(
const GetMetadataCallback& callback,
base::PlatformFileError rv,
const base::PlatformFileInfo& file_info,
const FilePath& platform_path) {
callback.Run(rv, file_info, platform_path);
}
void FileSystemOperation::DidReadDirectory(
const ReadDirectoryCallback& callback,
base::PlatformFileError rv,
const std::vector<base::FileUtilProxy::Entry>& entries,
bool has_more) {
callback.Run(rv, entries, has_more);
}
void FileSystemOperation::DidWrite(
base::PlatformFileError rv,
int64 bytes,
bool complete) {
if (write_callback_.is_null()) {
// If cancelled, callback is already invoked and set to null in Cancel().
// We must not call it twice. Just shut down this operation object.
delete this;
return;
}
write_callback_.Run(rv, bytes, complete);
if (complete || rv != base::PLATFORM_FILE_OK)
delete this;
}
void FileSystemOperation::DidTouchFile(const StatusCallback& callback,
base::PlatformFileError rv) {
callback.Run(rv);
}
void FileSystemOperation::DidOpenFile(
const OpenFileCallback& callback,
base::PlatformFileError rv,
base::PassPlatformFile file,
bool unused) {
if (rv == base::PLATFORM_FILE_OK)
CHECK_NE(base::kNullProcessHandle, peer_handle_);
callback.Run(rv, file.ReleaseValue(), peer_handle_);
}
void FileSystemOperation::OnFileOpenedForWrite(
base::PlatformFileError rv,
base::PassPlatformFile file,
bool created) {
if (rv != base::PLATFORM_FILE_OK) {
if (!write_callback_.is_null())
write_callback_.Run(rv, 0, false);
delete this;
return;
}
file_writer_delegate_->Start(file.ReleaseValue(), blob_request_.get());
}
base::PlatformFileError FileSystemOperation::SetUpFileSystemPath(
const GURL& path_url,
FileSystemPath* file_system_path,
FileSystemFileUtil** file_util,
SetUpPathMode mode) {
DCHECK(file_system_path);
GURL origin_url;
FileSystemType type;
FilePath cracked_path;
if (!CrackFileSystemURL(path_url, &origin_url, &type, &cracked_path))
return base::PLATFORM_FILE_ERROR_INVALID_URL;
if (!file_system_context()->GetMountPointProvider(type)->IsAccessAllowed(
origin_url, type, cracked_path))
return base::PLATFORM_FILE_ERROR_SECURITY;
DCHECK(file_util);
if (!*file_util)
*file_util = file_system_context()->GetFileUtil(type);
if (!*file_util)
return base::PLATFORM_FILE_ERROR_SECURITY;
file_system_path->set_origin(origin_url);
file_system_path->set_type(type);
file_system_path->set_internal_path(cracked_path);
if (mode == PATH_FOR_READ) {
// We notify this read access whether the read access succeeds or not.
// This must be ok since this is used to let the QM's eviction logic know
// someone is interested in reading the origin data and therefore to
// indicate that evicting this origin may not be a good idea.
FileSystemQuotaUtil* quota_util = file_system_context()->GetQuotaUtil(type);
if (quota_util) {
quota_util->NotifyOriginWasAccessedOnIOThread(
file_system_context()->quota_manager_proxy(),
file_system_path->origin(), file_system_path->type());
}
return base::PLATFORM_FILE_OK;
}
DCHECK(mode == PATH_FOR_WRITE || mode == PATH_FOR_CREATE);
// Any write access is disallowed on the root path.
if (cracked_path.value().length() == 0 ||
cracked_path.DirName().value() == cracked_path.value())
return base::PLATFORM_FILE_ERROR_SECURITY;
if (mode == PATH_FOR_CREATE) {
FileSystemMountPointProvider* provider = file_system_context()->
GetMountPointProvider(type);
// Check if the cracked file name looks good to create.
if (provider->IsRestrictedFileName(VirtualPath::BaseName(cracked_path)))
return base::PLATFORM_FILE_ERROR_SECURITY;
}
return base::PLATFORM_FILE_OK;
}
bool FileSystemOperation::SetPendingOperationType(OperationType type) {
if (pending_operation_ != kOperationNone)
return false;
pending_operation_ = type;
return true;
}
} // namespace fileapi