// Copyright 2014 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 "content/browser/loader/temporary_file_stream.h" #include #include #include #include "base/bind.h" #include "base/files/file_path.h" #include "base/files/file_util.h" #include "base/macros.h" #include "base/run_loop.h" #include "content/public/test/test_browser_thread_bundle.h" #include "net/base/file_stream.h" #include "net/base/io_buffer.h" #include "net/base/net_errors.h" #include "net/base/test_completion_callback.h" #include "storage/browser/blob/shareable_file_reference.h" #include "testing/gtest/include/gtest/gtest.h" using storage::ShareableFileReference; namespace content { namespace { const char kTestData[] = "0123456789"; const int kTestDataSize = arraysize(kTestData) - 1; class WaitForFileStream { public: base::File::Error error() const { return error_; } net::FileStream* file_stream() const { return file_stream_.get(); } ShareableFileReference* deletable_file() const { return deletable_file_.get(); } void OnFileStreamCreated(base::File::Error error, scoped_ptr file_stream, ShareableFileReference* deletable_file) { error_ = error; file_stream_ = std::move(file_stream); deletable_file_ = deletable_file; loop_.Quit(); } void Wait() { loop_.Run(); } void Release() { file_stream_.reset(NULL); deletable_file_ = NULL; } private: base::RunLoop loop_; base::File::Error error_; scoped_ptr file_stream_; scoped_refptr deletable_file_; }; } // namespace TEST(TemporaryFileStreamTest, Basic) { TestBrowserThreadBundle thread_bundle(TestBrowserThreadBundle::IO_MAINLOOP); // Create a temporary. WaitForFileStream file_stream_waiter; CreateTemporaryFileStream(base::Bind(&WaitForFileStream::OnFileStreamCreated, base::Unretained(&file_stream_waiter))); file_stream_waiter.Wait(); // The temporary should exist. EXPECT_EQ(base::File::FILE_OK, file_stream_waiter.error()); base::FilePath file_path = file_stream_waiter.deletable_file()->path(); EXPECT_TRUE(base::PathExists(file_path)); // Write some data to the temporary. int bytes_written = 0; scoped_refptr buf = new net::IOBufferWithSize(kTestDataSize); memcpy(buf->data(), kTestData, kTestDataSize); scoped_refptr drainable = new net::DrainableIOBuffer(buf.get(), buf->size()); while (bytes_written != kTestDataSize) { net::TestCompletionCallback write_callback; int rv = file_stream_waiter.file_stream()->Write( drainable.get(), drainable->BytesRemaining(), write_callback.callback()); if (rv == net::ERR_IO_PENDING) rv = write_callback.WaitForResult(); ASSERT_LT(0, rv); drainable->DidConsume(rv); bytes_written += rv; } // Verify the data matches. std::string contents; ASSERT_TRUE(base::ReadFileToString(file_path, &contents)); EXPECT_EQ(kTestData, contents); // Close the file. net::TestCompletionCallback close_callback; int rv = file_stream_waiter.file_stream()->Close(close_callback.callback()); if (rv == net::ERR_IO_PENDING) rv = close_callback.WaitForResult(); EXPECT_EQ(net::OK, rv); // Release everything. The file should be gone now. file_stream_waiter.Release(); base::MessageLoop::current()->RunUntilIdle(); // The temporary should be gone now. EXPECT_FALSE(base::PathExists(file_path)); } } // content