blob: dcedf5a2982695afef2f719f0b11b85cd8f3d916 [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 "base/bind.h"
#include "base/memory/weak_ptr.h"
#include "base/message_loop.h"
#include "webkit/chromeos/fileapi/memory_file_util.h"
namespace {
const int kDefaultReadDirectoryBufferSize = 100;
} // namespace
namespace fileapi {
// In-memory implementation of AsyncFileStream.
class MemoryFileUtilAsyncFileStream : public AsyncFileStream {
public:
// |file_entry| is owned by MemoryFileUtil.
MemoryFileUtilAsyncFileStream(MemoryFileUtil::FileEntry* file_entry,
int flags)
: file_entry_(file_entry),
flags_(flags),
offset_(0),
weak_ptr_factory_(ALLOW_THIS_IN_INITIALIZER_LIST(this)) {
}
// AsyncFileStream override.
virtual void Read(char* buffer,
int64 length,
const ReadWriteCallback& callback) OVERRIDE {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtilAsyncFileStream::DoRead,
weak_ptr_factory_.GetWeakPtr(),
buffer, length, callback));
}
// AsyncFileStream override.
virtual void Write(const char* buffer,
int64 length,
const ReadWriteCallback& callback) OVERRIDE {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtilAsyncFileStream::DoWrite,
weak_ptr_factory_.GetWeakPtr(),
buffer, length, callback));
}
// AsyncFileStream override.
virtual void Seek(int64 offset,
const SeekCallback& callback) OVERRIDE {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtilAsyncFileStream::DoSeek,
weak_ptr_factory_.GetWeakPtr(),
offset, callback));
}
private:
// Callback used for Read().
void DoRead(char* buffer,
int64 length,
const ReadWriteCallback& callback) {
if ((flags_ & base::PLATFORM_FILE_READ) == 0) {
callback.Run(base::PLATFORM_FILE_ERROR_INVALID_OPERATION, 0);
return;
}
// Shorten the length so the read does not overrun.
length = std::min(length, file_size() - offset_);
const std::string& contents = file_entry_->contents;
std::copy(contents.begin() + offset_,
contents.begin() + offset_ + length,
buffer);
offset_ += length;
callback.Run(base::PLATFORM_FILE_OK, length);
}
// Callback used for Write().
void DoWrite(const char* buffer,
int64 length,
const ReadWriteCallback& callback) {
if ((flags_ & base::PLATFORM_FILE_WRITE) == 0) {
callback.Run(base::PLATFORM_FILE_ERROR_INVALID_OPERATION, 0);
return;
}
// Extend the contents if needed.
std::string* contents = &file_entry_->contents;
if (offset_ + length > file_size())
contents->resize(offset_ + length, 0); // Fill with 0.
std::copy(buffer, buffer + length,
contents->begin() + offset_);
file_entry_->last_modified = base::Time::Now();
offset_ += length;
callback.Run(base::PLATFORM_FILE_OK, length);
}
// Callback used for Seek().
void DoSeek(int64 offset,
const SeekCallback& callback) {
if (offset > file_size()) {
// Unlike lseek(2), we don't allow an offset larger than the file
// size for this file implementation.
callback.Run(base::PLATFORM_FILE_ERROR_INVALID_OPERATION);
return;
}
offset_ = offset;
callback.Run(base::PLATFORM_FILE_OK);
}
// Returns the file size as int64.
int64 file_size() const {
return static_cast<int64>(file_entry_->contents.size());
}
MemoryFileUtil::FileEntry* file_entry_;
const int flags_;
int64 offset_;
base::WeakPtrFactory<MemoryFileUtilAsyncFileStream> weak_ptr_factory_;
};
MemoryFileUtil::FileEntry::FileEntry()
: is_directory(false) {
}
MemoryFileUtil::FileEntry::~FileEntry() {
}
MemoryFileUtil::MemoryFileUtil(const FilePath& root_path)
: read_directory_buffer_size_(kDefaultReadDirectoryBufferSize) {
FileEntry root;
root.is_directory = true;
root.last_modified = base::Time::Now();
files_[root_path] = root;
}
MemoryFileUtil::~MemoryFileUtil() {
}
// Depending on the flags value the flow of file opening will be one of
// the following:
//
// PLATFORM_FILE_OPEN:
// - GetFileInfo
// - DidGetFileInfoForOpen
// - OpenVerifiedFile
//
// PLATFORM_FILE_CREATE:
// - Create
// - DidCreateOrTruncateForOpen
// - OpenVerifiedFile
//
// PLATFORM_FILE_OPEN_ALWAYS:
// - GetFileInfo
// - DidGetFileInfoForOpen
// - OpenVerifiedFile OR Create
// DidCreateOrTruncateForOpen
// OpenVerifiedFile
//
// PLATFORM_FILE_CREATE_ALWAYS:
// - Truncate
// - OpenTruncatedFileOrCreate
// - OpenVerifiedFile OR Create
// DidCreateOrTruncateForOpen
// OpenVerifiedFile
//
// PLATFORM_FILE_OPEN_TRUNCATED:
// - Truncate
// - DidCreateOrTruncateForOpen
// - OpenVerifiedFile
//
void MemoryFileUtil::Open(
const FilePath& file_path,
int flags,
const OpenCallback& callback) {
int create_flag = flags & (base::PLATFORM_FILE_OPEN |
base::PLATFORM_FILE_CREATE |
base::PLATFORM_FILE_OPEN_ALWAYS |
base::PLATFORM_FILE_CREATE_ALWAYS |
base::PLATFORM_FILE_OPEN_TRUNCATED);
switch (create_flag) {
case base::PLATFORM_FILE_OPEN:
case base::PLATFORM_FILE_OPEN_ALWAYS:
GetFileInfo(
file_path,
base::Bind(&MemoryFileUtil::DidGetFileInfoForOpen,
base::Unretained(this), file_path, flags, callback));
break;
case base::PLATFORM_FILE_CREATE:
Create(
file_path,
base::Bind(&MemoryFileUtil::DidCreateOrTruncateForOpen,
base::Unretained(this), file_path, flags, 0, callback));
break;
case base::PLATFORM_FILE_CREATE_ALWAYS:
Truncate(
file_path,
0,
base::Bind(&MemoryFileUtil::OpenTruncatedFileOrCreate,
base::Unretained(this), file_path, flags, callback));
break;
case base::PLATFORM_FILE_OPEN_TRUNCATED:
Truncate(
file_path,
0,
base::Bind(&MemoryFileUtil::DidCreateOrTruncateForOpen,
base::Unretained(this), file_path, flags, 0, callback));
break;
default:
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(callback,
base::PLATFORM_FILE_ERROR_INVALID_OPERATION,
static_cast<AsyncFileStream*>(NULL)));
}
}
void MemoryFileUtil::GetFileInfo(
const FilePath& file_path,
const GetFileInfoCallback& callback) {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtil::DoGetFileInfo, base::Unretained(this),
file_path.StripTrailingSeparators(), callback));
}
void MemoryFileUtil::Create(
const FilePath& file_path,
const StatusCallback& callback) {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtil::DoCreate, base::Unretained(this),
file_path.StripTrailingSeparators(), false, callback));
}
void MemoryFileUtil::Truncate(
const FilePath& file_path,
int64 length,
const StatusCallback& callback) {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtil::DoTruncate, base::Unretained(this),
file_path.StripTrailingSeparators(), length, callback));
}
void MemoryFileUtil::Touch(
const FilePath& file_path,
const base::Time& last_access_time,
const base::Time& last_modified_time,
const StatusCallback& callback) {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtil::DoTouch, base::Unretained(this),
file_path.StripTrailingSeparators(),
last_modified_time, callback));
}
void MemoryFileUtil::Remove(
const FilePath& file_path,
bool recursive,
const StatusCallback& callback) {
if (recursive) {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtil::DoRemoveRecursive,
base::Unretained(this), file_path.StripTrailingSeparators(),
callback));
} else {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtil::DoRemoveSingleFile,
base::Unretained(this), file_path.StripTrailingSeparators(),
callback));
}
}
void MemoryFileUtil::CreateDirectory(
const FilePath& dir_path,
const StatusCallback& callback) {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtil::DoCreate,
base::Unretained(this), dir_path.StripTrailingSeparators(),
true, callback));
}
void MemoryFileUtil::ReadDirectory(
const FilePath& dir_path,
const ReadDirectoryCallback& callback) {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtil::DoReadDirectory,
base::Unretained(this), dir_path.StripTrailingSeparators(),
FilePath(), callback));
}
void MemoryFileUtil::DoGetFileInfo(const FilePath& file_path,
const GetFileInfoCallback& callback) {
base::PlatformFileInfo file_info;
FileIterator file_it = files_.find(file_path);
if (file_it == files_.end()) {
callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND, file_info);
return;
}
const FileEntry& file_entry = file_it->second;
file_info.size = file_entry.contents.size();
file_info.is_directory = file_entry.is_directory;
file_info.is_symbolic_link = false;
// In this file system implementation we store only one datetime. Many
// popular file systems do the same.
file_info.last_modified = file_entry.last_modified;
file_info.last_accessed = file_entry.last_modified;
file_info.creation_time = file_entry.last_modified;
callback.Run(base::PLATFORM_FILE_OK, file_info);
}
void MemoryFileUtil::DoCreate(
const FilePath& file_path,
bool is_directory,
const StatusCallback& callback) {
if (FileExists(file_path)) {
callback.Run(base::PLATFORM_FILE_ERROR_EXISTS);
return;
}
if (!IsDirectory(file_path.DirName())) {
callback.Run(base::PLATFORM_FILE_ERROR_FAILED);
return;
}
FileEntry file;
file.is_directory = is_directory;
file.last_modified = base::Time::Now();
files_[file_path] = file;
callback.Run(base::PLATFORM_FILE_OK);
}
void MemoryFileUtil::DoTruncate(
const FilePath& file_path,
int64 length,
const StatusCallback& callback) {
FileIterator file_it = files_.find(file_path);
if (file_it == files_.end()) {
callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND);
return;
}
FileEntry& file = file_it->second;
// Fill the extended part with 0 if |length| is larger than the original
// contents size.
file.contents.resize(length, 0);
callback.Run(base::PLATFORM_FILE_OK);
}
void MemoryFileUtil::DoTouch(
const FilePath& file_path,
const base::Time& last_modified_time,
const StatusCallback& callback) {
FileIterator file_it = files_.find(file_path);
if (file_it == files_.end()) {
callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND);
return;
}
FileEntry& file = file_it->second;
file.last_modified = last_modified_time;
callback.Run(base::PLATFORM_FILE_OK);
}
void MemoryFileUtil::DoRemoveSingleFile(
const FilePath& file_path,
const StatusCallback& callback) {
FileIterator file_it = files_.find(file_path);
if (file_it == files_.end()) {
callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND);
return;
}
FileEntry& file = file_it->second;
if (file.is_directory) {
// Check if the directory is empty. This can be done by checking if
// the next file is present under the directory. Note that |files_| is
// a map hence the file names are sorted by names.
FileIterator tmp_it = file_it;
++tmp_it;
if (tmp_it != files_.end() && file_path.IsParent(tmp_it->first)) {
callback.Run(base::PLATFORM_FILE_ERROR_NOT_A_FILE);
return;
}
}
files_.erase(file_it);
callback.Run(base::PLATFORM_FILE_OK);
}
void MemoryFileUtil::DoRemoveRecursive(
const FilePath& file_path,
const StatusCallback& callback) {
FileIterator file_it = files_.find(file_path);
if (file_it == files_.end()) {
callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND);
return;
}
FileEntry& file = file_it->second;
if (!file.is_directory) {
files_.erase(file_it);
callback.Run(base::PLATFORM_FILE_OK);
return;
}
// Remove the directory itself.
files_.erase(file_it++);
// Remove files under the directory.
while (file_it != files_.end()) {
if (file_path.IsParent(file_it->first)) {
files_.erase(file_it++);
} else {
break;
}
}
callback.Run(base::PLATFORM_FILE_OK);
}
void MemoryFileUtil::DoReadDirectory(
const FilePath& dir_path,
const FilePath& in_from,
const ReadDirectoryCallback& callback) {
FilePath from = in_from;
read_directory_buffer_.clear();
if (!FileExists(dir_path)) {
callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND,
read_directory_buffer_, true);
return;
}
if (!IsDirectory(dir_path)) {
callback.Run(base::PLATFORM_FILE_ERROR_NOT_A_DIRECTORY,
read_directory_buffer_, true);
return;
}
if (from.empty())
from = dir_path;
bool completed = true;
// Here we iterate over all paths in files_ starting with the prefix |from|.
// It is not very efficient in case of a deep tree with many files in
// subdirectories. If ever we'll need efficiency from this implementation of
// FS, this should be changed.
for (ConstFileIterator it = files_.lower_bound(from);
it != files_.end();
++it) {
if (dir_path == it->first) // Skip the directory itself.
continue;
if (!dir_path.IsParent(it->first)) // Not in the directory.
break;
if (it->first.DirName() != dir_path) // a file in subdirectory
continue;
if (read_directory_buffer_.size() == read_directory_buffer_size_) {
from = it->first;
completed = false;
break;
}
const FileEntry& file = it->second;
DirectoryEntry entry;
entry.name = it->first.BaseName().value();
entry.is_directory = file.is_directory;
entry.size = file.contents.size();
entry.last_modified_time = file.last_modified;
read_directory_buffer_.push_back(entry);
}
callback.Run(base::PLATFORM_FILE_OK, read_directory_buffer_, completed);
if (!completed) {
MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&MemoryFileUtil::DoReadDirectory,
base::Unretained(this), dir_path,
from, callback));
}
}
void MemoryFileUtil::OpenVerifiedFile(
const FilePath& file_path,
int flags,
const OpenCallback& callback) {
FileIterator file_it = files_.find(file_path);
// The existence of the file is guranteed here.
DCHECK(file_it != files_.end());
FileEntry* file_entry = &file_it->second;
callback.Run(base::PLATFORM_FILE_OK,
new MemoryFileUtilAsyncFileStream(file_entry, flags));
}
void MemoryFileUtil::DidGetFileInfoForOpen(
const FilePath& file_path,
int flags,
const OpenCallback& callback,
PlatformFileError get_info_result,
const base::PlatformFileInfo& file_info) {
if (get_info_result == base::PLATFORM_FILE_OK && file_info.is_directory) {
callback.Run(base::PLATFORM_FILE_ERROR_NOT_A_FILE, NULL);
return;
}
if (get_info_result == base::PLATFORM_FILE_OK) {
OpenVerifiedFile(file_path, flags, callback);
return;
}
if (get_info_result == base::PLATFORM_FILE_ERROR_NOT_FOUND &&
flags & base::PLATFORM_FILE_CREATE_ALWAYS) {
Create(file_path,
base::Bind(&MemoryFileUtil::DidCreateOrTruncateForOpen,
base::Unretained(this), file_path, flags, 0, callback));
return;
}
callback.Run(get_info_result, NULL);
}
void MemoryFileUtil::OpenTruncatedFileOrCreate(
const FilePath& file_path,
int flags,
const OpenCallback& callback,
PlatformFileError result) {
if (result == base::PLATFORM_FILE_OK) {
OpenVerifiedFile(file_path, flags, callback);
return;
}
if (result == base::PLATFORM_FILE_ERROR_NOT_FOUND) {
Create(
file_path,
base::Bind(&MemoryFileUtil::DidCreateOrTruncateForOpen,
base::Unretained(this), file_path, flags, 0, callback));
return;
}
callback.Run(result, NULL);
}
void MemoryFileUtil::DidCreateOrTruncateForOpen(
const FilePath& file_path,
int flags,
int64 size,
const OpenCallback& callback,
PlatformFileError result) {
if (result != base::PLATFORM_FILE_OK) {
callback.Run(result, NULL);
return;
}
OpenVerifiedFile(file_path, flags, callback);
}
} // namespace fileapi