Chromium Code Reviews| Index: base/file_util_proxy_unittest.cc |
| diff --git a/base/file_util_proxy_unittest.cc b/base/file_util_proxy_unittest.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..803a7c66b84b8e13a971a7fe0396876f7c506dbd |
| --- /dev/null |
| +++ b/base/file_util_proxy_unittest.cc |
| @@ -0,0 +1,406 @@ |
| +// 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 <map> |
| + |
| +#include "base/bind.h" |
| +#include "base/file_util_proxy.h" |
|
willchan no longer on Chromium
2012/04/21 06:58:39
This should go above <map> according to style guid
kinuko
2012/04/23 09:01:34
Done.
|
| +#include "base/logging.h" |
| +#include "base/memory/weak_ptr.h" |
| +#include "base/message_loop.h" |
| +#include "base/platform_file.h" |
| +#include "base/scoped_temp_dir.h" |
| +#include "base/threading/thread.h" |
| +#include "testing/gtest/include/gtest/gtest.h" |
| + |
| +namespace base { |
| + |
| +class FileUtilProxyTest : public testing::Test { |
| + public: |
| + FileUtilProxyTest() |
| + : message_loop_(MessageLoop::TYPE_IO), |
| + file_thread_("FileUtilProxyTestFileThread"), |
| + error_(PLATFORM_FILE_OK), |
| + created_(false), |
| + file_(kInvalidPlatformFileValue), |
| + bytes_written_(-1), |
| + weak_factory_(ALLOW_THIS_IN_INITIALIZER_LIST(this)) {} |
| + |
| + virtual void SetUp() OVERRIDE { |
| + ASSERT_TRUE(dir_.CreateUniqueTempDir()); |
| + ASSERT_TRUE(file_thread_.Start()); |
| + } |
| + |
| + virtual void TearDown() OVERRIDE { |
| + if (file_ != kInvalidPlatformFileValue) |
| + ClosePlatformFile(file_); |
| + } |
| + |
| + void DidFinish(PlatformFileError error) { |
| + error_ = error; |
| + MessageLoop::current()->Quit(); |
| + } |
| + |
| + void DidCreateOrOpen(PlatformFileError error, |
| + PassPlatformFile file, |
| + bool created) { |
| + error_ = error; |
| + file_ = file.ReleaseValue(); |
| + created_ = created; |
| + MessageLoop::current()->Quit(); |
| + } |
| + |
| + void DidCreateTemporary(PlatformFileError error, |
| + PassPlatformFile file, |
| + const FilePath& path) { |
| + error_ = error; |
| + file_ = file.ReleaseValue(); |
| + path_ = path; |
| + MessageLoop::current()->Quit(); |
| + } |
| + |
| + void DidGetFileInfo(PlatformFileError error, |
| + const PlatformFileInfo& file_info) { |
| + error_ = error; |
| + file_info_ = file_info; |
| + MessageLoop::current()->Quit(); |
| + } |
| + |
| + void DidRead(PlatformFileError error, |
| + const char* data, |
| + int bytes_read) { |
| + error_ = error; |
| + buffer_.resize(bytes_read); |
| + memcpy(&buffer_[0], data, bytes_read); |
| + MessageLoop::current()->Quit(); |
| + } |
| + |
| + void DidWrite(PlatformFileError error, |
| + int bytes_written) { |
| + error_ = error; |
| + bytes_written_ = bytes_written; |
| + MessageLoop::current()->Quit(); |
| + } |
| + |
| + protected: |
| + PlatformFile GetTestPlatformFile(int flags) { |
| + if (file_ != kInvalidPlatformFileValue) |
| + return file_; |
| + bool created; |
| + PlatformFileError error; |
| + file_ = CreatePlatformFile(test_path(), flags, &created, &error); |
| + EXPECT_EQ(PLATFORM_FILE_OK, error); |
| + EXPECT_NE(kInvalidPlatformFileValue, file_); |
| + return file_; |
| + } |
| + |
| + TaskRunner* file_task_runner() const { |
| + return file_thread_.message_loop_proxy().get(); |
| + } |
| + const FilePath& test_dir_path() const { return dir_.path(); } |
| + const FilePath test_path() const { return dir_.path().AppendASCII("test"); } |
| + |
| + MessageLoop message_loop_; |
| + Thread file_thread_; |
| + |
| + ScopedTempDir dir_; |
| + PlatformFileError error_; |
| + bool created_; |
| + PlatformFile file_; |
| + FilePath path_; |
| + PlatformFileInfo file_info_; |
| + std::vector<char> buffer_; |
| + int bytes_written_; |
| + WeakPtrFactory<FileUtilProxyTest> weak_factory_; |
| +}; |
| + |
| +TEST_F(FileUtilProxyTest, CreateOrOpen_Create) { |
| + FileUtilProxy::CreateOrOpen( |
| + file_task_runner(), |
| + test_path(), |
| + PLATFORM_FILE_CREATE | PLATFORM_FILE_READ, |
| + Bind(&FileUtilProxyTest::DidCreateOrOpen, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + |
| + EXPECT_EQ(PLATFORM_FILE_OK, error_); |
| + EXPECT_TRUE(created_); |
| + EXPECT_NE(kInvalidPlatformFileValue, file_); |
| + EXPECT_TRUE(file_util::PathExists(test_path())); |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, CreateOrOpen_Open) { |
| + // Creates a file. |
| + file_util::WriteFile(test_path(), NULL, 0); |
| + ASSERT_TRUE(file_util::PathExists(test_path())); |
| + |
| + // Opens the created file. |
| + FileUtilProxy::CreateOrOpen( |
| + file_task_runner(), |
| + test_path(), |
| + PLATFORM_FILE_OPEN | PLATFORM_FILE_READ, |
| + Bind(&FileUtilProxyTest::DidCreateOrOpen, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + |
| + EXPECT_EQ(PLATFORM_FILE_OK, error_); |
| + EXPECT_FALSE(created_); |
| + EXPECT_NE(kInvalidPlatformFileValue, file_); |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, CreateOrOpen_OpenNonExistent) { |
| + FileUtilProxy::CreateOrOpen( |
| + file_task_runner(), |
| + test_path(), |
| + PLATFORM_FILE_OPEN | PLATFORM_FILE_READ, |
| + Bind(&FileUtilProxyTest::DidCreateOrOpen, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + EXPECT_EQ(PLATFORM_FILE_ERROR_NOT_FOUND, error_); |
| + EXPECT_FALSE(created_); |
| + EXPECT_EQ(kInvalidPlatformFileValue, file_); |
| + EXPECT_FALSE(file_util::PathExists(test_path())); |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, Close) { |
| + // Creates a file. |
| + PlatformFile file = GetTestPlatformFile( |
| + PLATFORM_FILE_CREATE | PLATFORM_FILE_WRITE); |
| + |
| +#if defined(OS_WIN) |
| + // This fails on Windows if the file is not closed. |
| + EXPECT_FALSE(file_util::Move(test_path(), |
| + test_dir_path().AppendASCII("new"))); |
| +#endif |
| + |
| + FileUtilProxy::Close( |
| + file_task_runner(), |
| + file, |
| + Bind(&FileUtilProxyTest::DidFinish, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + EXPECT_EQ(PLATFORM_FILE_OK, error_); |
| + |
| + // Now it should pass on all platforms. |
| + EXPECT_TRUE(file_util::Move(test_path(), test_dir_path().AppendASCII("new"))); |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, CreateTemporary) { |
| + FileUtilProxy::CreateTemporary( |
| + file_task_runner(), 0 /* additional_file_flags */, |
| + Bind(&FileUtilProxyTest::DidCreateTemporary, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + EXPECT_EQ(PLATFORM_FILE_OK, error_); |
| + EXPECT_TRUE(file_util::PathExists(path_)); |
| + EXPECT_NE(kInvalidPlatformFileValue, file_); |
| + |
| + // The file should be writable. |
| +#if defined(OS_WIN) |
| + HANDLE hEvent = CreateEvent(NULL, FALSE, FALSE, NULL); |
| + OVERLAPPED overlapped = {0}; |
| + overlapped.hEvent = hEvent; |
| + DWORD bytes_written; |
| + if (!::WriteFile(file_, "test", 4, &bytes_written, &overlapped)) { |
| + // Temporary file is created with ASYNC flag, so WriteFile may return 0 |
| + // with ERROR_IO_PENDING. |
| + EXPECT_EQ(ERROR_IO_PENDING, GetLastError()); |
| + GetOverlappedResult(file_, &overlapped, &bytes_written, TRUE); |
| + } |
| + EXPECT_EQ(4, bytes_written); |
| +#else |
| + // On POSIX ASYNC flag does not affect synchronous read/write behavior. |
| + EXPECT_EQ(4, WritePlatformFile(file_, 0, "test", 4)); |
| +#endif |
| + EXPECT_TRUE(ClosePlatformFile(file_)); |
| + file_ = kInvalidPlatformFileValue; |
| + |
| + // Make sure the written data can be read from the returned path. |
| + std::string data; |
| + EXPECT_TRUE(file_util::ReadFileToString(path_, &data)); |
| + EXPECT_EQ("test", data); |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, GetFileInfo_File) { |
| + // Setup. |
| + ASSERT_EQ(4, file_util::WriteFile(test_path(), "test", 4)); |
| + PlatformFileInfo expected_info; |
| + file_util::GetFileInfo(test_path(), &expected_info); |
| + |
| + // Run. |
| + FileUtilProxy::GetFileInfo( |
| + file_task_runner(), |
| + test_path(), |
| + Bind(&FileUtilProxyTest::DidGetFileInfo, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + |
| + // Verify. |
| + EXPECT_EQ(PLATFORM_FILE_OK, error_); |
| + EXPECT_EQ(expected_info.size, file_info_.size); |
| + EXPECT_EQ(expected_info.is_directory, file_info_.is_directory); |
| + EXPECT_EQ(expected_info.is_symbolic_link, file_info_.is_symbolic_link); |
| + EXPECT_EQ(expected_info.last_modified, file_info_.last_modified); |
| + EXPECT_EQ(expected_info.last_accessed, file_info_.last_accessed); |
| + EXPECT_EQ(expected_info.creation_time, file_info_.creation_time); |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, GetFileInfo_Directory) { |
| + // Setup. |
| + ASSERT_TRUE(file_util::CreateDirectory(test_path())); |
| + PlatformFileInfo expected_info; |
| + file_util::GetFileInfo(test_path(), &expected_info); |
| + |
| + // Run. |
| + FileUtilProxy::GetFileInfo( |
| + file_task_runner(), |
| + test_path(), |
| + Bind(&FileUtilProxyTest::DidGetFileInfo, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + |
| + // Verify. |
| + EXPECT_EQ(PLATFORM_FILE_OK, error_); |
| + EXPECT_EQ(expected_info.size, file_info_.size); |
| + EXPECT_EQ(expected_info.is_directory, file_info_.is_directory); |
| + EXPECT_EQ(expected_info.is_symbolic_link, file_info_.is_symbolic_link); |
| + EXPECT_EQ(expected_info.last_modified, file_info_.last_modified); |
| + EXPECT_EQ(expected_info.last_accessed, file_info_.last_accessed); |
| + EXPECT_EQ(expected_info.creation_time, file_info_.creation_time); |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, Read) { |
| + // Setup. |
| + const char expected_data[] = "bleh"; |
| + int expected_bytes = ARRAYSIZE_UNSAFE(expected_data); |
|
willchan no longer on Chromium
2012/04/21 06:58:39
You should be able to use arraysize(), not ARRAYSI
kinuko
2012/04/23 09:01:34
Done.
|
| + ASSERT_EQ(expected_bytes, |
| + file_util::WriteFile(test_path(), expected_data, expected_bytes)); |
| + |
| + // Run. |
| + FileUtilProxy::Read( |
| + file_task_runner(), |
| + GetTestPlatformFile(PLATFORM_FILE_OPEN | PLATFORM_FILE_READ), |
| + 0, // offset |
| + 128, |
| + Bind(&FileUtilProxyTest::DidRead, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + |
| + // Verify. |
| + EXPECT_EQ(PLATFORM_FILE_OK, error_); |
| + EXPECT_EQ(expected_bytes, static_cast<int>(buffer_.size())); |
| + for (size_t i = 0; i < buffer_.size(); ++i) { |
| + EXPECT_EQ(expected_data[i], buffer_[i]); |
| + } |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, WriteAndFlush) { |
| + const char data[] = "foo!"; |
| + int data_bytes = ARRAYSIZE_UNSAFE(data); |
| + PlatformFile file = GetTestPlatformFile( |
| + PLATFORM_FILE_CREATE | PLATFORM_FILE_WRITE); |
| + |
| + FileUtilProxy::Write( |
| + file_task_runner(), |
| + file, |
| + 0, // offset |
| + data, |
| + data_bytes, |
| + Bind(&FileUtilProxyTest::DidWrite, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + EXPECT_EQ(PLATFORM_FILE_OK, error_); |
| + EXPECT_EQ(data_bytes, bytes_written_); |
| + |
| + // Flush the written data. (So that the following read should always |
| + // succeed. On some platforms it may work with or without this flush.) |
| + FileUtilProxy::Flush( |
| + file_task_runner(), |
| + file, |
| + Bind(&FileUtilProxyTest::DidFinish, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + EXPECT_EQ(PLATFORM_FILE_OK, error_); |
| + |
| + // Verify the written data. |
| + char buffer[10]; |
| + EXPECT_EQ(data_bytes, file_util::ReadFile(test_path(), buffer, data_bytes)); |
| + for (int i = 0; i < data_bytes; ++i) { |
| + EXPECT_EQ(data[i], buffer[i]); |
| + } |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, Touch) { |
| + Time last_accessed_time = Time::Now() - TimeDelta::FromDays(12345); |
| + Time last_modified_time = Time::Now() - TimeDelta::FromHours(98765); |
| + |
| + FileUtilProxy::Touch( |
| + file_task_runner(), |
| + GetTestPlatformFile(PLATFORM_FILE_CREATE | |
| + PLATFORM_FILE_WRITE | |
| + PLATFORM_FILE_WRITE_ATTRIBUTES), |
| + last_accessed_time, |
| + last_modified_time, |
| + Bind(&FileUtilProxyTest::DidFinish, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + EXPECT_EQ(PLATFORM_FILE_OK, error_); |
| + |
| + PlatformFileInfo info; |
| + file_util::GetFileInfo(test_path(), &info); |
| + |
| + // The returned values may only have the seconds precision, so we cast |
| + // the double values to int here. |
| + EXPECT_EQ(static_cast<int>(last_modified_time.ToDoubleT()), |
| + static_cast<int>(info.last_modified.ToDoubleT())); |
| + EXPECT_EQ(static_cast<int>(last_accessed_time.ToDoubleT()), |
| + static_cast<int>(info.last_accessed.ToDoubleT())); |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, Truncate_Shrink) { |
| + // Setup. |
| + const char kTestData[] = "0123456789"; |
| + ASSERT_EQ(10, file_util::WriteFile(test_path(), kTestData, 10)); |
| + PlatformFileInfo info; |
| + file_util::GetFileInfo(test_path(), &info); |
| + ASSERT_EQ(10, info.size); |
| + |
| + // Run. |
| + FileUtilProxy::Truncate( |
| + file_task_runner(), |
| + GetTestPlatformFile(PLATFORM_FILE_OPEN | PLATFORM_FILE_WRITE), |
| + 7, |
| + Bind(&FileUtilProxyTest::DidFinish, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + |
| + // Verify. |
| + file_util::GetFileInfo(test_path(), &info); |
| + ASSERT_EQ(7, info.size); |
| + |
| + char buffer[7]; |
| + EXPECT_EQ(7, file_util::ReadFile(test_path(), buffer, 7)); |
| + int i = 0; |
| + for (; i < 7; ++i) |
| + EXPECT_EQ(kTestData[i], buffer[i]); |
| +} |
| + |
| +TEST_F(FileUtilProxyTest, Truncate_Expand) { |
| + // Setup. |
| + const char kTestData[] = "9876543210"; |
| + ASSERT_EQ(10, file_util::WriteFile(test_path(), kTestData, 10)); |
| + PlatformFileInfo info; |
| + file_util::GetFileInfo(test_path(), &info); |
| + ASSERT_EQ(10, info.size); |
| + |
| + // Run. |
| + FileUtilProxy::Truncate( |
| + file_task_runner(), |
| + GetTestPlatformFile(PLATFORM_FILE_OPEN | PLATFORM_FILE_WRITE), |
| + 53, |
| + Bind(&FileUtilProxyTest::DidFinish, weak_factory_.GetWeakPtr())); |
| + MessageLoop::current()->Run(); |
| + |
| + // Verify. |
| + file_util::GetFileInfo(test_path(), &info); |
| + ASSERT_EQ(53, info.size); |
| + |
| + char buffer[53]; |
| + EXPECT_EQ(53, file_util::ReadFile(test_path(), buffer, 53)); |
| + int i = 0; |
| + for (; i < 10; ++i) |
| + EXPECT_EQ(kTestData[i], buffer[i]); |
| + for (; i < 53; ++i) |
| + EXPECT_EQ(0, buffer[i]); |
| +} |
| + |
| +} // namespace base |