1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
8 #include "base/basictypes.h"
10 #include "base/bind_helpers.h"
11 #include "base/files/scoped_temp_dir.h"
12 #include "base/message_loop/message_loop.h"
13 #include "base/run_loop.h"
14 #include "content/public/test/async_file_test_helper.h"
15 #include "content/public/test/test_file_system_context.h"
16 #include "net/base/io_buffer.h"
17 #include "net/base/request_priority.h"
18 #include "net/url_request/url_request.h"
19 #include "net/url_request/url_request_context.h"
20 #include "net/url_request/url_request_job.h"
21 #include "net/url_request/url_request_status.h"
22 #include "testing/platform_test.h"
24 #include "webkit/browser/fileapi/file_system_context.h"
25 #include "webkit/browser/fileapi/file_system_quota_util.h"
26 #include "webkit/browser/fileapi/file_writer_delegate.h"
27 #include "webkit/browser/fileapi/sandbox_file_stream_writer.h"
29 using content::AsyncFileTestHelper;
30 using fileapi::FileSystemURL;
31 using fileapi::FileWriterDelegate;
37 const GURL kOrigin("http://example.com");
38 const fileapi::FileSystemType kFileSystemType = fileapi::kFileSystemTypeTest;
40 const char kData[] = "The quick brown fox jumps over the lazy dog.\n";
41 const int kDataSize = ARRAYSIZE_UNSAFE(kData) - 1;
46 : status_(base::File::FILE_OK),
48 write_status_(FileWriterDelegate::SUCCESS_IO_PENDING) {}
50 base::File::Error status() const { return status_; }
51 int64 bytes_written() const { return bytes_written_; }
52 FileWriterDelegate::WriteProgressStatus write_status() const {
56 void DidWrite(base::File::Error status, int64 bytes,
57 FileWriterDelegate::WriteProgressStatus write_status) {
58 write_status_ = write_status;
59 if (status == base::File::FILE_OK) {
60 bytes_written_ += bytes;
61 if (write_status_ != FileWriterDelegate::SUCCESS_IO_PENDING)
62 base::MessageLoop::current()->Quit();
64 EXPECT_EQ(base::File::FILE_OK, status_);
66 base::MessageLoop::current()->Quit();
71 // For post-operation status.
72 base::File::Error status_;
74 FileWriterDelegate::WriteProgressStatus write_status_;
77 } // namespace (anonymous)
79 class FileWriterDelegateTest : public PlatformTest {
81 FileWriterDelegateTest() {}
84 virtual void SetUp() OVERRIDE;
85 virtual void TearDown() OVERRIDE;
88 return file_system_context_->GetQuotaUtil(kFileSystemType)
89 ->GetOriginUsageOnFileTaskRunner(
90 file_system_context_.get(), kOrigin, kFileSystemType);
93 int64 GetFileSizeOnDisk(const char* test_file_path) {
94 // There might be in-flight flush/write.
95 base::MessageLoop::current()->PostTask(
96 FROM_HERE, base::Bind(&base::DoNothing));
97 base::RunLoop().RunUntilIdle();
99 FileSystemURL url = GetFileSystemURL(test_file_path);
100 base::File::Info file_info;
101 EXPECT_EQ(base::File::FILE_OK,
102 AsyncFileTestHelper::GetMetadata(
103 file_system_context_, url, &file_info));
104 return file_info.size;
107 FileSystemURL GetFileSystemURL(const char* file_name) const {
108 return file_system_context_->CreateCrackedFileSystemURL(
109 kOrigin, kFileSystemType, base::FilePath().FromUTF8Unsafe(file_name));
112 FileWriterDelegate* CreateWriterDelegate(
113 const char* test_file_path,
115 int64 allowed_growth) {
116 fileapi::SandboxFileStreamWriter* writer =
117 new fileapi::SandboxFileStreamWriter(
118 file_system_context_.get(),
119 GetFileSystemURL(test_file_path),
121 *file_system_context_->GetUpdateObservers(kFileSystemType));
122 writer->set_default_quota(allowed_growth);
123 return new FileWriterDelegate(
124 scoped_ptr<fileapi::FileStreamWriter>(writer),
125 FileWriterDelegate::FLUSH_ON_COMPLETION);
128 FileWriterDelegate::DelegateWriteCallback GetWriteCallback(Result* result) {
129 return base::Bind(&Result::DidWrite, base::Unretained(result));
132 // Creates and sets up a FileWriterDelegate for writing the given |blob_url|,
133 // and creates a new FileWriterDelegate for the file.
134 void PrepareForWrite(const char* test_file_path,
135 const GURL& blob_url,
137 int64 allowed_growth) {
138 file_writer_delegate_.reset(
139 CreateWriterDelegate(test_file_path, offset, allowed_growth));
140 request_ = empty_context_.CreateRequest(
141 blob_url, net::DEFAULT_PRIORITY, file_writer_delegate_.get(), NULL);
144 static net::URLRequest::ProtocolFactory Factory;
146 // This should be alive until the very end of this instance.
147 base::MessageLoopForIO loop_;
149 scoped_refptr<fileapi::FileSystemContext> file_system_context_;
151 net::URLRequestContext empty_context_;
152 scoped_ptr<FileWriterDelegate> file_writer_delegate_;
153 scoped_ptr<net::URLRequest> request_;
155 base::ScopedTempDir dir_;
157 static const char* content_;
160 const char* FileWriterDelegateTest::content_ = NULL;
164 static std::string g_content;
166 class FileWriterDelegateTestJob : public net::URLRequestJob {
168 FileWriterDelegateTestJob(net::URLRequest* request,
169 net::NetworkDelegate* network_delegate,
170 const std::string& content)
171 : net::URLRequestJob(request, network_delegate),
173 remaining_bytes_(content.length()),
177 virtual void Start() OVERRIDE {
178 base::MessageLoop::current()->PostTask(
180 base::Bind(&FileWriterDelegateTestJob::NotifyHeadersComplete, this));
183 virtual bool ReadRawData(net::IOBuffer* buf,
185 int *bytes_read) OVERRIDE {
186 if (remaining_bytes_ < buf_size)
187 buf_size = static_cast<int>(remaining_bytes_);
189 for (int i = 0; i < buf_size; ++i)
190 buf->data()[i] = content_[cursor_++];
191 remaining_bytes_ -= buf_size;
193 SetStatus(net::URLRequestStatus());
194 *bytes_read = buf_size;
198 virtual int GetResponseCode() const OVERRIDE {
203 virtual ~FileWriterDelegateTestJob() {}
206 std::string content_;
207 int remaining_bytes_;
211 } // namespace (anonymous)
214 net::URLRequestJob* FileWriterDelegateTest::Factory(
215 net::URLRequest* request,
216 net::NetworkDelegate* network_delegate,
217 const std::string& scheme) {
218 return new FileWriterDelegateTestJob(
219 request, network_delegate, FileWriterDelegateTest::content_);
222 void FileWriterDelegateTest::SetUp() {
223 ASSERT_TRUE(dir_.CreateUniqueTempDir());
225 file_system_context_ = CreateFileSystemContextForTesting(
227 ASSERT_EQ(base::File::FILE_OK,
228 AsyncFileTestHelper::CreateFile(
229 file_system_context_, GetFileSystemURL("test")));
230 net::URLRequest::Deprecated::RegisterProtocolFactory("blob", &Factory);
233 void FileWriterDelegateTest::TearDown() {
234 net::URLRequest::Deprecated::RegisterProtocolFactory("blob", NULL);
235 file_system_context_ = NULL;
236 base::RunLoop().RunUntilIdle();
239 TEST_F(FileWriterDelegateTest, WriteSuccessWithoutQuotaLimit) {
240 const GURL kBlobURL("blob:nolimit");
243 PrepareForWrite("test", kBlobURL, 0, kint64max);
246 ASSERT_EQ(0, usage());
247 file_writer_delegate_->Start(request_.Pass(), GetWriteCallback(&result));
248 base::MessageLoop::current()->Run();
250 ASSERT_EQ(FileWriterDelegate::SUCCESS_COMPLETED, result.write_status());
251 file_writer_delegate_.reset();
253 ASSERT_EQ(kDataSize, usage());
254 EXPECT_EQ(GetFileSizeOnDisk("test"), usage());
255 EXPECT_EQ(kDataSize, result.bytes_written());
256 EXPECT_EQ(base::File::FILE_OK, result.status());
259 TEST_F(FileWriterDelegateTest, WriteSuccessWithJustQuota) {
260 const GURL kBlobURL("blob:just");
262 const int64 kAllowedGrowth = kDataSize;
263 PrepareForWrite("test", kBlobURL, 0, kAllowedGrowth);
266 ASSERT_EQ(0, usage());
267 file_writer_delegate_->Start(request_.Pass(), GetWriteCallback(&result));
268 base::MessageLoop::current()->Run();
269 ASSERT_EQ(FileWriterDelegate::SUCCESS_COMPLETED, result.write_status());
270 file_writer_delegate_.reset();
272 ASSERT_EQ(kAllowedGrowth, usage());
273 EXPECT_EQ(GetFileSizeOnDisk("test"), usage());
275 EXPECT_EQ(kAllowedGrowth, result.bytes_written());
276 EXPECT_EQ(base::File::FILE_OK, result.status());
279 TEST_F(FileWriterDelegateTest, DISABLED_WriteFailureByQuota) {
280 const GURL kBlobURL("blob:failure");
282 const int64 kAllowedGrowth = kDataSize - 1;
283 PrepareForWrite("test", kBlobURL, 0, kAllowedGrowth);
286 ASSERT_EQ(0, usage());
287 file_writer_delegate_->Start(request_.Pass(), GetWriteCallback(&result));
288 base::MessageLoop::current()->Run();
289 ASSERT_EQ(FileWriterDelegate::ERROR_WRITE_STARTED, result.write_status());
290 file_writer_delegate_.reset();
292 ASSERT_EQ(kAllowedGrowth, usage());
293 EXPECT_EQ(GetFileSizeOnDisk("test"), usage());
295 EXPECT_EQ(kAllowedGrowth, result.bytes_written());
296 EXPECT_EQ(base::File::FILE_ERROR_NO_SPACE, result.status());
297 ASSERT_EQ(FileWriterDelegate::ERROR_WRITE_STARTED, result.write_status());
300 TEST_F(FileWriterDelegateTest, WriteZeroBytesSuccessfullyWithZeroQuota) {
301 const GURL kBlobURL("blob:zero");
303 int64 kAllowedGrowth = 0;
304 PrepareForWrite("test", kBlobURL, 0, kAllowedGrowth);
307 ASSERT_EQ(0, usage());
308 file_writer_delegate_->Start(request_.Pass(), GetWriteCallback(&result));
309 base::MessageLoop::current()->Run();
310 ASSERT_EQ(FileWriterDelegate::SUCCESS_COMPLETED, result.write_status());
311 file_writer_delegate_.reset();
313 ASSERT_EQ(kAllowedGrowth, usage());
314 EXPECT_EQ(GetFileSizeOnDisk("test"), usage());
316 EXPECT_EQ(kAllowedGrowth, result.bytes_written());
317 EXPECT_EQ(base::File::FILE_OK, result.status());
318 ASSERT_EQ(FileWriterDelegate::SUCCESS_COMPLETED, result.write_status());
321 TEST_F(FileWriterDelegateTest, WriteSuccessWithoutQuotaLimitConcurrent) {
322 scoped_ptr<FileWriterDelegate> file_writer_delegate2;
323 scoped_ptr<net::URLRequest> request2;
325 ASSERT_EQ(base::File::FILE_OK,
326 AsyncFileTestHelper::CreateFile(
327 file_system_context_, GetFileSystemURL("test2")));
329 const GURL kBlobURL("blob:nolimitconcurrent");
330 const GURL kBlobURL2("blob:nolimitconcurrent2");
333 PrepareForWrite("test", kBlobURL, 0, kint64max);
335 // Credate another FileWriterDelegate for concurrent write.
336 file_writer_delegate2.reset(CreateWriterDelegate("test2", 0, kint64max));
337 request2 = empty_context_.CreateRequest(
338 kBlobURL2, net::DEFAULT_PRIORITY, file_writer_delegate2.get(), NULL);
340 Result result, result2;
341 ASSERT_EQ(0, usage());
342 file_writer_delegate_->Start(request_.Pass(), GetWriteCallback(&result));
343 file_writer_delegate2->Start(request2.Pass(), GetWriteCallback(&result2));
344 base::MessageLoop::current()->Run();
345 if (result.write_status() == FileWriterDelegate::SUCCESS_IO_PENDING ||
346 result2.write_status() == FileWriterDelegate::SUCCESS_IO_PENDING)
347 base::MessageLoop::current()->Run();
349 ASSERT_EQ(FileWriterDelegate::SUCCESS_COMPLETED, result.write_status());
350 ASSERT_EQ(FileWriterDelegate::SUCCESS_COMPLETED, result2.write_status());
351 file_writer_delegate_.reset();
352 file_writer_delegate2.reset();
354 ASSERT_EQ(kDataSize * 2, usage());
355 EXPECT_EQ(GetFileSizeOnDisk("test") + GetFileSizeOnDisk("test2"), usage());
357 EXPECT_EQ(kDataSize, result.bytes_written());
358 EXPECT_EQ(base::File::FILE_OK, result.status());
359 EXPECT_EQ(kDataSize, result2.bytes_written());
360 EXPECT_EQ(base::File::FILE_OK, result2.status());
363 TEST_F(FileWriterDelegateTest, WritesWithQuotaAndOffset) {
364 const GURL kBlobURL("blob:failure-with-updated-quota");
367 // Writing kDataSize (=45) bytes data while allowed_growth is 100.
369 int64 allowed_growth = 100;
370 ASSERT_LT(kDataSize, allowed_growth);
371 PrepareForWrite("test", kBlobURL, offset, allowed_growth);
375 ASSERT_EQ(0, usage());
376 file_writer_delegate_->Start(request_.Pass(), GetWriteCallback(&result));
377 base::MessageLoop::current()->Run();
378 ASSERT_EQ(FileWriterDelegate::SUCCESS_COMPLETED, result.write_status());
379 file_writer_delegate_.reset();
381 ASSERT_EQ(kDataSize, usage());
382 EXPECT_EQ(GetFileSizeOnDisk("test"), usage());
383 EXPECT_EQ(kDataSize, result.bytes_written());
384 EXPECT_EQ(base::File::FILE_OK, result.status());
387 // Trying to overwrite kDataSize bytes data while allowed_growth is 20.
390 PrepareForWrite("test", kBlobURL, offset, allowed_growth);
394 file_writer_delegate_->Start(request_.Pass(), GetWriteCallback(&result));
395 base::MessageLoop::current()->Run();
396 EXPECT_EQ(kDataSize, usage());
397 EXPECT_EQ(GetFileSizeOnDisk("test"), usage());
398 EXPECT_EQ(kDataSize, result.bytes_written());
399 EXPECT_EQ(base::File::FILE_OK, result.status());
400 ASSERT_EQ(FileWriterDelegate::SUCCESS_COMPLETED, result.write_status());
403 // Trying to write kDataSize bytes data from offset 25 while
404 // allowed_growth is 55.
407 PrepareForWrite("test", kBlobURL, offset, allowed_growth);
411 file_writer_delegate_->Start(request_.Pass(), GetWriteCallback(&result));
412 base::MessageLoop::current()->Run();
413 ASSERT_EQ(FileWriterDelegate::SUCCESS_COMPLETED, result.write_status());
414 file_writer_delegate_.reset();
416 EXPECT_EQ(offset + kDataSize, usage());
417 EXPECT_EQ(GetFileSizeOnDisk("test"), usage());
418 EXPECT_EQ(kDataSize, result.bytes_written());
419 EXPECT_EQ(base::File::FILE_OK, result.status());
422 // Trying to overwrite 45 bytes data while allowed_growth is -20.
424 allowed_growth = -20;
425 PrepareForWrite("test", kBlobURL, offset, allowed_growth);
426 int64 pre_write_usage = GetFileSizeOnDisk("test");
430 file_writer_delegate_->Start(request_.Pass(), GetWriteCallback(&result));
431 base::MessageLoop::current()->Run();
432 ASSERT_EQ(FileWriterDelegate::SUCCESS_COMPLETED, result.write_status());
433 file_writer_delegate_.reset();
435 EXPECT_EQ(pre_write_usage, usage());
436 EXPECT_EQ(GetFileSizeOnDisk("test"), usage());
437 EXPECT_EQ(kDataSize, result.bytes_written());
438 EXPECT_EQ(base::File::FILE_OK, result.status());
441 // Trying to overwrite 45 bytes data with offset pre_write_usage - 20,
442 // while allowed_growth is 10.
443 const int kOverlap = 20;
444 offset = pre_write_usage - kOverlap;
446 PrepareForWrite("test", kBlobURL, offset, allowed_growth);
450 file_writer_delegate_->Start(request_.Pass(), GetWriteCallback(&result));
451 base::MessageLoop::current()->Run();
452 ASSERT_EQ(FileWriterDelegate::ERROR_WRITE_STARTED, result.write_status());
453 file_writer_delegate_.reset();
455 EXPECT_EQ(pre_write_usage + allowed_growth, usage());
456 EXPECT_EQ(GetFileSizeOnDisk("test"), usage());
457 EXPECT_EQ(kOverlap + allowed_growth, result.bytes_written());
458 EXPECT_EQ(base::File::FILE_ERROR_NO_SPACE, result.status());
462 } // namespace content