// 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 "content/child/fileapi/file_system_dispatcher.h"
#include "base/callback.h"
#include "base/file_util.h"
#include "base/message_loop/message_loop_proxy.h"
#include "base/process/process.h"
#include "content/child/child_thread.h"
#include "content/common/fileapi/file_system_messages.h"
#include "webkit/common/fileapi/file_system_info.h"
namespace content {
class FileSystemDispatcher::CallbackDispatcher {
public:
typedef CallbackDispatcher self;
typedef FileSystemDispatcher::StatusCallback StatusCallback;
typedef FileSystemDispatcher::MetadataCallback MetadataCallback;
typedef FileSystemDispatcher::ReadDirectoryCallback ReadDirectoryCallback;
typedef FileSystemDispatcher::OpenFileSystemCallback OpenFileSystemCallback;
typedef FileSystemDispatcher::ResolveURLCallback ResolveURLCallback;
typedef FileSystemDispatcher::WriteCallback WriteCallback;
typedef FileSystemDispatcher::OpenFileCallback OpenFileCallback;
static CallbackDispatcher* Create(const StatusCallback& callback) {
CallbackDispatcher* dispatcher = new CallbackDispatcher;
dispatcher->status_callback_ = callback;
dispatcher->error_callback_ = callback;
return dispatcher;
}
static CallbackDispatcher* Create(const MetadataCallback& callback,
const StatusCallback& error_callback) {
CallbackDispatcher* dispatcher = new CallbackDispatcher;
dispatcher->metadata_callback_ = callback;
dispatcher->error_callback_ = error_callback;
return dispatcher;
}
static CallbackDispatcher* Create(const CreateSnapshotFileCallback& callback,
const StatusCallback& error_callback) {
CallbackDispatcher* dispatcher = new CallbackDispatcher;
dispatcher->snapshot_callback_ = callback;
dispatcher->error_callback_ = error_callback;
return dispatcher;
}
static CallbackDispatcher* Create(const ReadDirectoryCallback& callback,
const StatusCallback& error_callback) {
CallbackDispatcher* dispatcher = new CallbackDispatcher;
dispatcher->directory_callback_ = callback;
dispatcher->error_callback_ = error_callback;
return dispatcher;
}
static CallbackDispatcher* Create(const OpenFileSystemCallback& callback,
const StatusCallback& error_callback) {
CallbackDispatcher* dispatcher = new CallbackDispatcher;
dispatcher->filesystem_callback_ = callback;
dispatcher->error_callback_ = error_callback;
return dispatcher;
}
static CallbackDispatcher* Create(const ResolveURLCallback& callback,
const StatusCallback& error_callback) {
CallbackDispatcher* dispatcher = new CallbackDispatcher;
dispatcher->resolve_callback_ = callback;
dispatcher->error_callback_ = error_callback;
return dispatcher;
}
static CallbackDispatcher* Create(const WriteCallback& callback,
const StatusCallback& error_callback) {
CallbackDispatcher* dispatcher = new CallbackDispatcher;
dispatcher->write_callback_ = callback;
dispatcher->error_callback_ = error_callback;
return dispatcher;
}
static CallbackDispatcher* Create(const OpenFileCallback& callback,
const StatusCallback& error_callback) {
CallbackDispatcher* dispatcher = new CallbackDispatcher;
dispatcher->open_callback_ = callback;
dispatcher->error_callback_ = error_callback;
return dispatcher;
}
~CallbackDispatcher() {}
void DidSucceed() {
status_callback_.Run(base::PLATFORM_FILE_OK);
}
void DidFail(base::PlatformFileError error_code) {
error_callback_.Run(error_code);
}
void DidReadMetadata(
const base::PlatformFileInfo& file_info) {
metadata_callback_.Run(file_info);
}
void DidCreateSnapshotFile(
const base::PlatformFileInfo& file_info,
const base::FilePath& platform_path,
int request_id) {
snapshot_callback_.Run(file_info, platform_path, request_id);
}
void DidReadDirectory(
const std::vector<fileapi::DirectoryEntry>& entries,
bool has_more) {
directory_callback_.Run(entries, has_more);
}
void DidOpenFileSystem(const std::string& name,
const GURL& root) {
filesystem_callback_.Run(name, root);
}
void DidResolveURL(const fileapi::FileSystemInfo& info,
const base::FilePath& file_path,
bool is_directory) {
resolve_callback_.Run(info, file_path, is_directory);
}
void DidWrite(int64 bytes, bool complete) {
write_callback_.Run(bytes, complete);
}
void DidOpenFile(base::PlatformFile file,
int file_open_id,
quota::QuotaLimitType quota_policy) {
open_callback_.Run(file, file_open_id, quota_policy);
}
private:
CallbackDispatcher() {}
StatusCallback status_callback_;
MetadataCallback metadata_callback_;
CreateSnapshotFileCallback snapshot_callback_;
ReadDirectoryCallback directory_callback_;
OpenFileSystemCallback filesystem_callback_;
ResolveURLCallback resolve_callback_;
WriteCallback write_callback_;
OpenFileCallback open_callback_;
StatusCallback error_callback_;
DISALLOW_COPY_AND_ASSIGN(CallbackDispatcher);
};
FileSystemDispatcher::FileSystemDispatcher() {
}
FileSystemDispatcher::~FileSystemDispatcher() {
// Make sure we fire all the remaining callbacks.
for (IDMap<CallbackDispatcher, IDMapOwnPointer>::iterator
iter(&dispatchers_); !iter.IsAtEnd(); iter.Advance()) {
int request_id = iter.GetCurrentKey();
CallbackDispatcher* dispatcher = iter.GetCurrentValue();
DCHECK(dispatcher);
dispatcher->DidFail(base::PLATFORM_FILE_ERROR_ABORT);
dispatchers_.Remove(request_id);
}
}
bool FileSystemDispatcher::OnMessageReceived(const IPC::Message& msg) {
bool handled = true;
IPC_BEGIN_MESSAGE_MAP(FileSystemDispatcher, msg)
IPC_MESSAGE_HANDLER(FileSystemMsg_DidOpenFileSystem, OnDidOpenFileSystem)
IPC_MESSAGE_HANDLER(FileSystemMsg_DidResolveURL, OnDidResolveURL)
IPC_MESSAGE_HANDLER(FileSystemMsg_DidSucceed, OnDidSucceed)
IPC_MESSAGE_HANDLER(FileSystemMsg_DidReadDirectory, OnDidReadDirectory)
IPC_MESSAGE_HANDLER(FileSystemMsg_DidReadMetadata, OnDidReadMetadata)
IPC_MESSAGE_HANDLER(FileSystemMsg_DidCreateSnapshotFile,
OnDidCreateSnapshotFile)
IPC_MESSAGE_HANDLER(FileSystemMsg_DidFail, OnDidFail)
IPC_MESSAGE_HANDLER(FileSystemMsg_DidWrite, OnDidWrite)
IPC_MESSAGE_HANDLER(FileSystemMsg_DidOpenFile, OnDidOpenFile)
IPC_MESSAGE_UNHANDLED(handled = false)
IPC_END_MESSAGE_MAP()
return handled;
}
void FileSystemDispatcher::OpenFileSystem(
const GURL& origin_url,
fileapi::FileSystemType type,
const OpenFileSystemCallback& success_callback,
const StatusCallback& error_callback) {
int request_id = dispatchers_.Add(
CallbackDispatcher::Create(success_callback, error_callback));
ChildThread::current()->Send(new FileSystemHostMsg_OpenFileSystem(
request_id, origin_url, type));
}
void FileSystemDispatcher::ResolveURL(
const GURL& filesystem_url,
const ResolveURLCallback& success_callback,
const StatusCallback& error_callback) {
int request_id = dispatchers_.Add(
CallbackDispatcher::Create(success_callback, error_callback));
ChildThread::current()->Send(new FileSystemHostMsg_ResolveURL(
request_id, filesystem_url));
}
void FileSystemDispatcher::DeleteFileSystem(
const GURL& origin_url,
fileapi::FileSystemType type,
const StatusCallback& callback) {
int request_id = dispatchers_.Add(CallbackDispatcher::Create(callback));
ChildThread::current()->Send(new FileSystemHostMsg_DeleteFileSystem(
request_id, origin_url, type));
}
void FileSystemDispatcher::Move(
const GURL& src_path,
const GURL& dest_path,
const StatusCallback& callback) {
int request_id = dispatchers_.Add(CallbackDispatcher::Create(callback));
ChildThread::current()->Send(new FileSystemHostMsg_Move(
request_id, src_path, dest_path));
}
void FileSystemDispatcher::Copy(
const GURL& src_path,
const GURL& dest_path,
const StatusCallback& callback) {
int request_id = dispatchers_.Add(CallbackDispatcher::Create(callback));
ChildThread::current()->Send(new FileSystemHostMsg_Copy(
request_id, src_path, dest_path));
}
void FileSystemDispatcher::Remove(
const GURL& path,
bool recursive,
const StatusCallback& callback) {
int request_id = dispatchers_.Add(CallbackDispatcher::Create(callback));
ChildThread::current()->Send(
new FileSystemHostMsg_Remove(request_id, path, recursive));
}
void FileSystemDispatcher::ReadMetadata(
const GURL& path,
const MetadataCallback& success_callback,
const StatusCallback& error_callback) {
int request_id = dispatchers_.Add(
CallbackDispatcher::Create(success_callback, error_callback));
ChildThread::current()->Send(
new FileSystemHostMsg_ReadMetadata(request_id, path));
}
void FileSystemDispatcher::CreateFile(
const GURL& path,
bool exclusive,
const StatusCallback& callback) {
int request_id = dispatchers_.Add(CallbackDispatcher::Create(callback));
ChildThread::current()->Send(new FileSystemHostMsg_Create(
request_id, path, exclusive,
false /* is_directory */, false /* recursive */));
}
void FileSystemDispatcher::CreateDirectory(
const GURL& path,
bool exclusive,
bool recursive,
const StatusCallback& callback) {
int request_id = dispatchers_.Add(CallbackDispatcher::Create(callback));
ChildThread::current()->Send(new FileSystemHostMsg_Create(
request_id, path, exclusive, true /* is_directory */, recursive));
}
void FileSystemDispatcher::Exists(
const GURL& path,
bool is_directory,
const StatusCallback& callback) {
int request_id = dispatchers_.Add(CallbackDispatcher::Create(callback));
ChildThread::current()->Send(
new FileSystemHostMsg_Exists(request_id, path, is_directory));
}
void FileSystemDispatcher::ReadDirectory(
const GURL& path,
const ReadDirectoryCallback& success_callback,
const StatusCallback& error_callback) {
int request_id = dispatchers_.Add(
CallbackDispatcher::Create(success_callback, error_callback));
ChildThread::current()->Send(
new FileSystemHostMsg_ReadDirectory(request_id, path));
}
void FileSystemDispatcher::Truncate(
const GURL& path,
int64 offset,
int* request_id_out,
const StatusCallback& callback) {
int request_id = dispatchers_.Add(CallbackDispatcher::Create(callback));
ChildThread::current()->Send(
new FileSystemHostMsg_Truncate(request_id, path, offset));
if (request_id_out)
*request_id_out = request_id;
}
void FileSystemDispatcher::WriteDeprecated(
const GURL& path,
const GURL& blob_url,
int64 offset,
int* request_id_out,
const WriteCallback& success_callback,
const StatusCallback& error_callback) {
int request_id = dispatchers_.Add(
CallbackDispatcher::Create(success_callback, error_callback));
ChildThread::current()->Send(
new FileSystemHostMsg_WriteDeprecated(request_id, path,
blob_url, offset));
if (request_id_out)
*request_id_out = request_id;
}
void FileSystemDispatcher::Write(
const GURL& path,
const std::string& blob_id,
int64 offset,
int* request_id_out,
const WriteCallback& success_callback,
const StatusCallback& error_callback) {
int request_id = dispatchers_.Add(
CallbackDispatcher::Create(success_callback, error_callback));
ChildThread::current()->Send(
new FileSystemHostMsg_Write(request_id, path, blob_id, offset));
if (request_id_out)
*request_id_out = request_id;
}
void FileSystemDispatcher::Cancel(
int request_id_to_cancel,
const StatusCallback& callback) {
int request_id = dispatchers_.Add(CallbackDispatcher::Create(callback));
ChildThread::current()->Send(new FileSystemHostMsg_CancelWrite(
request_id, request_id_to_cancel));
}
void FileSystemDispatcher::TouchFile(
const GURL& path,
const base::Time& last_access_time,
const base::Time& last_modified_time,
const StatusCallback& callback) {
int request_id = dispatchers_.Add(CallbackDispatcher::Create(callback));
ChildThread::current()->Send(
new FileSystemHostMsg_TouchFile(
request_id, path, last_access_time, last_modified_time));
}
void FileSystemDispatcher::CreateSnapshotFile(
const GURL& file_path,
const CreateSnapshotFileCallback& success_callback,
const StatusCallback& error_callback) {
int request_id = dispatchers_.Add(
CallbackDispatcher::Create(success_callback, error_callback));
ChildThread::current()->Send(
new FileSystemHostMsg_CreateSnapshotFile(
request_id, file_path));
}
void FileSystemDispatcher::OnDidOpenFileSystem(int request_id,
const std::string& name,
const GURL& root) {
DCHECK(root.is_valid());
CallbackDispatcher* dispatcher = dispatchers_.Lookup(request_id);
DCHECK(dispatcher);
dispatcher->DidOpenFileSystem(name, root);
dispatchers_.Remove(request_id);
}
void FileSystemDispatcher::OnDidResolveURL(int request_id,
const fileapi::FileSystemInfo& info,
const base::FilePath& file_path,
bool is_directory) {
DCHECK(info.root_url.is_valid());
CallbackDispatcher* dispatcher = dispatchers_.Lookup(request_id);
DCHECK(dispatcher);
dispatcher->DidResolveURL(info, file_path, is_directory);
dispatchers_.Remove(request_id);
}
void FileSystemDispatcher::OnDidSucceed(int request_id) {
CallbackDispatcher* dispatcher = dispatchers_.Lookup(request_id);
DCHECK(dispatcher);
dispatcher->DidSucceed();
dispatchers_.Remove(request_id);
}
void FileSystemDispatcher::OnDidReadMetadata(
int request_id, const base::PlatformFileInfo& file_info) {
CallbackDispatcher* dispatcher = dispatchers_.Lookup(request_id);
DCHECK(dispatcher);
dispatcher->DidReadMetadata(file_info);
dispatchers_.Remove(request_id);
}
void FileSystemDispatcher::OnDidCreateSnapshotFile(
int request_id, const base::PlatformFileInfo& file_info,
const base::FilePath& platform_path) {
CallbackDispatcher* dispatcher = dispatchers_.Lookup(request_id);
DCHECK(dispatcher);
dispatcher->DidCreateSnapshotFile(file_info, platform_path, request_id);
dispatchers_.Remove(request_id);
}
void FileSystemDispatcher::OnDidReadDirectory(
int request_id,
const std::vector<fileapi::DirectoryEntry>& entries,
bool has_more) {
CallbackDispatcher* dispatcher = dispatchers_.Lookup(request_id);
DCHECK(dispatcher);
dispatcher->DidReadDirectory(entries, has_more);
dispatchers_.Remove(request_id);
}
void FileSystemDispatcher::OnDidFail(
int request_id, base::PlatformFileError error_code) {
CallbackDispatcher* dispatcher = dispatchers_.Lookup(request_id);
DCHECK(dispatcher);
dispatcher->DidFail(error_code);
dispatchers_.Remove(request_id);
}
void FileSystemDispatcher::OnDidWrite(
int request_id, int64 bytes, bool complete) {
CallbackDispatcher* dispatcher = dispatchers_.Lookup(request_id);
DCHECK(dispatcher);
dispatcher->DidWrite(bytes, complete);
if (complete)
dispatchers_.Remove(request_id);
}
void FileSystemDispatcher::OnDidOpenFile(
int request_id,
IPC::PlatformFileForTransit file,
int file_open_id,
quota::QuotaLimitType quota_policy) {
CallbackDispatcher* dispatcher = dispatchers_.Lookup(request_id);
DCHECK(dispatcher);
dispatcher->DidOpenFile(IPC::PlatformFileForTransitToPlatformFile(file),
file_open_id,
quota_policy);
dispatchers_.Remove(request_id);
}
} // namespace content