blob: a55257e1df9981e92d3fa3f7a31528fab4fc2533 [file] [log] [blame]
// 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/FetchFormDataConsumerHandle.h"
#include "core/dom/DOMTypedArray.h"
#include "core/html/FormData.h"
#include "core/loader/MockThreadableLoader.h"
#include "core/loader/ThreadableLoaderClient.h"
#include "core/testing/DummyPageHolder.h"
#include "modules/fetch/DataConsumerHandleTestUtil.h"
#include "platform/network/ResourceResponse.h"
#include "platform/testing/UnitTestHelpers.h"
#include "platform/weborigin/KURL.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "wtf/OwnPtr.h"
#include "wtf/PassOwnPtr.h"
#include "wtf/PassRefPtr.h"
#include "wtf/RefPtr.h"
#include "wtf/Vector.h"
#include "wtf/text/TextEncoding.h"
#include "wtf/text/WTFString.h"
#include <string.h>
namespace blink {
namespace {
using Result = WebDataConsumerHandle::Result;
const Result kOk = WebDataConsumerHandle::Ok;
const Result kDone = WebDataConsumerHandle::Done;
const Result kShouldWait = WebDataConsumerHandle::ShouldWait;
const WebDataConsumerHandle::Flags kNone = WebDataConsumerHandle::FlagNone;
using HandleReader = DataConsumerHandleTestUtil::HandleReader;
using HandleTwoPhaseReader = DataConsumerHandleTestUtil::HandleTwoPhaseReader;
using HandleReadResult = DataConsumerHandleTestUtil::HandleReadResult;
template <typename T>
using HandleReaderRunner = DataConsumerHandleTestUtil::HandleReaderRunner<T>;
using ReplayingHandle = DataConsumerHandleTestUtil::ReplayingHandle;
using Command = DataConsumerHandleTestUtil::Command;
using ::testing::_;
using ::testing::InvokeWithoutArgs;
String toString(const Vector<char>& data)
{
return String(data.data(), data.size());
}
class LoaderFactory : public FetchBlobDataConsumerHandle::LoaderFactory {
public:
explicit LoaderFactory(PassOwnPtr<WebDataConsumerHandle> handle)
: m_client(nullptr)
, m_handle(handle) {}
PassOwnPtr<ThreadableLoader> create(ExecutionContext&, ThreadableLoaderClient* client, const ThreadableLoaderOptions&, const ResourceLoaderOptions&) override
{
m_client = client;
OwnPtr<MockThreadableLoader> loader = MockThreadableLoader::create();
EXPECT_CALL(*loader, start(_)).WillOnce(InvokeWithoutArgs(this, &LoaderFactory::handleDidReceiveResponse));
EXPECT_CALL(*loader, cancel()).Times(1);
return loader.release();
}
private:
void handleDidReceiveResponse()
{
m_client->didReceiveResponse(0, ResourceResponse(), m_handle.release());
}
ThreadableLoaderClient* m_client;
OwnPtr<WebDataConsumerHandle> m_handle;
};
class FetchFormDataConsumerHandleTest : public ::testing::Test {
public:
FetchFormDataConsumerHandleTest() : m_page(DummyPageHolder::create(IntSize(1, 1))) {}
protected:
Document* getDocument() { return &m_page->document(); }
OwnPtr<DummyPageHolder> m_page;
};
PassRefPtr<EncodedFormData> complexFormData()
{
RefPtr<EncodedFormData> data = EncodedFormData::create();
data->appendData("foo", 3);
data->appendFileRange("/foo/bar/baz", 3, 4, 5);
data->appendFileSystemURLRange(KURL(KURL(), "file:///foo/bar/baz"), 6, 7, 8);
OwnPtr<BlobData> blobData = BlobData::create();
blobData->appendText("hello", false);
auto size = blobData->length();
RefPtr<BlobDataHandle> blobDataHandle = BlobDataHandle::create(blobData.release(), size);
data->appendBlob(blobDataHandle->uuid(), blobDataHandle);
Vector<char> boundary;
boundary.append("\0", 1);
data->setBoundary(boundary);
return data.release();
}
void verifyComplexFormData(EncodedFormData* data)
{
const auto& elements = data->elements();
if (4 != elements.size()) {
FAIL() << "data->elements().size() should be 4, but is " << data->elements().size() << ".";
}
EXPECT_EQ(FormDataElement::data, elements[0].m_type);
EXPECT_EQ("foo", String(elements[0].m_data.data(), elements[0].m_data.size()));
EXPECT_EQ(FormDataElement::encodedFile, elements[1].m_type);
EXPECT_EQ("/foo/bar/baz", elements[1].m_filename);
EXPECT_EQ(3, elements[1].m_fileStart);
EXPECT_EQ(4, elements[1].m_fileLength);
EXPECT_EQ(5, elements[1].m_expectedFileModificationTime);
EXPECT_EQ(FormDataElement::encodedFileSystemURL, elements[2].m_type);
EXPECT_EQ(KURL(KURL(), "file:///foo/bar/baz"), elements[2].m_fileSystemURL);
EXPECT_EQ(6, elements[2].m_fileStart);
EXPECT_EQ(7, elements[2].m_fileLength);
EXPECT_EQ(8, elements[2].m_expectedFileModificationTime);
EXPECT_EQ(FormDataElement::encodedBlob, elements[3].m_type);
if (!elements[3].m_optionalBlobDataHandle) {
FAIL() << "optional BlobDataHandle must be set.";
}
EXPECT_EQ(elements[3].m_blobUUID, elements[3].m_optionalBlobDataHandle->uuid());
EXPECT_EQ(5u, elements[3].m_optionalBlobDataHandle->size());
}
TEST_F(FetchFormDataConsumerHandleTest, ReadFromString)
{
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(String("hello, world"));
HandleReaderRunner<HandleReader> runner(handle.release());
OwnPtr<HandleReadResult> r = runner.wait();
EXPECT_EQ(kDone, r->result());
EXPECT_EQ("hello, world", toString(r->data()));
}
TEST_F(FetchFormDataConsumerHandleTest, TwoPhaseReadFromString)
{
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(String("hello, world"));
HandleReaderRunner<HandleTwoPhaseReader> runner(handle.release());
OwnPtr<HandleReadResult> r = runner.wait();
EXPECT_EQ(kDone, r->result());
EXPECT_EQ("hello, world", toString(r->data()));
}
TEST_F(FetchFormDataConsumerHandleTest, ReadFromStringNonLatin)
{
UChar cs[] = {0x3042, 0};
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(String(cs));
HandleReaderRunner<HandleReader> runner(handle.release());
OwnPtr<HandleReadResult> r = runner.wait();
EXPECT_EQ(kDone, r->result());
EXPECT_EQ("\xe3\x81\x82", toString(r->data()));
}
TEST_F(FetchFormDataConsumerHandleTest, ReadFromArrayBuffer)
{
const unsigned char data[] = { 0x21, 0xfe, 0x00, 0x00, 0xff, 0xa3, 0x42, 0x30, 0x42, 0x99, 0x88 };
DOMArrayBuffer* buffer = DOMArrayBuffer::create(data, WTF_ARRAY_LENGTH(data));
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(buffer);
HandleReaderRunner<HandleReader> runner(handle.release());
OwnPtr<HandleReadResult> r = runner.wait();
EXPECT_EQ(kDone, r->result());
Vector<char> expected;
expected.append(data, WTF_ARRAY_LENGTH(data));
EXPECT_EQ(expected, r->data());
}
TEST_F(FetchFormDataConsumerHandleTest, ReadFromArrayBufferView)
{
const unsigned char data[] = { 0x21, 0xfe, 0x00, 0x00, 0xff, 0xa3, 0x42, 0x30, 0x42, 0x99, 0x88 };
const size_t offset = 1, size = 4;
DOMArrayBuffer* buffer = DOMArrayBuffer::create(data, WTF_ARRAY_LENGTH(data));
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(DOMUint8Array::create(buffer, offset, size));
HandleReaderRunner<HandleReader> runner(handle.release());
OwnPtr<HandleReadResult> r = runner.wait();
EXPECT_EQ(kDone, r->result());
Vector<char> expected;
expected.append(data + offset, size);
EXPECT_EQ(expected, r->data());
}
TEST_F(FetchFormDataConsumerHandleTest, ReadFromSimplFormData)
{
RefPtr<EncodedFormData> data = EncodedFormData::create();
data->appendData("foo", 3);
data->appendData("hoge", 4);
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(getDocument(), data);
HandleReaderRunner<HandleReader> runner(handle.release());
testing::runPendingTasks();
OwnPtr<HandleReadResult> r = runner.wait();
EXPECT_EQ(kDone, r->result());
EXPECT_EQ("foohoge", toString(r->data()));
}
TEST_F(FetchFormDataConsumerHandleTest, ReadFromComplexFormData)
{
RefPtr<EncodedFormData> data = complexFormData();
OwnPtr<ReplayingHandle> src = ReplayingHandle::create();
src->add(Command(Command::Data, "bar"));
src->add(Command(Command::Done));
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::createForTest(getDocument(), data, new LoaderFactory(src.release()));
char c;
size_t readSize;
EXPECT_EQ(kShouldWait, handle->obtainReader(nullptr)->read(&c, 1, kNone, &readSize));
HandleReaderRunner<HandleReader> runner(handle.release());
testing::runPendingTasks();
OwnPtr<HandleReadResult> r = runner.wait();
EXPECT_EQ(kDone, r->result());
EXPECT_EQ("bar", toString(r->data()));
}
TEST_F(FetchFormDataConsumerHandleTest, TwoPhaseReadFromComplexFormData)
{
RefPtr<EncodedFormData> data = complexFormData();
OwnPtr<ReplayingHandle> src = ReplayingHandle::create();
src->add(Command(Command::Data, "bar"));
src->add(Command(Command::Done));
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::createForTest(getDocument(), data, new LoaderFactory(src.release()));
char c;
size_t readSize;
EXPECT_EQ(kShouldWait, handle->obtainReader(nullptr)->read(&c, 1, kNone, &readSize));
HandleReaderRunner<HandleTwoPhaseReader> runner(handle.release());
testing::runPendingTasks();
OwnPtr<HandleReadResult> r = runner.wait();
EXPECT_EQ(kDone, r->result());
EXPECT_EQ("bar", toString(r->data()));
}
TEST_F(FetchFormDataConsumerHandleTest, DrainAsBlobDataHandleFromString)
{
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(String("hello, world"));
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
RefPtr<BlobDataHandle> blobDataHandle = reader->drainAsBlobDataHandle();
ASSERT_TRUE(blobDataHandle);
EXPECT_EQ(String(), blobDataHandle->type());
EXPECT_EQ(12u, blobDataHandle->size());
EXPECT_EQ(nullptr, reader->drainAsFormData());
char c;
size_t readSize;
EXPECT_EQ(kDone, reader->read(&c, 1, kNone, &readSize));
}
TEST_F(FetchFormDataConsumerHandleTest, DrainAsBlobDataHandleFromArrayBuffer)
{
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(DOMArrayBuffer::create("foo", 3));
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
RefPtr<BlobDataHandle> blobDataHandle = reader->drainAsBlobDataHandle();
ASSERT_TRUE(blobDataHandle);
EXPECT_EQ(String(), blobDataHandle->type());
EXPECT_EQ(3u, blobDataHandle->size());
EXPECT_EQ(nullptr, reader->drainAsFormData());
char c;
size_t readSize;
EXPECT_EQ(kDone, reader->read(&c, 1, kNone, &readSize));
}
TEST_F(FetchFormDataConsumerHandleTest, DrainAsBlobDataHandleFromSimpleFormData)
{
FormData* data = FormData::create(UTF8Encoding());
data->append("name1", "value1");
data->append("name2", "value2");
RefPtr<EncodedFormData> inputFormData = data->encodeMultiPartFormData();
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(getDocument(), inputFormData);
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
RefPtr<BlobDataHandle> blobDataHandle = reader->drainAsBlobDataHandle();
ASSERT_TRUE(blobDataHandle);
EXPECT_EQ(String(), blobDataHandle->type());
EXPECT_EQ(inputFormData->flattenToString().utf8().length(), blobDataHandle->size());
EXPECT_EQ(nullptr, reader->drainAsFormData());
char c;
size_t readSize;
EXPECT_EQ(kDone, reader->read(&c, 1, kNone, &readSize));
}
TEST_F(FetchFormDataConsumerHandleTest, DrainAsBlobDataHandleFromComplexFormData)
{
RefPtr<EncodedFormData> inputFormData = complexFormData();
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(getDocument(), inputFormData);
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
RefPtr<BlobDataHandle> blobDataHandle = reader->drainAsBlobDataHandle();
ASSERT_TRUE(blobDataHandle);
EXPECT_EQ(nullptr, reader->drainAsFormData());
char c;
size_t readSize;
EXPECT_EQ(kDone, reader->read(&c, 1, kNone, &readSize));
}
TEST_F(FetchFormDataConsumerHandleTest, DrainAsFormDataFromString)
{
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(String("hello, world"));
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
RefPtr<EncodedFormData> formData = reader->drainAsFormData();
ASSERT_TRUE(formData);
EXPECT_TRUE(formData->isSafeToSendToAnotherThread());
EXPECT_EQ("hello, world", formData->flattenToString());
const void* buffer = nullptr;
size_t size;
EXPECT_EQ(kDone, reader->read(nullptr, 0, kNone, &size));
EXPECT_EQ(kDone, reader->beginRead(&buffer, kNone, &size));
}
TEST_F(FetchFormDataConsumerHandleTest, DrainAsFormDataFromArrayBuffer)
{
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(DOMArrayBuffer::create("foo", 3));
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
RefPtr<EncodedFormData> formData = reader->drainAsFormData();
ASSERT_TRUE(formData);
EXPECT_TRUE(formData->isSafeToSendToAnotherThread());
EXPECT_EQ("foo", formData->flattenToString());
}
TEST_F(FetchFormDataConsumerHandleTest, DrainAsFormDataFromSimpleFormData)
{
FormData* data = FormData::create(UTF8Encoding());
data->append("name1", "value1");
data->append("name2", "value2");
RefPtr<EncodedFormData> inputFormData = data->encodeMultiPartFormData();
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(getDocument(), inputFormData);
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
RefPtr<EncodedFormData> outputFormData = reader->drainAsFormData();
ASSERT_TRUE(outputFormData);
EXPECT_TRUE(outputFormData->isSafeToSendToAnotherThread());
EXPECT_NE(outputFormData.get(), inputFormData.get());
EXPECT_EQ(inputFormData->flattenToString(), outputFormData->flattenToString());
}
TEST_F(FetchFormDataConsumerHandleTest, DrainAsFormDataFromComplexFormData)
{
RefPtr<EncodedFormData> inputFormData = complexFormData();
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(getDocument(), inputFormData);
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
RefPtr<EncodedFormData> outputFormData = reader->drainAsFormData();
ASSERT_TRUE(outputFormData);
EXPECT_TRUE(outputFormData->isSafeToSendToAnotherThread());
EXPECT_NE(outputFormData.get(), inputFormData.get());
verifyComplexFormData(outputFormData.get());
}
TEST_F(FetchFormDataConsumerHandleTest, ZeroByteReadDoesNotAffectDraining)
{
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(String("hello, world"));
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
size_t readSize;
EXPECT_EQ(kOk, reader->read(nullptr, 0, kNone, &readSize));
RefPtr<EncodedFormData> formData = reader->drainAsFormData();
ASSERT_TRUE(formData);
EXPECT_TRUE(formData->isSafeToSendToAnotherThread());
EXPECT_EQ("hello, world", formData->flattenToString());
}
TEST_F(FetchFormDataConsumerHandleTest, OneByteReadAffectsDraining)
{
char c;
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(String("hello, world"));
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
size_t readSize;
EXPECT_EQ(kOk, reader->read(&c, 1, kNone, &readSize));
EXPECT_EQ(1u, readSize);
EXPECT_EQ('h', c);
EXPECT_FALSE(reader->drainAsFormData());
}
TEST_F(FetchFormDataConsumerHandleTest, BeginReadAffectsDraining)
{
const void* buffer = nullptr;
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::create(String("hello, world"));
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
size_t available;
EXPECT_EQ(kOk, reader->beginRead(&buffer, kNone, &available));
ASSERT_TRUE(buffer);
EXPECT_EQ("hello, world", String(static_cast<const char*>(buffer), available));
EXPECT_FALSE(reader->drainAsFormData());
reader->endRead(0);
EXPECT_FALSE(reader->drainAsFormData());
}
TEST_F(FetchFormDataConsumerHandleTest, ZeroByteReadDoesNotAffectDrainingForComplexFormData)
{
OwnPtr<ReplayingHandle> src = ReplayingHandle::create();
src->add(Command(Command::Data, "bar"));
src->add(Command(Command::Done));
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::createForTest(getDocument(), complexFormData(), new LoaderFactory(src.release()));
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
size_t readSize;
EXPECT_EQ(kShouldWait, reader->read(nullptr, 0, kNone, &readSize));
testing::runPendingTasks();
EXPECT_EQ(kOk, reader->read(nullptr, 0, kNone, &readSize));
RefPtr<EncodedFormData> formData = reader->drainAsFormData();
ASSERT_TRUE(formData);
EXPECT_TRUE(formData->isSafeToSendToAnotherThread());
verifyComplexFormData(formData.get());
}
TEST_F(FetchFormDataConsumerHandleTest, OneByteReadAffectsDrainingForComplexFormData)
{
OwnPtr<ReplayingHandle> src = ReplayingHandle::create();
src->add(Command(Command::Data, "bar"));
src->add(Command(Command::Done));
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::createForTest(getDocument(), complexFormData(), new LoaderFactory(src.release()));
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
char c;
size_t readSize;
EXPECT_EQ(kShouldWait, reader->read(&c, 1, kNone, &readSize));
testing::runPendingTasks();
EXPECT_EQ(kOk, reader->read(&c, 1, kNone, &readSize));
EXPECT_EQ(1u, readSize);
EXPECT_EQ('b', c);
EXPECT_FALSE(reader->drainAsFormData());
}
TEST_F(FetchFormDataConsumerHandleTest, BeginReadAffectsDrainingForComplexFormData)
{
OwnPtr<ReplayingHandle> src = ReplayingHandle::create();
src->add(Command(Command::Data, "bar"));
src->add(Command(Command::Done));
const void* buffer = nullptr;
OwnPtr<FetchDataConsumerHandle> handle = FetchFormDataConsumerHandle::createForTest(getDocument(), complexFormData(), new LoaderFactory(src.release()));
OwnPtr<FetchDataConsumerHandle::Reader> reader = handle->obtainReader(nullptr);
size_t available;
EXPECT_EQ(kShouldWait, reader->beginRead(&buffer, kNone, &available));
testing::runPendingTasks();
EXPECT_EQ(kOk, reader->beginRead(&buffer, kNone, &available));
EXPECT_FALSE(reader->drainAsFormData());
reader->endRead(0);
EXPECT_FALSE(reader->drainAsFormData());
}
} // namespace
} // namespace blink