// Copyright 2015 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 "modules/fetch/DataConsumerTee.h" #include "core/testing/DummyPageHolder.h" #include "core/testing/NullExecutionContext.h" #include "modules/fetch/DataConsumerHandleTestUtil.h" #include "platform/ThreadSafeFunctional.h" #include "platform/WaitableEvent.h" #include "platform/WebThreadSupportingGC.h" #include "public/platform/Platform.h" #include "public/platform/WebThread.h" #include "public/platform/WebTraceLocation.h" #include "testing/gtest/include/gtest/gtest.h" #include "wtf/PassRefPtr.h" #include "wtf/RefPtr.h" #include #include namespace blink { namespace { using ::testing::InSequence; using ::testing::Return; using ::testing::StrictMock; using ::testing::_; using Checkpoint = StrictMock<::testing::MockFunction>; using Result = WebDataConsumerHandle::Result; using Thread = DataConsumerHandleTestUtil::Thread; const Result kDone = WebDataConsumerHandle::Done; const Result kUnexpectedError = WebDataConsumerHandle::UnexpectedError; const FetchDataConsumerHandle::Reader::BlobSizePolicy kDisallowBlobWithInvalidSize = FetchDataConsumerHandle::Reader::DisallowBlobWithInvalidSize; const FetchDataConsumerHandle::Reader::BlobSizePolicy kAllowBlobWithInvalidSize = FetchDataConsumerHandle::Reader::AllowBlobWithInvalidSize; using Command = DataConsumerHandleTestUtil::Command; using Handle = DataConsumerHandleTestUtil::ReplayingHandle; using HandleReader = DataConsumerHandleTestUtil::HandleReader; using HandleTwoPhaseReader = DataConsumerHandleTestUtil::HandleTwoPhaseReader; using HandleReadResult = DataConsumerHandleTestUtil::HandleReadResult; using MockFetchDataConsumerHandle = DataConsumerHandleTestUtil::MockFetchDataConsumerHandle; using MockFetchDataConsumerReader = DataConsumerHandleTestUtil::MockFetchDataConsumerReader; template using HandleReaderRunner = DataConsumerHandleTestUtil::HandleReaderRunner; String toString(const Vector& v) { return String(v.data(), v.size()); } template class TeeCreationThread { public: void run(PassOwnPtr src, OwnPtr* dest1, OwnPtr* dest2) { m_thread = adoptPtr(new Thread("src thread", Thread::WithExecutionContext)); m_waitableEvent = adoptPtr(new WaitableEvent()); m_thread->thread()->postTask(BLINK_FROM_HERE, threadSafeBind(&TeeCreationThread::runInternal, AllowCrossThreadAccess(this), src, AllowCrossThreadAccess(dest1), AllowCrossThreadAccess(dest2))); m_waitableEvent->wait(); } Thread* getThread() { return m_thread.get(); } private: void runInternal(PassOwnPtr src, OwnPtr* dest1, OwnPtr* dest2) { DataConsumerTee::create(m_thread->getExecutionContext(), src, dest1, dest2); m_waitableEvent->signal(); } OwnPtr m_thread; OwnPtr m_waitableEvent; }; TEST(DataConsumerTeeTest, CreateDone) { OwnPtr src(Handle::create()); OwnPtr dest1, dest2; src->add(Command(Command::Done)); OwnPtr> t = adoptPtr(new TeeCreationThread()); t->run(src.release(), &dest1, &dest2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); HandleReaderRunner r1(dest1.release()), r2(dest2.release()); OwnPtr res1 = r1.wait(); OwnPtr res2 = r2.wait(); EXPECT_EQ(kDone, res1->result()); EXPECT_EQ(0u, res1->data().size()); EXPECT_EQ(kDone, res2->result()); EXPECT_EQ(0u, res2->data().size()); } TEST(DataConsumerTeeTest, Read) { OwnPtr src(Handle::create()); OwnPtr dest1, dest2; src->add(Command(Command::Wait)); src->add(Command(Command::Data, "hello, ")); src->add(Command(Command::Wait)); src->add(Command(Command::Data, "world")); src->add(Command(Command::Wait)); src->add(Command(Command::Wait)); src->add(Command(Command::Done)); OwnPtr> t = adoptPtr(new TeeCreationThread()); t->run(src.release(), &dest1, &dest2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); HandleReaderRunner r1(dest1.release()); HandleReaderRunner r2(dest2.release()); OwnPtr res1 = r1.wait(); OwnPtr res2 = r2.wait(); EXPECT_EQ(kDone, res1->result()); EXPECT_EQ("hello, world", toString(res1->data())); EXPECT_EQ(kDone, res2->result()); EXPECT_EQ("hello, world", toString(res2->data())); } TEST(DataConsumerTeeTest, TwoPhaseRead) { OwnPtr src(Handle::create()); OwnPtr dest1, dest2; src->add(Command(Command::Wait)); src->add(Command(Command::Data, "hello, ")); src->add(Command(Command::Wait)); src->add(Command(Command::Wait)); src->add(Command(Command::Wait)); src->add(Command(Command::Data, "world")); src->add(Command(Command::Wait)); src->add(Command(Command::Done)); OwnPtr> t = adoptPtr(new TeeCreationThread()); t->run(src.release(), &dest1, &dest2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); HandleReaderRunner r1(dest1.release()); HandleReaderRunner r2(dest2.release()); OwnPtr res1 = r1.wait(); OwnPtr res2 = r2.wait(); EXPECT_EQ(kDone, res1->result()); EXPECT_EQ("hello, world", toString(res1->data())); EXPECT_EQ(kDone, res2->result()); EXPECT_EQ("hello, world", toString(res2->data())); } TEST(DataConsumerTeeTest, Error) { OwnPtr src(Handle::create()); OwnPtr dest1, dest2; src->add(Command(Command::Data, "hello, ")); src->add(Command(Command::Data, "world")); src->add(Command(Command::Error)); OwnPtr> t = adoptPtr(new TeeCreationThread()); t->run(src.release(), &dest1, &dest2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); HandleReaderRunner r1(dest1.release()); HandleReaderRunner r2(dest2.release()); OwnPtr res1 = r1.wait(); OwnPtr res2 = r2.wait(); EXPECT_EQ(kUnexpectedError, res1->result()); EXPECT_EQ(kUnexpectedError, res2->result()); } void postStop(Thread* thread) { thread->getExecutionContext()->stopActiveDOMObjects(); } TEST(DataConsumerTeeTest, StopSource) { OwnPtr src(Handle::create()); OwnPtr dest1, dest2; src->add(Command(Command::Data, "hello, ")); src->add(Command(Command::Data, "world")); OwnPtr> t = adoptPtr(new TeeCreationThread()); t->run(src.release(), &dest1, &dest2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); HandleReaderRunner r1(dest1.release()); HandleReaderRunner r2(dest2.release()); // We can pass a raw pointer because the subsequent |wait| calls ensure // t->thread() is alive. t->getThread()->thread()->postTask(BLINK_FROM_HERE, threadSafeBind(postStop, AllowCrossThreadAccess(t->getThread()))); OwnPtr res1 = r1.wait(); OwnPtr res2 = r2.wait(); EXPECT_EQ(kUnexpectedError, res1->result()); EXPECT_EQ(kUnexpectedError, res2->result()); } TEST(DataConsumerTeeTest, DetachSource) { OwnPtr src(Handle::create()); OwnPtr dest1, dest2; src->add(Command(Command::Data, "hello, ")); src->add(Command(Command::Data, "world")); OwnPtr> t = adoptPtr(new TeeCreationThread()); t->run(src.release(), &dest1, &dest2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); HandleReaderRunner r1(dest1.release()); HandleReaderRunner r2(dest2.release()); t = nullptr; OwnPtr res1 = r1.wait(); OwnPtr res2 = r2.wait(); EXPECT_EQ(kUnexpectedError, res1->result()); EXPECT_EQ(kUnexpectedError, res2->result()); } TEST(DataConsumerTeeTest, DetachSourceAfterReadingDone) { OwnPtr src(Handle::create()); OwnPtr dest1, dest2; src->add(Command(Command::Data, "hello, ")); src->add(Command(Command::Data, "world")); src->add(Command(Command::Done)); OwnPtr> t = adoptPtr(new TeeCreationThread()); t->run(src.release(), &dest1, &dest2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); HandleReaderRunner r1(dest1.release()); OwnPtr res1 = r1.wait(); EXPECT_EQ(kDone, res1->result()); EXPECT_EQ("hello, world", toString(res1->data())); t = nullptr; HandleReaderRunner r2(dest2.release()); OwnPtr res2 = r2.wait(); EXPECT_EQ(kDone, res2->result()); EXPECT_EQ("hello, world", toString(res2->data())); } TEST(DataConsumerTeeTest, DetachOneDestination) { OwnPtr src(Handle::create()); OwnPtr dest1, dest2; src->add(Command(Command::Data, "hello, ")); src->add(Command(Command::Data, "world")); src->add(Command(Command::Done)); OwnPtr> t = adoptPtr(new TeeCreationThread()); t->run(src.release(), &dest1, &dest2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); dest1 = nullptr; HandleReaderRunner r2(dest2.release()); OwnPtr res2 = r2.wait(); EXPECT_EQ(kDone, res2->result()); EXPECT_EQ("hello, world", toString(res2->data())); } TEST(DataConsumerTeeTest, DetachBothDestinationsShouldStopSourceReader) { OwnPtr src(Handle::create()); RefPtr context(src->getContext()); OwnPtr dest1, dest2; src->add(Command(Command::Data, "hello, ")); src->add(Command(Command::Data, "world")); OwnPtr> t = adoptPtr(new TeeCreationThread()); t->run(src.release(), &dest1, &dest2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); dest1 = nullptr; dest2 = nullptr; // Collect garbage to finalize the source reader. Heap::collectAllGarbage(); context->detached()->wait(); } TEST(FetchDataConsumerTeeTest, Create) { RefPtr blobDataHandle = BlobDataHandle::create(); OwnPtr src(MockFetchDataConsumerHandle::create()); OwnPtr reader(MockFetchDataConsumerReader::create()); Checkpoint checkpoint; InSequence s; EXPECT_CALL(checkpoint, Call(1)); EXPECT_CALL(*src, obtainReaderInternal(_)).WillOnce(Return(reader.get())); EXPECT_CALL(*reader, drainAsBlobDataHandle(kAllowBlobWithInvalidSize)).WillOnce(Return(blobDataHandle)); EXPECT_CALL(*reader, destruct()); EXPECT_CALL(checkpoint, Call(2)); // |reader| is adopted by |obtainReader|. ASSERT_TRUE(reader.leakPtr()); OwnPtr dest1, dest2; OwnPtr> t = adoptPtr(new TeeCreationThread()); checkpoint.Call(1); t->run(src.release(), &dest1, &dest2); checkpoint.Call(2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); EXPECT_EQ(blobDataHandle, dest1->obtainReader(nullptr)->drainAsBlobDataHandle(kAllowBlobWithInvalidSize)); EXPECT_EQ(blobDataHandle, dest2->obtainReader(nullptr)->drainAsBlobDataHandle(kAllowBlobWithInvalidSize)); } TEST(FetchDataConsumerTeeTest, CreateFromBlobWithInvalidSize) { RefPtr blobDataHandle = BlobDataHandle::create(BlobData::create(), -1); OwnPtr src(MockFetchDataConsumerHandle::create()); OwnPtr reader(MockFetchDataConsumerReader::create()); Checkpoint checkpoint; InSequence s; EXPECT_CALL(checkpoint, Call(1)); EXPECT_CALL(*src, obtainReaderInternal(_)).WillOnce(Return(reader.get())); EXPECT_CALL(*reader, drainAsBlobDataHandle(kAllowBlobWithInvalidSize)).WillOnce(Return(blobDataHandle)); EXPECT_CALL(*reader, destruct()); EXPECT_CALL(checkpoint, Call(2)); // |reader| is adopted by |obtainReader|. ASSERT_TRUE(reader.leakPtr()); OwnPtr dest1, dest2; OwnPtr> t = adoptPtr(new TeeCreationThread()); checkpoint.Call(1); t->run(src.release(), &dest1, &dest2); checkpoint.Call(2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); EXPECT_FALSE(dest1->obtainReader(nullptr)->drainAsBlobDataHandle(kDisallowBlobWithInvalidSize)); EXPECT_EQ(blobDataHandle, dest1->obtainReader(nullptr)->drainAsBlobDataHandle(kAllowBlobWithInvalidSize)); EXPECT_FALSE(dest2->obtainReader(nullptr)->drainAsBlobDataHandle(kDisallowBlobWithInvalidSize)); EXPECT_EQ(blobDataHandle, dest2->obtainReader(nullptr)->drainAsBlobDataHandle(kAllowBlobWithInvalidSize)); } TEST(FetchDataConsumerTeeTest, CreateDone) { OwnPtr src(Handle::create()); OwnPtr dest1, dest2; src->add(Command(Command::Done)); OwnPtr> t = adoptPtr(new TeeCreationThread()); t->run(createFetchDataConsumerHandleFromWebHandle(src.release()), &dest1, &dest2); ASSERT_TRUE(dest1); ASSERT_TRUE(dest2); EXPECT_FALSE(dest1->obtainReader(nullptr)->drainAsBlobDataHandle(kAllowBlobWithInvalidSize)); EXPECT_FALSE(dest2->obtainReader(nullptr)->drainAsBlobDataHandle(kAllowBlobWithInvalidSize)); HandleReaderRunner r1(dest1.release()), r2(dest2.release()); OwnPtr res1 = r1.wait(); OwnPtr res2 = r2.wait(); EXPECT_EQ(kDone, res1->result()); EXPECT_EQ(0u, res1->data().size()); EXPECT_EQ(kDone, res2->result()); EXPECT_EQ(0u, res2->data().size()); } } // namespace } // namespace blink