|  | // 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 "net/base/elements_upload_data_stream.h" | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <limits> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/bind.h" | 
|  | #include "base/files/file_path.h" | 
|  | #include "base/files/file_util.h" | 
|  | #include "base/files/scoped_temp_dir.h" | 
|  | #include "base/location.h" | 
|  | #include "base/run_loop.h" | 
|  | #include "base/single_thread_task_runner.h" | 
|  | #include "base/stl_util.h" | 
|  | #include "base/strings/string_piece.h" | 
|  | #include "base/threading/thread_task_runner_handle.h" | 
|  | #include "base/time/time.h" | 
|  | #include "net/base/completion_once_callback.h" | 
|  | #include "net/base/io_buffer.h" | 
|  | #include "net/base/net_errors.h" | 
|  | #include "net/base/test_completion_callback.h" | 
|  | #include "net/base/upload_bytes_element_reader.h" | 
|  | #include "net/base/upload_data_stream.h" | 
|  | #include "net/base/upload_file_element_reader.h" | 
|  | #include "net/log/net_log_with_source.h" | 
|  | #include "net/test/gtest_util.h" | 
|  | #include "net/test/test_with_scoped_task_environment.h" | 
|  | #include "starboard/types.h" | 
|  | #include "testing/gmock/include/gmock/gmock.h" | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  | #include "testing/platform_test.h" | 
|  |  | 
|  | using net::test::IsError; | 
|  | using net::test::IsOk; | 
|  |  | 
|  | using ::testing::DoAll; | 
|  | using ::testing::Invoke; | 
|  | using ::testing::Return; | 
|  | using ::testing::_; | 
|  |  | 
|  | namespace net { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | const char kTestData[] = "0123456789"; | 
|  | const size_t kTestDataSize = base::size(kTestData) - 1; | 
|  | const size_t kTestBufferSize = 1 << 14;  // 16KB. | 
|  |  | 
|  | // Reads data from the upload data stream, and returns the data as string. | 
|  | std::string ReadFromUploadDataStream(UploadDataStream* stream) { | 
|  | std::string data_read; | 
|  | scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kTestBufferSize); | 
|  | while (!stream->IsEOF()) { | 
|  | TestCompletionCallback callback; | 
|  | const int result = | 
|  | stream->Read(buf.get(), kTestBufferSize, callback.callback()); | 
|  | const int bytes_read = | 
|  | result != ERR_IO_PENDING ? result : callback.WaitForResult(); | 
|  | data_read.append(buf->data(), bytes_read); | 
|  | } | 
|  | return data_read; | 
|  | } | 
|  |  | 
|  | // A mock class of UploadElementReader. | 
|  | class MockUploadElementReader : public UploadElementReader { | 
|  | public: | 
|  | MockUploadElementReader(int content_length, bool is_in_memory) | 
|  | : content_length_(content_length), | 
|  | bytes_remaining_(content_length), | 
|  | is_in_memory_(is_in_memory), | 
|  | init_result_(OK), | 
|  | read_result_(OK) {} | 
|  |  | 
|  | ~MockUploadElementReader() override = default; | 
|  |  | 
|  | // UploadElementReader overrides. | 
|  | int Init(CompletionOnceCallback callback) override { | 
|  | // This is a back to get around Gmock's lack of support for move-only types. | 
|  | return Init(&callback); | 
|  | } | 
|  | MOCK_METHOD1(Init, int(CompletionOnceCallback* callback)); | 
|  | uint64_t GetContentLength() const override { return content_length_; } | 
|  | uint64_t BytesRemaining() const override { return bytes_remaining_; } | 
|  | bool IsInMemory() const override { return is_in_memory_; } | 
|  | int Read(IOBuffer* buf, | 
|  | int buf_length, | 
|  | CompletionOnceCallback callback) override { | 
|  | return Read(buf, buf_length, &callback); | 
|  | } | 
|  | MOCK_METHOD3(Read, | 
|  | int(IOBuffer* buf, | 
|  | int buf_length, | 
|  | CompletionOnceCallback* callback)); | 
|  |  | 
|  | // Sets expectation to return the specified result from Init() asynchronously. | 
|  | void SetAsyncInitExpectation(int result) { | 
|  | init_result_ = result; | 
|  | EXPECT_CALL(*this, Init(_)) | 
|  | .WillOnce(DoAll(Invoke(this, &MockUploadElementReader::OnInit), | 
|  | Return(ERR_IO_PENDING))); | 
|  | } | 
|  |  | 
|  | // Sets expectation to return the specified result from Read(). | 
|  | void SetReadExpectation(int result) { | 
|  | read_result_ = result; | 
|  | EXPECT_CALL(*this, Read(_, _, _)) | 
|  | .WillOnce(Invoke(this, &MockUploadElementReader::OnRead)); | 
|  | } | 
|  |  | 
|  | private: | 
|  | void OnInit(CompletionOnceCallback* callback) { | 
|  | base::ThreadTaskRunnerHandle::Get()->PostTask( | 
|  | FROM_HERE, base::BindOnce(std::move(*callback), init_result_)); | 
|  | } | 
|  |  | 
|  | int OnRead(IOBuffer* buf, int buf_length, CompletionOnceCallback* callback) { | 
|  | if (read_result_ > 0) | 
|  | bytes_remaining_ = std::max(0, bytes_remaining_ - read_result_); | 
|  | if (IsInMemory()) { | 
|  | return read_result_; | 
|  | } else { | 
|  | base::ThreadTaskRunnerHandle::Get()->PostTask( | 
|  | FROM_HERE, base::BindOnce(std::move(*callback), read_result_)); | 
|  | return ERR_IO_PENDING; | 
|  | } | 
|  | } | 
|  |  | 
|  | int content_length_; | 
|  | int bytes_remaining_; | 
|  | bool is_in_memory_; | 
|  |  | 
|  | // Result value returned from Init(). | 
|  | int init_result_; | 
|  |  | 
|  | // Result value returned from Read(). | 
|  | int read_result_; | 
|  | }; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | class ElementsUploadDataStreamTest : public PlatformTest, | 
|  | public WithScopedTaskEnvironment { | 
|  | public: | 
|  | void SetUp() override { | 
|  | PlatformTest::SetUp(); | 
|  | ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); | 
|  | } | 
|  | ~ElementsUploadDataStreamTest() override { | 
|  | element_readers_.clear(); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  | } | 
|  |  | 
|  | void FileChangedHelper(const base::FilePath& file_path, | 
|  | const base::Time& time, | 
|  | bool error_expected); | 
|  |  | 
|  | base::ScopedTempDir temp_dir_; | 
|  | std::vector<std::unique_ptr<UploadElementReader>> element_readers_; | 
|  | }; | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, EmptyUploadData) { | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  | ASSERT_THAT(stream->Init(CompletionOnceCallback(), NetLogWithSource()), | 
|  | IsOk()); | 
|  | EXPECT_TRUE(stream->IsInMemory()); | 
|  | EXPECT_EQ(0U, stream->size()); | 
|  | EXPECT_EQ(0U, stream->position()); | 
|  | EXPECT_TRUE(stream->IsEOF()); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, ConsumeAllBytes) { | 
|  | element_readers_.push_back( | 
|  | std::make_unique<UploadBytesElementReader>(kTestData, kTestDataSize)); | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  | ASSERT_THAT(stream->Init(CompletionOnceCallback(), NetLogWithSource()), | 
|  | IsOk()); | 
|  | EXPECT_TRUE(stream->IsInMemory()); | 
|  | EXPECT_EQ(kTestDataSize, stream->size()); | 
|  | EXPECT_EQ(0U, stream->position()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kTestBufferSize); | 
|  | while (!stream->IsEOF()) { | 
|  | int bytes_read = | 
|  | stream->Read(buf.get(), kTestBufferSize, CompletionOnceCallback()); | 
|  | ASSERT_LE(0, bytes_read);  // Not an error. | 
|  | } | 
|  | EXPECT_EQ(kTestDataSize, stream->position()); | 
|  | ASSERT_TRUE(stream->IsEOF()); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, File) { | 
|  | base::FilePath temp_file_path; | 
|  | ASSERT_TRUE( | 
|  | base::CreateTemporaryFileInDir(temp_dir_.GetPath(), &temp_file_path)); | 
|  | ASSERT_EQ(static_cast<int>(kTestDataSize), | 
|  | base::WriteFile(temp_file_path, kTestData, kTestDataSize)); | 
|  |  | 
|  | element_readers_.push_back(std::make_unique<UploadFileElementReader>( | 
|  | base::ThreadTaskRunnerHandle::Get().get(), temp_file_path, 0, | 
|  | std::numeric_limits<uint64_t>::max(), base::Time())); | 
|  |  | 
|  | TestCompletionCallback init_callback; | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  | ASSERT_THAT(stream->Init(init_callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | ASSERT_THAT(init_callback.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsInMemory()); | 
|  | EXPECT_EQ(kTestDataSize, stream->size()); | 
|  | EXPECT_EQ(0U, stream->position()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kTestBufferSize); | 
|  | while (!stream->IsEOF()) { | 
|  | TestCompletionCallback read_callback; | 
|  | ASSERT_EQ( | 
|  | ERR_IO_PENDING, | 
|  | stream->Read(buf.get(), kTestBufferSize, read_callback.callback())); | 
|  | ASSERT_LE(0, read_callback.WaitForResult());  // Not an error. | 
|  | } | 
|  | EXPECT_EQ(kTestDataSize, stream->position()); | 
|  | ASSERT_TRUE(stream->IsEOF()); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, FileSmallerThanLength) { | 
|  | base::FilePath temp_file_path; | 
|  | ASSERT_TRUE( | 
|  | base::CreateTemporaryFileInDir(temp_dir_.GetPath(), &temp_file_path)); | 
|  | ASSERT_EQ(static_cast<int>(kTestDataSize), | 
|  | base::WriteFile(temp_file_path, kTestData, kTestDataSize)); | 
|  | const uint64_t kFakeSize = kTestDataSize * 2; | 
|  |  | 
|  | UploadFileElementReader::ScopedOverridingContentLengthForTests | 
|  | overriding_content_length(kFakeSize); | 
|  |  | 
|  | element_readers_.push_back(std::make_unique<UploadFileElementReader>( | 
|  | base::ThreadTaskRunnerHandle::Get().get(), temp_file_path, 0, | 
|  | std::numeric_limits<uint64_t>::max(), base::Time())); | 
|  |  | 
|  | TestCompletionCallback init_callback; | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  | ASSERT_THAT(stream->Init(init_callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | ASSERT_THAT(init_callback.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsInMemory()); | 
|  | EXPECT_EQ(kFakeSize, stream->size()); | 
|  | EXPECT_EQ(0U, stream->position()); | 
|  |  | 
|  | scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kTestBufferSize); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  |  | 
|  | TestCompletionCallback read_callback; | 
|  | ASSERT_EQ(ERR_IO_PENDING, | 
|  | stream->Read(buf.get(), kTestBufferSize, read_callback.callback())); | 
|  | int bytes_read = read_callback.WaitForResult(); | 
|  |  | 
|  | EXPECT_EQ(10, bytes_read); | 
|  | EXPECT_EQ(10U, stream->position()); | 
|  |  | 
|  | // UpdateDataStream will return error if there is something wrong. | 
|  | EXPECT_EQ(ERR_UPLOAD_FILE_CHANGED, | 
|  | stream->Read(buf.get(), kTestBufferSize, read_callback.callback())); | 
|  | EXPECT_EQ(10U, stream->position()); | 
|  |  | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, ReadErrorSync) { | 
|  | // This element cannot be read. | 
|  | std::unique_ptr<MockUploadElementReader> reader( | 
|  | new MockUploadElementReader(kTestDataSize, true)); | 
|  | EXPECT_CALL(*reader, Init(_)).WillOnce(Return(OK)); | 
|  | reader->SetReadExpectation(ERR_FAILED); | 
|  | element_readers_.push_back(std::move(reader)); | 
|  |  | 
|  | // This element is ignored because of the error from the previous reader. | 
|  | element_readers_.push_back( | 
|  | std::make_unique<UploadBytesElementReader>(kTestData, kTestDataSize)); | 
|  |  | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | // Run Init(). | 
|  | ASSERT_THAT(stream->Init(CompletionOnceCallback(), NetLogWithSource()), | 
|  | IsOk()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  | EXPECT_EQ(0U, stream->position()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  |  | 
|  | // Prepare a buffer filled with non-zero data. | 
|  | scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kTestBufferSize); | 
|  | std::fill_n(buf->data(), kTestBufferSize, -1); | 
|  |  | 
|  | // Read() results in success even when the reader returns error. | 
|  | EXPECT_EQ(ERR_FAILED, | 
|  | stream->Read(buf.get(), kTestBufferSize, CompletionOnceCallback())); | 
|  | EXPECT_EQ(0U, stream->position()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  |  | 
|  | // The buffer is filled with zero. | 
|  | EXPECT_EQ(0, std::count(buf->data(), buf->data() + kTestBufferSize, 0)); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, ReadErrorAsync) { | 
|  | // This element cannot be read. | 
|  | std::unique_ptr<MockUploadElementReader> reader( | 
|  | new MockUploadElementReader(kTestDataSize, false)); | 
|  | reader->SetAsyncInitExpectation(OK); | 
|  | reader->SetReadExpectation(ERR_FAILED); | 
|  | element_readers_.push_back(std::move(reader)); | 
|  |  | 
|  | // This element is ignored because of the error from the previous reader. | 
|  | element_readers_.push_back( | 
|  | std::make_unique<UploadBytesElementReader>(kTestData, kTestDataSize)); | 
|  |  | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | // Run Init(). | 
|  | TestCompletionCallback init_callback; | 
|  | ASSERT_THAT(stream->Init(init_callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(init_callback.WaitForResult(), IsOk()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  | EXPECT_EQ(0U, stream->position()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  |  | 
|  | // Prepare a buffer filled with non-zero data. | 
|  | scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kTestBufferSize); | 
|  | std::fill_n(buf->data(), kTestBufferSize, -1); | 
|  |  | 
|  | // Read() results in success even when the reader returns error. | 
|  | TestCompletionCallback read_callback; | 
|  | ASSERT_EQ(ERR_IO_PENDING, | 
|  | stream->Read(buf.get(), kTestBufferSize, read_callback.callback())); | 
|  | EXPECT_THAT(read_callback.WaitForResult(), IsError(ERR_FAILED)); | 
|  | EXPECT_EQ(0U, stream->position()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  |  | 
|  | // The buffer is empty | 
|  | EXPECT_EQ(0, std::count(buf->data(), buf->data() + kTestBufferSize, 0)); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, FileAndBytes) { | 
|  | base::FilePath temp_file_path; | 
|  | ASSERT_TRUE( | 
|  | base::CreateTemporaryFileInDir(temp_dir_.GetPath(), &temp_file_path)); | 
|  | ASSERT_EQ(static_cast<int>(kTestDataSize), | 
|  | base::WriteFile(temp_file_path, kTestData, kTestDataSize)); | 
|  |  | 
|  | const uint64_t kFileRangeOffset = 1; | 
|  | const uint64_t kFileRangeLength = 4; | 
|  | element_readers_.push_back(std::make_unique<UploadFileElementReader>( | 
|  | base::ThreadTaskRunnerHandle::Get().get(), temp_file_path, | 
|  | kFileRangeOffset, kFileRangeLength, base::Time())); | 
|  |  | 
|  | element_readers_.push_back( | 
|  | std::make_unique<UploadBytesElementReader>(kTestData, kTestDataSize)); | 
|  |  | 
|  | const uint64_t kStreamSize = kTestDataSize + kFileRangeLength; | 
|  | TestCompletionCallback init_callback; | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  | ASSERT_THAT(stream->Init(init_callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | ASSERT_THAT(init_callback.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsInMemory()); | 
|  | EXPECT_EQ(kStreamSize, stream->size()); | 
|  | EXPECT_EQ(0U, stream->position()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kTestBufferSize); | 
|  | while (!stream->IsEOF()) { | 
|  | TestCompletionCallback read_callback; | 
|  | const int result = | 
|  | stream->Read(buf.get(), kTestBufferSize, read_callback.callback()); | 
|  | const int bytes_read = | 
|  | result != ERR_IO_PENDING ? result : read_callback.WaitForResult(); | 
|  | ASSERT_LE(0, bytes_read);  // Not an error. | 
|  | } | 
|  | EXPECT_EQ(kStreamSize, stream->position()); | 
|  | ASSERT_TRUE(stream->IsEOF()); | 
|  | } | 
|  |  | 
|  | // Init() with on-memory and not-on-memory readers. | 
|  | TEST_F(ElementsUploadDataStreamTest, InitAsync) { | 
|  | // Create UploadDataStream with mock readers. | 
|  | std::unique_ptr<MockUploadElementReader> reader( | 
|  | new MockUploadElementReader(kTestDataSize, true)); | 
|  | EXPECT_CALL(*reader, Init(_)).WillOnce(Return(OK)); | 
|  | element_readers_.push_back(std::move(reader)); | 
|  |  | 
|  | std::unique_ptr<MockUploadElementReader> reader2( | 
|  | new MockUploadElementReader(kTestDataSize, true)); | 
|  | EXPECT_CALL(*reader2, Init(_)).WillOnce(Return(OK)); | 
|  | element_readers_.push_back(std::move(reader2)); | 
|  |  | 
|  | std::unique_ptr<MockUploadElementReader> reader3( | 
|  | new MockUploadElementReader(kTestDataSize, false)); | 
|  | reader3->SetAsyncInitExpectation(OK); | 
|  | element_readers_.push_back(std::move(reader3)); | 
|  |  | 
|  | std::unique_ptr<MockUploadElementReader> reader4( | 
|  | new MockUploadElementReader(kTestDataSize, false)); | 
|  | reader4->SetAsyncInitExpectation(OK); | 
|  | element_readers_.push_back(std::move(reader4)); | 
|  |  | 
|  | std::unique_ptr<MockUploadElementReader> reader5( | 
|  | new MockUploadElementReader(kTestDataSize, true)); | 
|  | EXPECT_CALL(*reader5, Init(_)).WillOnce(Return(OK)); | 
|  | element_readers_.push_back(std::move(reader5)); | 
|  |  | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | // Run Init(). | 
|  | TestCompletionCallback callback; | 
|  | ASSERT_THAT(stream->Init(callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(callback.WaitForResult(), IsOk()); | 
|  | } | 
|  |  | 
|  | // Init() of a reader fails asynchronously. | 
|  | TEST_F(ElementsUploadDataStreamTest, InitAsyncFailureAsync) { | 
|  | // Create UploadDataStream with a mock reader. | 
|  | std::unique_ptr<MockUploadElementReader> reader( | 
|  | new MockUploadElementReader(kTestDataSize, false)); | 
|  | reader->SetAsyncInitExpectation(ERR_FAILED); | 
|  | element_readers_.push_back(std::move(reader)); | 
|  |  | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | // Run Init(). | 
|  | TestCompletionCallback callback; | 
|  | ASSERT_THAT(stream->Init(callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(callback.WaitForResult(), IsError(ERR_FAILED)); | 
|  | } | 
|  |  | 
|  | // Init() of a reader fails synchronously. | 
|  | TEST_F(ElementsUploadDataStreamTest, InitAsyncFailureSync) { | 
|  | // Create UploadDataStream with mock readers. | 
|  | std::unique_ptr<MockUploadElementReader> reader( | 
|  | new MockUploadElementReader(kTestDataSize, false)); | 
|  | reader->SetAsyncInitExpectation(OK); | 
|  | element_readers_.push_back(std::move(reader)); | 
|  |  | 
|  | std::unique_ptr<MockUploadElementReader> reader2( | 
|  | new MockUploadElementReader(kTestDataSize, true)); | 
|  | EXPECT_CALL(*reader2, Init(_)).WillOnce(Return(ERR_FAILED)); | 
|  | element_readers_.push_back(std::move(reader2)); | 
|  |  | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | // Run Init(). | 
|  | TestCompletionCallback callback; | 
|  | ASSERT_THAT(stream->Init(callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(callback.WaitForResult(), IsError(ERR_FAILED)); | 
|  | } | 
|  |  | 
|  | // Read with a buffer whose size is same as the data. | 
|  | TEST_F(ElementsUploadDataStreamTest, ReadAsyncWithExactSizeBuffer) { | 
|  | element_readers_.push_back( | 
|  | std::make_unique<UploadBytesElementReader>(kTestData, kTestDataSize)); | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | ASSERT_THAT(stream->Init(CompletionOnceCallback(), NetLogWithSource()), | 
|  | IsOk()); | 
|  | EXPECT_TRUE(stream->IsInMemory()); | 
|  | EXPECT_EQ(kTestDataSize, stream->size()); | 
|  | EXPECT_EQ(0U, stream->position()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kTestDataSize); | 
|  | int bytes_read = | 
|  | stream->Read(buf.get(), kTestDataSize, CompletionOnceCallback()); | 
|  | ASSERT_EQ(static_cast<int>(kTestDataSize), bytes_read);  // Not an error. | 
|  | EXPECT_EQ(kTestDataSize, stream->position()); | 
|  | ASSERT_TRUE(stream->IsEOF()); | 
|  | } | 
|  |  | 
|  | // Async Read() with on-memory and not-on-memory readers. | 
|  | TEST_F(ElementsUploadDataStreamTest, ReadAsync) { | 
|  | // Create UploadDataStream with mock readers. | 
|  | std::unique_ptr<MockUploadElementReader> reader( | 
|  | new MockUploadElementReader(kTestDataSize, true)); | 
|  | EXPECT_CALL(*reader, Init(_)).WillOnce(Return(OK)); | 
|  | reader->SetReadExpectation(kTestDataSize); | 
|  | element_readers_.push_back(std::move(reader)); | 
|  |  | 
|  | std::unique_ptr<MockUploadElementReader> reader2( | 
|  | new MockUploadElementReader(kTestDataSize, false)); | 
|  | reader2->SetAsyncInitExpectation(OK); | 
|  | reader2->SetReadExpectation(kTestDataSize); | 
|  | element_readers_.push_back(std::move(reader2)); | 
|  |  | 
|  | std::unique_ptr<MockUploadElementReader> reader3( | 
|  | new MockUploadElementReader(kTestDataSize, true)); | 
|  | EXPECT_CALL(*reader3, Init(_)).WillOnce(Return(OK)); | 
|  | reader3->SetReadExpectation(kTestDataSize); | 
|  | element_readers_.push_back(std::move(reader3)); | 
|  |  | 
|  | std::unique_ptr<MockUploadElementReader> reader4( | 
|  | new MockUploadElementReader(kTestDataSize, false)); | 
|  | reader4->SetAsyncInitExpectation(OK); | 
|  | reader4->SetReadExpectation(kTestDataSize); | 
|  | element_readers_.push_back(std::move(reader4)); | 
|  |  | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | // Run Init(). | 
|  | TestCompletionCallback init_callback; | 
|  | EXPECT_THAT(stream->Init(init_callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(init_callback.WaitForResult(), IsOk()); | 
|  |  | 
|  | scoped_refptr<IOBuffer> buf = base::MakeRefCounted<IOBuffer>(kTestBufferSize); | 
|  |  | 
|  | // Consume the first element. | 
|  | TestCompletionCallback read_callback1; | 
|  | EXPECT_EQ(static_cast<int>(kTestDataSize), | 
|  | stream->Read(buf.get(), kTestDataSize, read_callback1.callback())); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  | EXPECT_FALSE(read_callback1.have_result()); | 
|  |  | 
|  | // Consume the second element. | 
|  | TestCompletionCallback read_callback2; | 
|  | ASSERT_EQ(ERR_IO_PENDING, | 
|  | stream->Read(buf.get(), kTestDataSize, read_callback2.callback())); | 
|  | EXPECT_EQ(static_cast<int>(kTestDataSize), read_callback2.WaitForResult()); | 
|  |  | 
|  | // Consume the third and the fourth elements. | 
|  | TestCompletionCallback read_callback3; | 
|  | ASSERT_EQ(ERR_IO_PENDING, stream->Read(buf.get(), kTestDataSize * 2, | 
|  | read_callback3.callback())); | 
|  | EXPECT_EQ(static_cast<int>(kTestDataSize * 2), | 
|  | read_callback3.WaitForResult()); | 
|  | } | 
|  |  | 
|  | void ElementsUploadDataStreamTest::FileChangedHelper( | 
|  | const base::FilePath& file_path, | 
|  | const base::Time& time, | 
|  | bool error_expected) { | 
|  | // Don't use element_readers_ here, as this function is called twice, and | 
|  | // reusing element_readers_ is wrong. | 
|  | std::vector<std::unique_ptr<UploadElementReader>> element_readers; | 
|  | element_readers.push_back(std::make_unique<UploadFileElementReader>( | 
|  | base::ThreadTaskRunnerHandle::Get().get(), file_path, 1, 2, time)); | 
|  |  | 
|  | TestCompletionCallback init_callback; | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers), 0)); | 
|  | ASSERT_THAT(stream->Init(init_callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | int error_code = init_callback.WaitForResult(); | 
|  | if (error_expected) | 
|  | ASSERT_THAT(error_code, IsError(ERR_UPLOAD_FILE_CHANGED)); | 
|  | else | 
|  | ASSERT_THAT(error_code, IsOk()); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, FileChanged) { | 
|  | base::FilePath temp_file_path; | 
|  | ASSERT_TRUE( | 
|  | base::CreateTemporaryFileInDir(temp_dir_.GetPath(), &temp_file_path)); | 
|  | ASSERT_EQ(static_cast<int>(kTestDataSize), | 
|  | base::WriteFile(temp_file_path, kTestData, kTestDataSize)); | 
|  |  | 
|  | base::File::Info file_info; | 
|  | ASSERT_TRUE(base::GetFileInfo(temp_file_path, &file_info)); | 
|  |  | 
|  | // Test file not changed. | 
|  | FileChangedHelper(temp_file_path, file_info.last_modified, false); | 
|  |  | 
|  | // Test file changed. | 
|  | FileChangedHelper(temp_file_path, | 
|  | file_info.last_modified - base::TimeDelta::FromSeconds(1), | 
|  | true); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, MultipleInit) { | 
|  | base::FilePath temp_file_path; | 
|  | ASSERT_TRUE( | 
|  | base::CreateTemporaryFileInDir(temp_dir_.GetPath(), &temp_file_path)); | 
|  | ASSERT_EQ(static_cast<int>(kTestDataSize), | 
|  | base::WriteFile(temp_file_path, kTestData, kTestDataSize)); | 
|  |  | 
|  | // Prepare data. | 
|  | element_readers_.push_back( | 
|  | std::make_unique<UploadBytesElementReader>(kTestData, kTestDataSize)); | 
|  | element_readers_.push_back(std::make_unique<UploadFileElementReader>( | 
|  | base::ThreadTaskRunnerHandle::Get().get(), temp_file_path, 0, | 
|  | std::numeric_limits<uint64_t>::max(), base::Time())); | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | std::string expected_data(kTestData, kTestData + kTestDataSize); | 
|  | expected_data += expected_data; | 
|  |  | 
|  | // Call Init(). | 
|  | TestCompletionCallback init_callback1; | 
|  | ASSERT_THAT(stream->Init(init_callback1.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | ASSERT_THAT(init_callback1.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  |  | 
|  | // Read. | 
|  | EXPECT_EQ(expected_data, ReadFromUploadDataStream(stream.get())); | 
|  | EXPECT_TRUE(stream->IsEOF()); | 
|  |  | 
|  | // Call Init() again to reset. | 
|  | TestCompletionCallback init_callback2; | 
|  | ASSERT_THAT(stream->Init(init_callback2.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | ASSERT_THAT(init_callback2.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  |  | 
|  | // Read again. | 
|  | EXPECT_EQ(expected_data, ReadFromUploadDataStream(stream.get())); | 
|  | EXPECT_TRUE(stream->IsEOF()); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, MultipleInitAsync) { | 
|  | base::FilePath temp_file_path; | 
|  | ASSERT_TRUE( | 
|  | base::CreateTemporaryFileInDir(temp_dir_.GetPath(), &temp_file_path)); | 
|  | ASSERT_EQ(static_cast<int>(kTestDataSize), | 
|  | base::WriteFile(temp_file_path, kTestData, kTestDataSize)); | 
|  | TestCompletionCallback test_callback; | 
|  |  | 
|  | // Prepare data. | 
|  | element_readers_.push_back( | 
|  | std::make_unique<UploadBytesElementReader>(kTestData, kTestDataSize)); | 
|  | element_readers_.push_back(std::make_unique<UploadFileElementReader>( | 
|  | base::ThreadTaskRunnerHandle::Get().get(), temp_file_path, 0, | 
|  | std::numeric_limits<uint64_t>::max(), base::Time())); | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | std::string expected_data(kTestData, kTestData + kTestDataSize); | 
|  | expected_data += expected_data; | 
|  |  | 
|  | // Call Init(). | 
|  | ASSERT_THAT(stream->Init(test_callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(test_callback.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  |  | 
|  | // Read. | 
|  | EXPECT_EQ(expected_data, ReadFromUploadDataStream(stream.get())); | 
|  | EXPECT_TRUE(stream->IsEOF()); | 
|  |  | 
|  | // Call Init() again to reset. | 
|  | ASSERT_THAT(stream->Init(test_callback.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(test_callback.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  |  | 
|  | // Read again. | 
|  | EXPECT_EQ(expected_data, ReadFromUploadDataStream(stream.get())); | 
|  | EXPECT_TRUE(stream->IsEOF()); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, InitToReset) { | 
|  | base::FilePath temp_file_path; | 
|  | ASSERT_TRUE( | 
|  | base::CreateTemporaryFileInDir(temp_dir_.GetPath(), &temp_file_path)); | 
|  | ASSERT_EQ(static_cast<int>(kTestDataSize), | 
|  | base::WriteFile(temp_file_path, kTestData, kTestDataSize)); | 
|  |  | 
|  | // Prepare data. | 
|  | element_readers_.push_back( | 
|  | std::make_unique<UploadBytesElementReader>(kTestData, kTestDataSize)); | 
|  | element_readers_.push_back(std::make_unique<UploadFileElementReader>( | 
|  | base::ThreadTaskRunnerHandle::Get().get(), temp_file_path, 0, | 
|  | std::numeric_limits<uint64_t>::max(), base::Time())); | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | std::vector<char> expected_data(kTestData, kTestData + kTestDataSize); | 
|  | expected_data.insert(expected_data.end(), kTestData, | 
|  | kTestData + kTestDataSize); | 
|  |  | 
|  | // Call Init(). | 
|  | TestCompletionCallback init_callback1; | 
|  | ASSERT_THAT(stream->Init(init_callback1.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(init_callback1.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  |  | 
|  | // Read some. | 
|  | TestCompletionCallback read_callback1; | 
|  | std::vector<char> buf(kTestDataSize + kTestDataSize / 2); | 
|  | scoped_refptr<IOBuffer> wrapped_buffer = | 
|  | base::MakeRefCounted<WrappedIOBuffer>(&buf[0]); | 
|  | EXPECT_EQ( | 
|  | ERR_IO_PENDING, | 
|  | stream->Read(wrapped_buffer.get(), buf.size(), | 
|  | read_callback1.callback())); | 
|  | EXPECT_EQ(static_cast<int>(buf.size()), read_callback1.WaitForResult()); | 
|  | EXPECT_EQ(buf.size(), stream->position()); | 
|  |  | 
|  | // Call Init to reset the state. | 
|  | TestCompletionCallback init_callback2; | 
|  | ASSERT_THAT(stream->Init(init_callback2.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(init_callback2.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  |  | 
|  | // Read. | 
|  | TestCompletionCallback read_callback2; | 
|  | std::vector<char> buf2(kTestDataSize * 2); | 
|  | scoped_refptr<IOBuffer> wrapped_buffer2 = | 
|  | base::MakeRefCounted<WrappedIOBuffer>(&buf2[0]); | 
|  | EXPECT_EQ(ERR_IO_PENDING, | 
|  | stream->Read( | 
|  | wrapped_buffer2.get(), buf2.size(), read_callback2.callback())); | 
|  | EXPECT_EQ(static_cast<int>(buf2.size()), read_callback2.WaitForResult()); | 
|  | EXPECT_EQ(expected_data, buf2); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, InitDuringAsyncInit) { | 
|  | base::FilePath temp_file_path; | 
|  | ASSERT_TRUE( | 
|  | base::CreateTemporaryFileInDir(temp_dir_.GetPath(), &temp_file_path)); | 
|  | ASSERT_EQ(static_cast<int>(kTestDataSize), | 
|  | base::WriteFile(temp_file_path, kTestData, kTestDataSize)); | 
|  |  | 
|  | // Prepare data. | 
|  | element_readers_.push_back( | 
|  | std::make_unique<UploadBytesElementReader>(kTestData, kTestDataSize)); | 
|  | element_readers_.push_back(std::make_unique<UploadFileElementReader>( | 
|  | base::ThreadTaskRunnerHandle::Get().get(), temp_file_path, 0, | 
|  | std::numeric_limits<uint64_t>::max(), base::Time())); | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | std::vector<char> expected_data(kTestData, kTestData + kTestDataSize); | 
|  | expected_data.insert(expected_data.end(), kTestData, | 
|  | kTestData + kTestDataSize); | 
|  |  | 
|  | // Start Init. | 
|  | TestCompletionCallback init_callback1; | 
|  | EXPECT_THAT(stream->Init(init_callback1.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  |  | 
|  | // Call Init again to cancel the previous init. | 
|  | TestCompletionCallback init_callback2; | 
|  | EXPECT_THAT(stream->Init(init_callback2.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(init_callback2.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  |  | 
|  | // Read. | 
|  | TestCompletionCallback read_callback2; | 
|  | std::vector<char> buf2(kTestDataSize * 2); | 
|  | scoped_refptr<IOBuffer> wrapped_buffer2 = | 
|  | base::MakeRefCounted<WrappedIOBuffer>(&buf2[0]); | 
|  | EXPECT_EQ(ERR_IO_PENDING, | 
|  | stream->Read( | 
|  | wrapped_buffer2.get(), buf2.size(), read_callback2.callback())); | 
|  | EXPECT_EQ(static_cast<int>(buf2.size()), read_callback2.WaitForResult()); | 
|  | EXPECT_EQ(expected_data, buf2); | 
|  | EXPECT_TRUE(stream->IsEOF()); | 
|  |  | 
|  | // Make sure callbacks are not called for cancelled operations. | 
|  | EXPECT_FALSE(init_callback1.have_result()); | 
|  | } | 
|  |  | 
|  | TEST_F(ElementsUploadDataStreamTest, InitDuringAsyncRead) { | 
|  | base::FilePath temp_file_path; | 
|  | ASSERT_TRUE( | 
|  | base::CreateTemporaryFileInDir(temp_dir_.GetPath(), &temp_file_path)); | 
|  | ASSERT_EQ(static_cast<int>(kTestDataSize), | 
|  | base::WriteFile(temp_file_path, kTestData, kTestDataSize)); | 
|  |  | 
|  | // Prepare data. | 
|  | element_readers_.push_back( | 
|  | std::make_unique<UploadBytesElementReader>(kTestData, kTestDataSize)); | 
|  | element_readers_.push_back(std::make_unique<UploadFileElementReader>( | 
|  | base::ThreadTaskRunnerHandle::Get().get(), temp_file_path, 0, | 
|  | std::numeric_limits<uint64_t>::max(), base::Time())); | 
|  | std::unique_ptr<UploadDataStream> stream( | 
|  | new ElementsUploadDataStream(std::move(element_readers_), 0)); | 
|  |  | 
|  | std::vector<char> expected_data(kTestData, kTestData + kTestDataSize); | 
|  | expected_data.insert(expected_data.end(), kTestData, | 
|  | kTestData + kTestDataSize); | 
|  |  | 
|  | // Call Init(). | 
|  | TestCompletionCallback init_callback1; | 
|  | ASSERT_THAT(stream->Init(init_callback1.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(init_callback1.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  |  | 
|  | // Start reading. | 
|  | TestCompletionCallback read_callback1; | 
|  | std::vector<char> buf(kTestDataSize * 2); | 
|  | scoped_refptr<IOBuffer> wrapped_buffer = | 
|  | base::MakeRefCounted<WrappedIOBuffer>(&buf[0]); | 
|  | EXPECT_EQ( | 
|  | ERR_IO_PENDING, | 
|  | stream->Read(wrapped_buffer.get(), buf.size(), | 
|  | read_callback1.callback())); | 
|  |  | 
|  | // Call Init to cancel the previous read. | 
|  | TestCompletionCallback init_callback2; | 
|  | EXPECT_THAT(stream->Init(init_callback2.callback(), NetLogWithSource()), | 
|  | IsError(ERR_IO_PENDING)); | 
|  | EXPECT_THAT(init_callback2.WaitForResult(), IsOk()); | 
|  | EXPECT_FALSE(stream->IsEOF()); | 
|  | EXPECT_EQ(kTestDataSize * 2, stream->size()); | 
|  |  | 
|  | // Read. | 
|  | TestCompletionCallback read_callback2; | 
|  | std::vector<char> buf2(kTestDataSize * 2); | 
|  | scoped_refptr<IOBuffer> wrapped_buffer2 = | 
|  | base::MakeRefCounted<WrappedIOBuffer>(&buf2[0]); | 
|  | EXPECT_EQ(ERR_IO_PENDING, | 
|  | stream->Read( | 
|  | wrapped_buffer2.get(), buf2.size(), read_callback2.callback())); | 
|  | EXPECT_EQ(static_cast<int>(buf2.size()), read_callback2.WaitForResult()); | 
|  | EXPECT_EQ(expected_data, buf2); | 
|  | EXPECT_TRUE(stream->IsEOF()); | 
|  |  | 
|  | // Make sure callbacks are not called for cancelled operations. | 
|  | EXPECT_FALSE(read_callback1.have_result()); | 
|  | } | 
|  |  | 
|  | }  // namespace net |