Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(354)

Unified Diff: webkit/plugins/ppapi/quota_file_io_unittest.cc

Issue 7433006: Pepper quota support (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: '' Created 9 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « webkit/plugins/ppapi/quota_file_io.cc ('k') | webkit/tools/test_shell/test_shell.gypi » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: webkit/plugins/ppapi/quota_file_io_unittest.cc
diff --git a/webkit/plugins/ppapi/quota_file_io_unittest.cc b/webkit/plugins/ppapi/quota_file_io_unittest.cc
new file mode 100644
index 0000000000000000000000000000000000000000..89b1a808998424a378b3ca2182a2e15686b9fed1
--- /dev/null
+++ b/webkit/plugins/ppapi/quota_file_io_unittest.cc
@@ -0,0 +1,488 @@
+// 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 <deque>
+
+#include "base/basictypes.h"
+#include "base/memory/scoped_callback_factory.h"
+#include "base/message_loop.h"
+#include "base/platform_file.h"
+#include "base/scoped_temp_dir.h"
+#include "base/task.h"
+#include "webkit/plugins/ppapi/mock_plugin_delegate.h"
+#include "webkit/plugins/ppapi/ppapi_unittest.h"
+#include "webkit/plugins/ppapi/quota_file_io.h"
+
+using base::MessageLoopProxy;
+using base::PlatformFile;
+using base::PlatformFileError;
+
+namespace webkit {
+namespace ppapi {
+
+namespace {
+class QuotaMockPluginDelegate : public MockPluginDelegate {
+ public:
+ typedef PluginDelegate::AvailableSpaceCallback Callback;
+
+ QuotaMockPluginDelegate()
+ : available_space_(0),
+ will_update_count_(0),
+ file_thread_(MessageLoopProxy::CreateForCurrentThread()),
+ runnable_factory_(ALLOW_THIS_IN_INITIALIZER_LIST(this)) {
+ }
+ virtual ~QuotaMockPluginDelegate() {}
+
+ virtual scoped_refptr<MessageLoopProxy> GetFileThreadMessageLoopProxy() {
+ return file_thread_;
+ }
+
+ virtual void QueryAvailableSpace(
+ const GURL& origin,
+ quota::StorageType type,
+ Callback* callback) OVERRIDE {
+ DCHECK(callback);
+ MessageLoopProxy::CreateForCurrentThread()->PostTask(
+ FROM_HERE, runnable_factory_.NewRunnableMethod(
+ &QuotaMockPluginDelegate::RunAvailableSpaceCallback, callback));
+ }
+
+ virtual void WillUpdateFile(const GURL& file_path) OVERRIDE {
+ file_path_ = file_path;
+ ++will_update_count_;
+ }
+
+ virtual void DidUpdateFile(const GURL& file_path, int64_t delta) OVERRIDE {
+ ASSERT_EQ(file_path_, file_path);
+ ASSERT_GT(will_update_count_, 0);
+ --will_update_count_;
+ available_space_ -= delta;
+ }
+
+ void set_available_space(int64 available) { available_space_ = available; }
+ int64_t available_space() const { return available_space_; }
+
+ private:
+ void RunAvailableSpaceCallback(Callback* callback) {
+ callback->Run(available_space_);
+ delete callback;
+ }
+
+ int64_t available_space_;
+ int will_update_count_;
+ GURL file_path_;
+ scoped_refptr<MessageLoopProxy> file_thread_;
+ ScopedRunnableMethodFactory<QuotaMockPluginDelegate> runnable_factory_;
+};
+} // namespace
+
+class QuotaFileIOTest : public PpapiUnittest {
+ public:
+ QuotaFileIOTest()
+ : callback_factory_(ALLOW_THIS_IN_INITIALIZER_LIST(this)) {}
+
+ virtual void SetUp() OVERRIDE {
+ PpapiUnittest::SetUp();
+ ASSERT_TRUE(dir_.CreateUniqueTempDir());
+ FilePath path;
+ ASSERT_TRUE(file_util::CreateTemporaryFileInDir(dir_.path(), &path));
+ int file_flags = base::PLATFORM_FILE_OPEN |
+ base::PLATFORM_FILE_READ |
+ base::PLATFORM_FILE_WRITE |
+ base::PLATFORM_FILE_WRITE_ATTRIBUTES;
+ bool created = false;
+ file_ = base::kInvalidPlatformFileValue;
+ PlatformFileError error = base::PLATFORM_FILE_OK;
+ file_ = base::CreatePlatformFile(path, file_flags, &created, &error);
+ ASSERT_EQ(base::PLATFORM_FILE_OK, error);
+ ASSERT_NE(base::kInvalidPlatformFileValue, file_);
+ ASSERT_FALSE(created);
+ quota_file_io_.reset(new QuotaFileIO(
+ instance(), file_, GURL(), PP_FILESYSTEMTYPE_LOCALTEMPORARY));
+ }
+
+ virtual void TearDown() OVERRIDE {
+ quota_file_io_.reset();
+ if (file_ != base::kInvalidPlatformFileValue)
+ base::ClosePlatformFile(file_);
+ PpapiUnittest::TearDown();
+ }
+
+ protected:
+ virtual MockPluginDelegate* NewPluginDelegate() OVERRIDE {
+ return static_cast<MockPluginDelegate*>(new QuotaMockPluginDelegate);
+ }
+
+ void WriteTestBody(bool will_operation) {
+ quota_plugin_delegate()->set_available_space(100);
+ std::string read_buffer;
+
+ // Write 8 bytes at offset 0 (-> length=8).
+ std::string data("12345678");
+ Write(0, data, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(static_cast<int>(data.size()), bytes_written().front());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(100 - 8, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ if (will_operation) {
+ // WillWrite doesn't actually write.
+ EXPECT_EQ(0, GetPlatformFileSize());
+ // Adjust the actual file size to 'fake' write to proceed testing.
+ SetPlatformFileSize(8);
+ } else {
+ EXPECT_EQ(8, GetPlatformFileSize());
+ ReadPlatformFile(&read_buffer);
+ EXPECT_EQ(data, read_buffer);
+ }
+
+ // Write 5 bytes at offset 5 (-> length=10).
+ data = "55555";
+ Write(5, data, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(static_cast<int>(data.size()), bytes_written().front());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(100 - 10, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ if (will_operation) {
+ EXPECT_EQ(8, GetPlatformFileSize());
+ SetPlatformFileSize(10);
+ } else {
+ EXPECT_EQ(10, GetPlatformFileSize());
+ ReadPlatformFile(&read_buffer);
+ EXPECT_EQ("1234555555", read_buffer);
+ }
+
+ // Write 7 bytes at offset 8 (-> length=15).
+ data = "9012345";
+ Write(8, data, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(static_cast<int>(data.size()), bytes_written().front());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(100 - 15, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ if (will_operation) {
+ EXPECT_EQ(10, GetPlatformFileSize());
+ SetPlatformFileSize(15);
+ } else {
+ EXPECT_EQ(15, GetPlatformFileSize());
+ ReadPlatformFile(&read_buffer);
+ EXPECT_EQ("123455559012345", read_buffer);
+ }
+
+ // Write 2 bytes at offset 2 (-> length=15).
+ data = "33";
+ Write(2, data, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(static_cast<int>(data.size()), bytes_written().front());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(100 - 15, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ if (will_operation) {
+ EXPECT_EQ(15, GetPlatformFileSize());
+ } else {
+ EXPECT_EQ(15, GetPlatformFileSize());
+ ReadPlatformFile(&read_buffer);
+ EXPECT_EQ("123355559012345", read_buffer);
+ }
+
+ // Write 4 bytes at offset 20 (-> length=24).
+ data = "XXXX";
+ Write(20, data, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(static_cast<int>(data.size()), bytes_written().front());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(100 - 24, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ if (will_operation) {
+ EXPECT_EQ(15, GetPlatformFileSize());
+ SetPlatformFileSize(24);
+ } else {
+ EXPECT_EQ(24, GetPlatformFileSize());
+ ReadPlatformFile(&read_buffer);
+ EXPECT_EQ(std::string("123355559012345\0\0\0\0\0XXXX", 24), read_buffer);
+ }
+
+ quota_plugin_delegate()->set_available_space(5);
+
+ // Quota error case. Write 7 bytes at offset 23 (-> length is unchanged)
+ data = "ABCDEFG";
+ Write(23, data, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(base::PLATFORM_FILE_ERROR_NO_SPACE, status().front());
+ EXPECT_EQ(5, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ // Overlapping write. Write 6 bytes at offset 2 (-> length is unchanged)
+ data = "ABCDEF";
+ Write(2, data, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(static_cast<int>(data.size()), bytes_written().front());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(5, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ // Overlapping + extending the file size, but within the quota.
+ // Write 6 bytes at offset 23 (-> length=29).
+ Write(23, data, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(static_cast<int>(data.size()), bytes_written().front());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(0, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ if (!will_operation) {
+ EXPECT_EQ(29, GetPlatformFileSize());
+ ReadPlatformFile(&read_buffer);
+ EXPECT_EQ(std::string("12ABCDEF9012345\0\0\0\0\0XXXABCDEF", 29),
+ read_buffer);
+ }
+ }
+
+ void SetLengthTestBody(bool will_operation) {
+ quota_plugin_delegate()->set_available_space(100);
+
+ SetLength(0, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(0, GetPlatformFileSize());
+ EXPECT_EQ(100, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ SetLength(8, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(100 - 8, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ if (will_operation) {
+ EXPECT_EQ(0, GetPlatformFileSize());
+ SetPlatformFileSize(8);
+ } else {
+ EXPECT_EQ(8, GetPlatformFileSize());
+ }
+
+ SetLength(16, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(100 - 16, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ if (will_operation) {
+ EXPECT_EQ(8, GetPlatformFileSize());
+ SetPlatformFileSize(16);
+ } else {
+ EXPECT_EQ(16, GetPlatformFileSize());
+ }
+
+ SetLength(4, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(100 - 4, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ if (will_operation) {
+ EXPECT_EQ(16, GetPlatformFileSize());
+ SetPlatformFileSize(4);
+ } else {
+ EXPECT_EQ(4, GetPlatformFileSize());
+ }
+
+ SetLength(0, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ EXPECT_EQ(100, quota_plugin_delegate()->available_space());
+ reset_results();
+
+ if (will_operation) {
+ EXPECT_EQ(4, GetPlatformFileSize());
+ SetPlatformFileSize(0);
+ } else {
+ EXPECT_EQ(0, GetPlatformFileSize());
+ }
+
+ quota_plugin_delegate()->set_available_space(5);
+
+ // Quota error case.
+ SetLength(7, will_operation);
+ MessageLoop::current()->RunAllPending();
+ ASSERT_EQ(1U, num_results());
+ EXPECT_EQ(base::PLATFORM_FILE_ERROR_NO_SPACE, status().front());
+ EXPECT_EQ(5, quota_plugin_delegate()->available_space());
+ reset_results();
+ }
+
+ QuotaMockPluginDelegate* quota_plugin_delegate() {
+ return static_cast<QuotaMockPluginDelegate*>(delegate());
+ }
+
+ void Write(int64_t offset, const std::string& data, bool will_operation) {
+ if (will_operation) {
+ ASSERT_TRUE(quota_file_io_->WillWrite(
+ offset, data.size(),
+ callback_factory_.NewCallback(
+ &QuotaFileIOTest::DidWrite)));
+ } else {
+ ASSERT_TRUE(quota_file_io_->Write(
+ offset, data.c_str(), data.size(),
+ callback_factory_.NewCallback(
+ &QuotaFileIOTest::DidWrite)));
+ }
+ }
+
+ void SetLength(int64_t length, bool will_operation) {
+ if (will_operation) {
+ ASSERT_TRUE(quota_file_io_->WillSetLength(
+ length,
+ callback_factory_.NewCallback(
+ &QuotaFileIOTest::DidSetLength)));
+ } else {
+ ASSERT_TRUE(quota_file_io_->SetLength(
+ length,
+ callback_factory_.NewCallback(
+ &QuotaFileIOTest::DidSetLength)));
+ }
+ }
+
+ void DidWrite(PlatformFileError status, int bytes_written) {
+ status_.push_back(status);
+ bytes_written_.push_back(bytes_written);
+ }
+
+ void DidSetLength(PlatformFileError status) {
+ status_.push_back(status);
+ }
+
+ size_t num_results() const { return status_.size(); }
+ const std::deque<int>& bytes_written() const { return bytes_written_; }
+ const std::deque<PlatformFileError>& status() const { return status_; }
+
+ void reset_results() {
+ bytes_written_.clear();
+ status_.clear();
+ }
+
+ void pop_result() {
+ bytes_written_.pop_front();
+ status_.pop_front();
+ }
+
+ void ReadPlatformFile(std::string* data) {
+ data->clear();
+ char buf[256];
+ int32_t read_offset = 0;
+ for (;;) {
+ int rv = base::ReadPlatformFile(file_, read_offset, buf, sizeof(buf));
+ ASSERT_GE(rv, 0);
+ if (rv == 0)
+ break;
+ read_offset += rv;
+ data->append(buf, rv);
+ }
+ }
+
+ int64_t GetPlatformFileSize() {
+ base::PlatformFileInfo info;
+ EXPECT_TRUE(base::GetPlatformFileInfo(file_, &info));
+ return info.size;
+ }
+
+ void SetPlatformFileSize(int64_t length) {
+ EXPECT_TRUE(base::TruncatePlatformFile(file_, length));
+ }
+
+ private:
+ ScopedTempDir dir_;
+ PlatformFile file_;
+ scoped_ptr<QuotaFileIO> quota_file_io_;
+ std::deque<int> bytes_written_;
+ std::deque<PlatformFileError> status_;
+ base::ScopedCallbackFactory<QuotaFileIOTest> callback_factory_;
+};
+
+TEST_F(QuotaFileIOTest, Write) {
+ WriteTestBody(false);
+}
+
+TEST_F(QuotaFileIOTest, WillWrite) {
+ WriteTestBody(true);
+}
+
+TEST_F(QuotaFileIOTest, SetLength) {
+ SetLengthTestBody(false);
+}
+
+TEST_F(QuotaFileIOTest, WillSetLength) {
+ SetLengthTestBody(true);
+}
+
+TEST_F(QuotaFileIOTest, ParallelWrites) {
+ quota_plugin_delegate()->set_available_space(22);
+ std::string read_buffer;
+
+ const std::string data1[] = {
+ std::string("12345678"),
+ std::string("55555"),
+ std::string("9012345"),
+ };
+ Write(0, data1[0], false);
+ Write(5, data1[1], false);
+ Write(8, data1[2], false);
+ MessageLoop::current()->RunAllPending();
+
+ ASSERT_EQ(ARRAYSIZE_UNSAFE(data1), num_results());
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(data1); ++i) {
+ EXPECT_EQ(static_cast<int>(data1[i].size()), bytes_written().front());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ pop_result();
+ }
+
+ EXPECT_EQ(22 - 15, quota_plugin_delegate()->available_space());
+ EXPECT_EQ(15, GetPlatformFileSize());
+ ReadPlatformFile(&read_buffer);
+ EXPECT_EQ("123455559012345", read_buffer);
+
+ // The second write will fail for quota error.
+ const std::string data2[] = {
+ std::string("33"),
+ std::string("XXXX"),
+ };
+ Write(2, data2[0], false);
+ Write(20, data2[1], false);
+ MessageLoop::current()->RunAllPending();
+
+ ASSERT_EQ(ARRAYSIZE_UNSAFE(data2), num_results());
+ EXPECT_EQ(static_cast<int>(data2[0].size()), bytes_written().front());
+ EXPECT_EQ(base::PLATFORM_FILE_OK, status().front());
+ pop_result();
+ EXPECT_EQ(0, bytes_written().front());
+ EXPECT_EQ(base::PLATFORM_FILE_ERROR_NO_SPACE, status().front());
+ pop_result();
+
+ EXPECT_EQ(22 - 15, quota_plugin_delegate()->available_space());
+ EXPECT_EQ(15, GetPlatformFileSize());
+ ReadPlatformFile(&read_buffer);
+ EXPECT_EQ("123355559012345", read_buffer);
+}
+
+} // namespace ppapi
+} // namespace webkit
« no previous file with comments | « webkit/plugins/ppapi/quota_file_io.cc ('k') | webkit/tools/test_shell/test_shell.gypi » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698