blob: 8011de244757cc80ab06a3d6ac711c327f127ba6 [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 "base/bind.h"
#include "base/message_loop.h"
#include "base/platform_file.h"
#include "base/time.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "webkit/chromeos/fileapi/memory_file_util.h"
namespace {
const FilePath::CharType kRootPath[] = "/mnt/memory";
const char kTestString[] = "A test string. A test string.";
const char kTestStringLength = arraysize(kTestString) - 1;
} // namespace
namespace fileapi {
// This test is actually testing MemoryFileUtil — an async in-memory file
// system, based on FileUtilAsync.
class MemoryFileUtilTest : public testing::Test {
public:
MemoryFileUtilTest() : max_request_id_(0) {
}
~MemoryFileUtilTest() {
for (std::map<int, CallbackStatus>::iterator iter = status_map_.begin();
iter != status_map_.end();
++iter) {
delete iter->second.file_stream;
}
}
void SetUp() {
file_util_.reset(new MemoryFileUtil(FilePath(kRootPath)));
}
MemoryFileUtil* file_util() {
return file_util_.get();
}
enum CallbackType {
CALLBACK_TYPE_ERROR,
CALLBACK_TYPE_STATUS,
CALLBACK_TYPE_GET_FILE_INFO,
CALLBACK_TYPE_OPEN,
CALLBACK_TYPE_READ_WRITE,
CALLBACK_TYPE_READ_DIRECTORY
};
struct CallbackStatus {
CallbackStatus()
: type(CALLBACK_TYPE_ERROR),
result(base::PLATFORM_FILE_OK),
file_stream(NULL),
length(-1),
completed(false),
called_after_completed(false),
called(0) {}
CallbackType type;
base::PlatformFileError result;
base::PlatformFileInfo file_info;
// This object should be deleted in the test code.
AsyncFileStream* file_stream;
int64 length;
// Following 4 fields only for ReadDirectory.
FileUtilAsync::FileList entries;
bool completed;
bool called_after_completed;
int called;
};
FileUtilAsync::StatusCallback GetStatusCallback(int request_id) {
return base::Bind(&MemoryFileUtilTest::StatusCallbackImpl,
base::Unretained(this),
request_id);
}
FileUtilAsync::GetFileInfoCallback GetGetFileInfoCallback(
int request_id) {
return base::Bind(&MemoryFileUtilTest::GetFileInfoCallback,
base::Unretained(this),
request_id);
}
FileUtilAsync::OpenCallback GetOpenCallback(int request_id) {
return base::Bind(&MemoryFileUtilTest::OpenCallback,
base::Unretained(this),
request_id);
}
AsyncFileStream::ReadWriteCallback GetReadWriteCallback(int request_id) {
return base::Bind(&MemoryFileUtilTest::ReadWriteCallbackImpl,
base::Unretained(this),
request_id);
}
FileUtilAsync::ReadDirectoryCallback GetReadDirectoryCallback(
int request_id) {
return base::Bind(&MemoryFileUtilTest::ReadDirectoryCallback,
base::Unretained(this),
request_id);
}
int CreateEmptyFile(const FilePath& file_path) {
int request_id = GetNextRequestId();
file_util_->Create(file_path, GetStatusCallback(request_id));
return request_id;
}
int CreateNonEmptyFile(const FilePath& file_path,
const char* data,
int length) {
int request_id = GetNextRequestId();
file_util_->Open(
file_path,
base::PLATFORM_FILE_CREATE | base::PLATFORM_FILE_WRITE,
base::Bind(&MemoryFileUtilTest::WriteToOpenedFile,
base::Unretained(this),
request_id, data, length));
return request_id;
}
CallbackType GetStatusType(int request_id) {
if (status_map_.find(request_id) == status_map_.end())
return CALLBACK_TYPE_ERROR;
return status_map_[request_id].type;
}
// Return the operation status.
CallbackStatus& GetStatus(int request_id) {
return status_map_[request_id];
}
int StatusQueueSize() {
return status_map_.size();
}
int GetNextRequestId() {
return ++max_request_id_;
}
void set_read_directory_buffer_size(int size) {
file_util_->set_read_directory_buffer_size(size);
}
private:
void StatusCallbackImpl(int request_id, PlatformFileError result) {
CallbackStatus status;
status.type = CALLBACK_TYPE_STATUS;
status.result = result;
status_map_[request_id] = status;
}
void GetFileInfoCallback(int request_id,
PlatformFileError result,
const base::PlatformFileInfo& file_info) {
CallbackStatus status;
status.type = CALLBACK_TYPE_GET_FILE_INFO;
status.result = result;
status.file_info = file_info;
status_map_[request_id] = status;
}
void OpenCallback(int request_id,
PlatformFileError result,
AsyncFileStream* file_stream) {
DCHECK(status_map_.find(request_id) == status_map_.end());
CallbackStatus status;
status.type = CALLBACK_TYPE_OPEN;
status.result = result;
status.file_stream = file_stream;
status_map_[request_id] = status;
}
void ReadWriteCallbackImpl(int request_id,
PlatformFileError result,
int64 length) {
CallbackStatus status;
status.type = CALLBACK_TYPE_READ_WRITE;
status.result = result;
status.length = length;
status_map_[request_id] = status;
}
void ReadDirectoryCallback(int request_id,
PlatformFileError result,
const FileUtilAsync::FileList& entries,
bool completed) {
if (status_map_.find(request_id) == status_map_.end()) {
CallbackStatus status;
status.type = CALLBACK_TYPE_READ_DIRECTORY;
status.called_after_completed = false;
status.result = result;
status.completed = completed;
status.called = 1;
status.entries = entries;
status_map_[request_id] = status;
} else {
CallbackStatus& status = status_map_[request_id];
if (status.completed)
status.called_after_completed = true;
status.result = result;
status.completed = completed;
++status.called;
status.entries.insert(status.entries.begin(), entries.begin(),
entries.end());
}
}
void WriteToOpenedFile(int request_id,
const char* data,
int length,
PlatformFileError result,
AsyncFileStream* stream) {
DCHECK(status_map_.find(request_id) == status_map_.end());
CallbackStatus status;
status.type = CALLBACK_TYPE_OPEN;
status.result = result;
status.file_stream = stream;
status_map_[request_id] = status;
stream->Write(data, length, GetReadWriteCallback(GetNextRequestId()));
}
scoped_ptr<MemoryFileUtil> file_util_;
std::map<int, CallbackStatus> status_map_;
int max_request_id_;
DISALLOW_COPY_AND_ASSIGN(MemoryFileUtilTest);
};
TEST_F(MemoryFileUtilTest, TestCreateGetFileInfo) {
const int request_id1 = GetNextRequestId();
file_util()->GetFileInfo(FilePath("/mnt/memory/test.txt"),
GetGetFileInfoCallback(request_id1));
// In case the file system is truely asynchronous, RunAllPending is not
// enough to wait for answer. In that case the thread should be blocked
// until the callback is called (ex. use Run() instead here, and call
// Quit() from callback).
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_GET_FILE_INFO, GetStatusType(request_id1));
CallbackStatus status = GetStatus(request_id1);
ASSERT_EQ(base::PLATFORM_FILE_ERROR_NOT_FOUND, status.result);
base::Time start_create = base::Time::Now();
const int request_id2 = GetNextRequestId();
file_util()->Create(FilePath("/mnt/memory/test.txt"),
GetStatusCallback(request_id2));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_STATUS, GetStatusType(request_id2));
status = GetStatus(request_id2);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
const int request_id3 = GetNextRequestId();
file_util()->GetFileInfo(FilePath("/mnt/memory/test.txt"),
GetGetFileInfoCallback(request_id3));
MessageLoop::current()->RunAllPending();
base::Time end_create = base::Time::Now();
ASSERT_EQ(CALLBACK_TYPE_GET_FILE_INFO, GetStatusType(request_id3));
status = GetStatus(request_id3);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(0, status.file_info.size);
ASSERT_FALSE(status.file_info.is_directory);
ASSERT_FALSE(status.file_info.is_symbolic_link);
ASSERT_GE(status.file_info.last_modified, start_create);
ASSERT_LE(status.file_info.last_modified, end_create);
ASSERT_GE(status.file_info.creation_time, start_create);
ASSERT_LE(status.file_info.creation_time, end_create);
}
TEST_F(MemoryFileUtilTest, TestReadWrite) {
// Check that the file does not exist.
const int request_id1 = GetNextRequestId();
file_util()->GetFileInfo(FilePath("/mnt/memory/test1.txt"),
GetGetFileInfoCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_GET_FILE_INFO, GetStatusType(request_id1));
CallbackStatus status = GetStatus(request_id1);
ASSERT_EQ(base::PLATFORM_FILE_ERROR_NOT_FOUND, status.result);
// Create & open file for writing.
base::Time start_create = base::Time::Now();
const int request_id2 = GetNextRequestId();
file_util()->Open(FilePath("/mnt/memory/test1.txt"),
base::PLATFORM_FILE_CREATE_ALWAYS |
base::PLATFORM_FILE_WRITE,
GetOpenCallback(request_id2));
MessageLoop::current()->RunAllPending();
base::Time end_create = base::Time::Now();
ASSERT_EQ(CALLBACK_TYPE_OPEN, GetStatusType(request_id2));
status = GetStatus(request_id2);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
AsyncFileStream* write_file_stream = status.file_stream;
// Check that file was created and has 0 size.
const int request_id3 = GetNextRequestId();
file_util()->GetFileInfo(FilePath("/mnt/memory/test1.txt"),
GetGetFileInfoCallback(request_id3));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_GET_FILE_INFO, GetStatusType(request_id3));
status = GetStatus(request_id3);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(0, status.file_info.size);
ASSERT_FALSE(status.file_info.is_directory);
ASSERT_FALSE(status.file_info.is_symbolic_link);
ASSERT_GE(status.file_info.last_modified, start_create);
ASSERT_LE(status.file_info.last_modified, end_create);
// Write 10 bytes to file.
const int request_id4 = GetNextRequestId();
base::Time start_write = base::Time::Now();
write_file_stream->Write(kTestString, 10,
GetReadWriteCallback(request_id4));
MessageLoop::current()->RunAllPending();
base::Time end_write = base::Time::Now();
ASSERT_EQ(CALLBACK_TYPE_READ_WRITE, GetStatusType(request_id4));
status = GetStatus(request_id4);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(10, status.length);
// Check that the file has now size 10 and correct modification time.
const int request_id5 = GetNextRequestId();
file_util()->GetFileInfo(FilePath("/mnt/memory/test1.txt"),
GetGetFileInfoCallback(request_id5));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_GET_FILE_INFO, GetStatusType(request_id5));
status = GetStatus(request_id5);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(10, status.file_info.size);
ASSERT_GE(status.file_info.last_modified, start_write);
ASSERT_LE(status.file_info.last_modified, end_write);
// Write the rest of the string to file.
const int request_id6 = GetNextRequestId();
start_write = base::Time::Now();
write_file_stream->Write(kTestString + 10,
kTestStringLength - 10,
GetReadWriteCallback(request_id6));
MessageLoop::current()->RunAllPending();
end_write = base::Time::Now();
ASSERT_EQ(CALLBACK_TYPE_READ_WRITE, GetStatusType(request_id6));
status = GetStatus(request_id6);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(static_cast<int64>(kTestStringLength) - 10, status.length);
// Check the file size & modification time.
const int request_id7 = GetNextRequestId();
file_util()->GetFileInfo(FilePath("/mnt/memory/test1.txt"),
GetGetFileInfoCallback(request_id7));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_GET_FILE_INFO, GetStatusType(request_id7));
status = GetStatus(request_id7);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(static_cast<int64>(kTestStringLength), status.file_info.size);
ASSERT_GE(status.file_info.last_modified, start_write);
ASSERT_LE(status.file_info.last_modified, end_write);
// Open file for reading.
const int request_id8 = GetNextRequestId();
file_util()->Open(FilePath("/mnt/memory/test1.txt"),
base::PLATFORM_FILE_OPEN | base::PLATFORM_FILE_READ,
GetOpenCallback(request_id8));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_OPEN, GetStatusType(request_id8));
status = GetStatus(request_id8);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
AsyncFileStream* read_file_stream = status.file_stream;
// Read the whole file
char buffer[1024];
const int request_id9 = GetNextRequestId();
read_file_stream->Read(buffer, 1023, GetReadWriteCallback(request_id9));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_READ_WRITE, GetStatusType(request_id9));
status = GetStatus(request_id9);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(static_cast<int64>(kTestStringLength), status.length);
buffer[status.length] = '\0';
std::string result_string(buffer);
ASSERT_EQ(kTestString, result_string);
// Check that size & modification time have not changed.
const int request_id10 = GetNextRequestId();
file_util()->GetFileInfo(FilePath("/mnt/memory/test1.txt"),
GetGetFileInfoCallback(request_id10));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_GET_FILE_INFO, GetStatusType(request_id10));
status = GetStatus(request_id10);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(static_cast<int64>(kTestStringLength), status.file_info.size);
ASSERT_GE(status.file_info.last_modified, start_write);
ASSERT_LE(status.file_info.last_modified, end_write);
// Open once more for writing.
const int request_id11 = GetNextRequestId();
file_util()->Open(FilePath("/mnt/memory/test1.txt"),
base::PLATFORM_FILE_OPEN | base::PLATFORM_FILE_WRITE,
GetOpenCallback(request_id11));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_OPEN, GetStatusType(request_id11));
status = GetStatus(request_id11);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
AsyncFileStream* write_file_stream2 = status.file_stream;
// Check that the size has not changed.
const int request_id12 = GetNextRequestId();
file_util()->GetFileInfo(FilePath("/mnt/memory/test1.txt"),
GetGetFileInfoCallback(request_id12));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_GET_FILE_INFO, GetStatusType(request_id12));
status = GetStatus(request_id12);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(static_cast<int64>(kTestStringLength), status.file_info.size);
// Seek beyond the end of file. Should return error.
const int request_id13 = GetNextRequestId();
write_file_stream2->Seek(1000, GetStatusCallback(request_id13));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_STATUS, GetStatusType(request_id13));
status = GetStatus(request_id13);
ASSERT_EQ(base::PLATFORM_FILE_ERROR_INVALID_OPERATION, status.result);
// Try to write to read-only stream.
const int request_id14 = GetNextRequestId();
read_file_stream->Write(kTestString,
kTestStringLength,
GetReadWriteCallback(request_id14));
MessageLoop::current()->RunAllPending();
status = GetStatus(request_id14);
ASSERT_EQ(base::PLATFORM_FILE_ERROR_INVALID_OPERATION, status.result);
// Write data overlapping with already written.
const int request_id15 = GetNextRequestId();
write_file_stream2->Seek(10, GetStatusCallback(request_id15));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id15).result);
const int request_id16 = GetNextRequestId();
write_file_stream2->Write(kTestString,
kTestStringLength,
GetReadWriteCallback(request_id16));
MessageLoop::current()->RunAllPending();
status = GetStatus(request_id16);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(static_cast<int64>(kTestStringLength), status.length);
// Check size.
const int request_id17 = GetNextRequestId();
file_util()->GetFileInfo(FilePath("/mnt/memory/test1.txt"),
GetGetFileInfoCallback(request_id17));
MessageLoop::current()->RunAllPending();
status = GetStatus(request_id17);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(static_cast<int64>(kTestStringLength) + 10,
status.file_info.size);
// Read from 10th byte.
const int request_id18 = GetNextRequestId();
read_file_stream->Seek(10, GetStatusCallback(request_id18));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id18).result);
const int request_id19 = GetNextRequestId();
read_file_stream->Read(buffer, 1023, GetReadWriteCallback(request_id19));
MessageLoop::current()->RunAllPending();
status = GetStatus(request_id19);
ASSERT_EQ(base::PLATFORM_FILE_OK, status.result);
ASSERT_EQ(static_cast<int64>(kTestStringLength), status.length);
buffer[status.length] = '\0';
std::string result_string2(buffer);
ASSERT_EQ(kTestString, result_string2);
}
// The directory structure we'll be testing on:
// path size
//
// /mnt/memory/a 0
// /mnt/memory/b/
// /mnt/memory/b/c kTestStringLength
// /mnt/memory/b/d/
// /mnt/memory/b/e 0
// /mnt/memory/b/f kTestStringLength
// /mnt/memory/b/g/
// /mnt/memory/b/g/h 0
// /mnt/memory/b/i kTestStringLength
// /mnt/memory/c/
// /mnt/memory/longer_file_name.txt kTestStringLength
TEST_F(MemoryFileUtilTest, TestDirectoryOperations) {
// Check the directory is empty.
const int request_id0 = GetNextRequestId();
file_util()->ReadDirectory(FilePath("/mnt/memory/"),
GetReadDirectoryCallback(request_id0));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(CALLBACK_TYPE_READ_DIRECTORY, GetStatusType(request_id0));
CallbackStatus& status = GetStatus(request_id0);
ASSERT_TRUE(status.completed);
ASSERT_FALSE(status.called_after_completed);
ASSERT_EQ(1, status.called);
ASSERT_EQ(0u, status.entries.size());
// Create /mnt/memory/a, /mnt/memory/b/, /mnt/memory/longer_file_name.txt,
// /mnt/memory/c/ asyncronously (i.e. we do not wait for each operation to
// complete before starting the next one.
base::Time start_create = base::Time::Now();
CreateEmptyFile(FilePath("/mnt/memory/a"));
int request_id1 = GetNextRequestId();
file_util()->CreateDirectory(FilePath("/mnt/memory/b"),
GetStatusCallback(request_id1));
CreateNonEmptyFile(FilePath("/mnt/memory/longer_file_name.txt"),
kTestString,
kTestStringLength);
int request_id2 = GetNextRequestId();
file_util()->CreateDirectory(FilePath("/mnt/memory/c"),
GetStatusCallback(request_id2));
MessageLoop::current()->RunAllPending();
base::Time end_create = base::Time::Now();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id1).result);
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id2).result);
// ReadDirectory /mnt/memory, /mnt/memory/a (not a dir), /mnt/memory/b/,
// /mnt/memory/d (not found)
set_read_directory_buffer_size(5); // Should complete in one go.
request_id1 = GetNextRequestId();
file_util()->ReadDirectory(FilePath("/mnt/memory"),
GetReadDirectoryCallback(request_id1));
request_id2 = GetNextRequestId();
file_util()->ReadDirectory(FilePath("/mnt/memory/a"),
GetReadDirectoryCallback(request_id2));
const int request_id3 = GetNextRequestId();
file_util()->ReadDirectory(FilePath("/mnt/memory/b/"),
GetReadDirectoryCallback(request_id3));
const int request_id4 = GetNextRequestId();
file_util()->ReadDirectory(FilePath("/mnt/memory/d/"),
GetReadDirectoryCallback(request_id4));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id1).result);
status = GetStatus(request_id1);
ASSERT_TRUE(status.completed);
ASSERT_FALSE(status.called_after_completed);
ASSERT_EQ(1, status.called); // Because the number of entries < 5.
ASSERT_EQ(4u, status.entries.size());
std::set<FilePath::StringType> seen;
for (FileUtilAsync::FileList::const_iterator it = status.entries.begin();
it != status.entries.end();
++it) {
ASSERT_LE(start_create, it->last_modified_time);
ASSERT_GE(end_create, it->last_modified_time);
ASSERT_EQ(seen.end(), seen.find(it->name));
seen.insert(it->name);
if (it->name == "a") {
ASSERT_FALSE(it->is_directory);
ASSERT_EQ(0, it->size);
} else if (it->name == "b") {
ASSERT_TRUE(it->is_directory);
} else if (it->name == "c") {
ASSERT_TRUE(it->is_directory);
} else if (it->name == "longer_file_name.txt") {
ASSERT_FALSE(it->is_directory);
ASSERT_EQ(static_cast<int64>(kTestStringLength), it->size);
} else {
LOG(ERROR) << "Unexpected file: " << it->name;
ASSERT_TRUE(false);
}
}
ASSERT_EQ(base::PLATFORM_FILE_ERROR_NOT_A_DIRECTORY,
GetStatus(request_id2).result);
status = GetStatus(request_id3);
ASSERT_TRUE(status.completed);
ASSERT_FALSE(status.called_after_completed);
ASSERT_EQ(1, status.called);
ASSERT_EQ(0u, status.entries.size());
ASSERT_EQ(base::PLATFORM_FILE_ERROR_NOT_FOUND, GetStatus(request_id4).result);
// Create files & dirs inside b/:
// /mnt/memory/b/c kTestStringLength
// /mnt/memory/b/d/
// /mnt/memory/b/e 0
// /mnt/memory/b/f kTestStringLength
// /mnt/memory/b/g/
// /mnt/memory/b/i kTestStringLength
//
// /mnt/memory/b/g/h 0
start_create = base::Time::Now();
CreateNonEmptyFile(FilePath("/mnt/memory/b/c"),
kTestString,
kTestStringLength);
request_id1 = GetNextRequestId();
file_util()->CreateDirectory(FilePath("/mnt/memory/b/d"),
GetStatusCallback(request_id1));
CreateEmptyFile(FilePath("/mnt/memory/b/e"));
CreateNonEmptyFile(FilePath("/mnt/memory/b/f"),
kTestString,
kTestStringLength);
request_id2 = GetNextRequestId();
file_util()->CreateDirectory(FilePath("/mnt/memory/b/g"),
GetStatusCallback(request_id1));
CreateNonEmptyFile(FilePath("/mnt/memory/b/i"),
kTestString,
kTestStringLength);
MessageLoop::current()->RunAllPending();
CreateEmptyFile(FilePath("/mnt/memory/b/g/h"));
MessageLoop::current()->RunAllPending();
end_create = base::Time::Now();
// Read /mnt/memory and check that the number of entries is unchanged.
request_id1 = GetNextRequestId();
file_util()->ReadDirectory(FilePath("/mnt/memory"),
GetReadDirectoryCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id1).result);
status = GetStatus(request_id1);
ASSERT_TRUE(status.completed);
ASSERT_FALSE(status.called_after_completed);
ASSERT_EQ(1, status.called); // Because the number of entries < 5.
ASSERT_EQ(4u, status.entries.size());
// Read /mnt/memory/b
request_id1 = GetNextRequestId();
file_util()->ReadDirectory(FilePath("/mnt/memory/b"),
GetReadDirectoryCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id1).result);
status = GetStatus(request_id1);
ASSERT_TRUE(status.completed);
ASSERT_FALSE(status.called_after_completed);
ASSERT_EQ(2, status.called); // Because the number of entries > 5.
ASSERT_EQ(6u, status.entries.size());
seen.clear();
for (FileUtilAsync::FileList::const_iterator it = status.entries.begin();
it != status.entries.end();
++it) {
ASSERT_LE(start_create, it->last_modified_time);
ASSERT_GE(end_create, it->last_modified_time);
ASSERT_EQ(seen.end(), seen.find(it->name));
seen.insert(it->name);
// /mnt/memory/b/c kTestStringLength
// /mnt/memory/b/d/
// /mnt/memory/b/e 0
// /mnt/memory/b/f kTestStringLength
// /mnt/memory/b/g/
// /mnt/memory/b/i kTestStringLength
if (it->name == "c") {
ASSERT_FALSE(it->is_directory);
ASSERT_EQ(static_cast<int64>(kTestStringLength), it->size);
} else if (it->name == "d") {
ASSERT_TRUE(it->is_directory);
} else if (it->name == "e") {
ASSERT_FALSE(it->is_directory);
ASSERT_EQ(0, it->size);
} else if (it->name == "f") {
ASSERT_FALSE(it->is_directory);
ASSERT_EQ(static_cast<int64>(kTestStringLength), it->size);
} else if (it->name == "g") {
ASSERT_TRUE(it->is_directory);
} else if (it->name == "i") {
ASSERT_FALSE(it->is_directory);
ASSERT_EQ(static_cast<int64>(kTestStringLength), it->size);
} else {
LOG(ERROR) << "Unexpected file: " << it->name;
ASSERT_TRUE(false);
}
}
// Remove single file: /mnt/memory/b/f
request_id1 = GetNextRequestId();
file_util()->Remove(FilePath("/mnt/memory/b/f"), false /* recursive */,
GetStatusCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id1).result);
// Check the number of files in b/
request_id1 = GetNextRequestId();
file_util()->ReadDirectory(FilePath("/mnt/memory/b"),
GetReadDirectoryCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id1).result);
status = GetStatus(request_id1);
ASSERT_TRUE(status.completed);
ASSERT_FALSE(status.called_after_completed);
ASSERT_EQ(5u, status.entries.size());
// Try remove /mnt/memory/b non-recursively (error)
request_id1 = GetNextRequestId();
file_util()->Remove(FilePath("/mnt/memory/b"), false /* recursive */,
GetStatusCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_ERROR_NOT_A_FILE,
GetStatus(request_id1).result);
// Non-recursively remove empty directory.
request_id1 = GetNextRequestId();
file_util()->Remove(FilePath("/mnt/memory/b/d"), false /* recursive */,
GetStatusCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id1).result);
request_id1 = GetNextRequestId();
file_util()->GetFileInfo(FilePath("/mnt/memory/b/d"),
GetGetFileInfoCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_ERROR_NOT_FOUND, GetStatus(request_id1).result);
// Remove /mnt/memory/b recursively.
request_id1 = GetNextRequestId();
file_util()->Remove(FilePath("/mnt/memory/b"), true /* recursive */,
GetStatusCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id1).result);
// ReadDirectory /mnt/memory/b -> not found
request_id1 = GetNextRequestId();
file_util()->ReadDirectory(FilePath("/mnt/memory/b"),
GetReadDirectoryCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_ERROR_NOT_FOUND, GetStatus(request_id1).result);
// ReadDirectory /mnt/memory
request_id1 = GetNextRequestId();
file_util()->ReadDirectory(FilePath("/mnt/memory"),
GetReadDirectoryCallback(request_id1));
MessageLoop::current()->RunAllPending();
ASSERT_EQ(base::PLATFORM_FILE_OK, GetStatus(request_id1).result);
status = GetStatus(request_id1);
ASSERT_TRUE(status.completed);
ASSERT_FALSE(status.called_after_completed);
ASSERT_EQ(3u, status.entries.size());
seen.clear();
for (FileUtilAsync::FileList::const_iterator it = status.entries.begin();
it != status.entries.end();
++it) {
ASSERT_EQ(seen.end(), seen.find(it->name));
seen.insert(it->name);
if (it->name == "a") {
ASSERT_FALSE(it->is_directory);
ASSERT_EQ(0, it->size);
} else if (it->name == "c") {
ASSERT_TRUE(it->is_directory);
} else if (it->name == "longer_file_name.txt") {
ASSERT_FALSE(it->is_directory);
ASSERT_EQ(static_cast<int64>(kTestStringLength), it->size);
} else {
LOG(ERROR) << "Unexpected file: " << it->name;
ASSERT_TRUE(false);
}
}
}
} // namespace fileapi