blob: 1a1b58359f6e17ea73801d1fc80c3dea174214f6 [file] [log] [blame]
// Copyright 2016 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 "third_party/blink/renderer/core/fetch/readable_stream_bytes_consumer.h"
#include <memory>
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/blink/renderer/bindings/core/v8/v8_binding_for_core.h"
#include "third_party/blink/renderer/bindings/core/v8/v8_binding_for_testing.h"
#include "third_party/blink/renderer/core/fetch/bytes_consumer_test_util.h"
#include "third_party/blink/renderer/core/streams/readable_stream.h"
#include "third_party/blink/renderer/core/streams/test_underlying_source.h"
#include "third_party/blink/renderer/platform/bindings/script_state.h"
#include "third_party/blink/renderer/platform/bindings/v8_binding_macros.h"
#include "third_party/blink/renderer/platform/heap/handle.h"
#include "third_party/blink/renderer/platform/testing/unit_test_helpers.h"
#include "v8/include/v8.h"
namespace blink {
namespace {
using testing::InSequence;
using testing::StrictMock;
using Checkpoint = StrictMock<testing::MockFunction<void(int)>>;
using Result = BytesConsumer::Result;
using PublicState = BytesConsumer::PublicState;
class MockClient : public GarbageCollectedFinalized<MockClient>,
public BytesConsumer::Client {
USING_GARBAGE_COLLECTED_MIXIN(MockClient);
public:
static MockClient* Create() { return new StrictMock<MockClient>(); }
MOCK_METHOD0(OnStateChange, void());
String DebugName() const override { return "MockClient"; }
void Trace(blink::Visitor* visitor) override {}
protected:
MockClient() = default;
};
TEST(ReadableStreamBytesConsumerTest, Create) {
V8TestingScope scope;
ScriptState* script_state = scope.GetScriptState();
ExceptionState& exception_state = scope.GetExceptionState();
auto* stream = ReadableStream::Create(script_state, exception_state);
ASSERT_TRUE(stream);
ASSERT_FALSE(exception_state.HadException());
ScriptValue reader = stream->getReader(script_state, exception_state);
ASSERT_FALSE(reader.IsEmpty());
ASSERT_FALSE(exception_state.HadException());
Persistent<BytesConsumer> consumer =
MakeGarbageCollected<ReadableStreamBytesConsumer>(script_state, reader);
EXPECT_EQ(PublicState::kReadableOrWaiting, consumer->GetPublicState());
}
TEST(ReadableStreamBytesConsumerTest, EmptyStream) {
V8TestingScope scope;
ScriptState* script_state = scope.GetScriptState();
ExceptionState& exception_state = scope.GetExceptionState();
auto* underlying_source =
MakeGarbageCollected<TestUnderlyingSource>(script_state);
auto* stream = ReadableStream::CreateWithCountQueueingStrategy(
script_state, underlying_source, 0);
underlying_source->Close();
ScriptValue reader = stream->getReader(script_state, exception_state);
ASSERT_FALSE(reader.IsEmpty());
ASSERT_FALSE(exception_state.HadException());
Persistent<BytesConsumer> consumer =
MakeGarbageCollected<ReadableStreamBytesConsumer>(script_state, reader);
Persistent<MockClient> client = MockClient::Create();
consumer->SetClient(client);
Checkpoint checkpoint;
InSequence s;
EXPECT_CALL(checkpoint, Call(1));
EXPECT_CALL(checkpoint, Call(2));
EXPECT_CALL(checkpoint, Call(3));
EXPECT_CALL(*client, OnStateChange());
EXPECT_CALL(checkpoint, Call(4));
const char* buffer = nullptr;
size_t available = 0;
checkpoint.Call(1);
test::RunPendingTasks();
checkpoint.Call(2);
EXPECT_EQ(PublicState::kReadableOrWaiting, consumer->GetPublicState());
EXPECT_EQ(Result::kShouldWait, consumer->BeginRead(&buffer, &available));
checkpoint.Call(3);
test::RunPendingTasks();
checkpoint.Call(4);
EXPECT_EQ(PublicState::kClosed, consumer->GetPublicState());
EXPECT_EQ(Result::kDone, consumer->BeginRead(&buffer, &available));
}
TEST(ReadableStreamBytesConsumerTest, ErroredStream) {
V8TestingScope scope;
ScriptState* script_state = scope.GetScriptState();
ExceptionState& exception_state = scope.GetExceptionState();
auto* underlying_source =
MakeGarbageCollected<TestUnderlyingSource>(script_state);
auto* stream = ReadableStream::CreateWithCountQueueingStrategy(
script_state, underlying_source, 0);
underlying_source->SetError(
ScriptValue(script_state, v8::Undefined(script_state->GetIsolate())));
ScriptValue reader = stream->getReader(script_state, exception_state);
ASSERT_FALSE(reader.IsEmpty());
ASSERT_FALSE(exception_state.HadException());
Persistent<BytesConsumer> consumer =
MakeGarbageCollected<ReadableStreamBytesConsumer>(script_state, reader);
Persistent<MockClient> client = MockClient::Create();
consumer->SetClient(client);
Checkpoint checkpoint;
InSequence s;
EXPECT_CALL(checkpoint, Call(1));
EXPECT_CALL(checkpoint, Call(2));
EXPECT_CALL(checkpoint, Call(3));
EXPECT_CALL(*client, OnStateChange());
EXPECT_CALL(checkpoint, Call(4));
const char* buffer = nullptr;
size_t available = 0;
checkpoint.Call(1);
test::RunPendingTasks();
checkpoint.Call(2);
EXPECT_EQ(PublicState::kReadableOrWaiting, consumer->GetPublicState());
EXPECT_EQ(Result::kShouldWait, consumer->BeginRead(&buffer, &available));
checkpoint.Call(3);
test::RunPendingTasks();
checkpoint.Call(4);
EXPECT_EQ(PublicState::kErrored, consumer->GetPublicState());
EXPECT_EQ(Result::kError, consumer->BeginRead(&buffer, &available));
}
TEST(ReadableStreamBytesConsumerTest, TwoPhaseRead) {
V8TestingScope scope;
ScriptState* script_state = scope.GetScriptState();
ExceptionState& exception_state = scope.GetExceptionState();
auto* underlying_source =
MakeGarbageCollected<TestUnderlyingSource>(script_state);
auto* stream = ReadableStream::CreateWithCountQueueingStrategy(
script_state, underlying_source, 0);
{
auto* chunk1 = DOMUint8Array::Create(0);
auto* chunk2 = DOMUint8Array::Create(4);
chunk2->Data()[0] = 0x43;
chunk2->Data()[1] = 0x44;
chunk2->Data()[2] = 0x45;
chunk2->Data()[3] = 0x46;
auto* chunk3 = DOMUint8Array::Create(4);
chunk3->Data()[0] = 0x47;
chunk3->Data()[1] = 0x48;
chunk3->Data()[2] = 0x49;
chunk3->Data()[3] = 0x4a;
underlying_source->Enqueue(
ScriptValue(script_state, ToV8(chunk1, script_state)));
underlying_source->Enqueue(
ScriptValue(script_state, ToV8(chunk2, script_state)));
underlying_source->Enqueue(
ScriptValue(script_state, ToV8(chunk3, script_state)));
underlying_source->Close();
}
ScriptValue reader = stream->getReader(script_state, exception_state);
ASSERT_FALSE(reader.IsEmpty());
ASSERT_FALSE(exception_state.HadException());
Persistent<BytesConsumer> consumer =
MakeGarbageCollected<ReadableStreamBytesConsumer>(script_state, reader);
Persistent<MockClient> client = MockClient::Create();
consumer->SetClient(client);
Checkpoint checkpoint;
InSequence s;
EXPECT_CALL(checkpoint, Call(1));
EXPECT_CALL(checkpoint, Call(2));
EXPECT_CALL(checkpoint, Call(3));
EXPECT_CALL(*client, OnStateChange());
EXPECT_CALL(checkpoint, Call(4));
EXPECT_CALL(checkpoint, Call(5));
EXPECT_CALL(*client, OnStateChange());
EXPECT_CALL(checkpoint, Call(6));
EXPECT_CALL(checkpoint, Call(7));
EXPECT_CALL(*client, OnStateChange());
EXPECT_CALL(checkpoint, Call(8));
EXPECT_CALL(checkpoint, Call(9));
EXPECT_CALL(*client, OnStateChange());
EXPECT_CALL(checkpoint, Call(10));
const char* buffer = nullptr;
size_t available = 0;
checkpoint.Call(1);
test::RunPendingTasks();
checkpoint.Call(2);
EXPECT_EQ(Result::kShouldWait, consumer->BeginRead(&buffer, &available));
checkpoint.Call(3);
test::RunPendingTasks();
checkpoint.Call(4);
EXPECT_EQ(Result::kOk, consumer->BeginRead(&buffer, &available));
ASSERT_EQ(0u, available);
EXPECT_EQ(Result::kOk, consumer->EndRead(0));
EXPECT_EQ(Result::kShouldWait, consumer->BeginRead(&buffer, &available));
checkpoint.Call(5);
test::RunPendingTasks();
checkpoint.Call(6);
EXPECT_EQ(PublicState::kReadableOrWaiting, consumer->GetPublicState());
EXPECT_EQ(Result::kOk, consumer->BeginRead(&buffer, &available));
ASSERT_EQ(4u, available);
EXPECT_EQ(0x43, buffer[0]);
EXPECT_EQ(0x44, buffer[1]);
EXPECT_EQ(0x45, buffer[2]);
EXPECT_EQ(0x46, buffer[3]);
EXPECT_EQ(Result::kOk, consumer->EndRead(0));
EXPECT_EQ(Result::kOk, consumer->BeginRead(&buffer, &available));
ASSERT_EQ(4u, available);
EXPECT_EQ(0x43, buffer[0]);
EXPECT_EQ(0x44, buffer[1]);
EXPECT_EQ(0x45, buffer[2]);
EXPECT_EQ(0x46, buffer[3]);
EXPECT_EQ(Result::kOk, consumer->EndRead(1));
EXPECT_EQ(Result::kOk, consumer->BeginRead(&buffer, &available));
ASSERT_EQ(3u, available);
EXPECT_EQ(0x44, buffer[0]);
EXPECT_EQ(0x45, buffer[1]);
EXPECT_EQ(0x46, buffer[2]);
EXPECT_EQ(Result::kOk, consumer->EndRead(3));
EXPECT_EQ(Result::kShouldWait, consumer->BeginRead(&buffer, &available));
checkpoint.Call(7);
test::RunPendingTasks();
checkpoint.Call(8);
EXPECT_EQ(PublicState::kReadableOrWaiting, consumer->GetPublicState());
EXPECT_EQ(Result::kOk, consumer->BeginRead(&buffer, &available));
ASSERT_EQ(4u, available);
EXPECT_EQ(0x47, buffer[0]);
EXPECT_EQ(0x48, buffer[1]);
EXPECT_EQ(0x49, buffer[2]);
EXPECT_EQ(0x4a, buffer[3]);
EXPECT_EQ(Result::kOk, consumer->EndRead(4));
EXPECT_EQ(Result::kShouldWait, consumer->BeginRead(&buffer, &available));
checkpoint.Call(9);
test::RunPendingTasks();
checkpoint.Call(10);
EXPECT_EQ(PublicState::kClosed, consumer->GetPublicState());
EXPECT_EQ(Result::kDone, consumer->BeginRead(&buffer, &available));
}
TEST(ReadableStreamBytesConsumerTest, EnqueueUndefined) {
V8TestingScope scope;
ScriptState* script_state = scope.GetScriptState();
ExceptionState& exception_state = scope.GetExceptionState();
auto* underlying_source =
MakeGarbageCollected<TestUnderlyingSource>(script_state);
auto* stream = ReadableStream::CreateWithCountQueueingStrategy(
script_state, underlying_source, 0);
underlying_source->Enqueue(
ScriptValue(script_state, v8::Undefined(script_state->GetIsolate())));
underlying_source->Close();
ScriptValue reader = stream->getReader(script_state, exception_state);
ASSERT_FALSE(reader.IsEmpty());
ASSERT_FALSE(exception_state.HadException());
Persistent<BytesConsumer> consumer =
MakeGarbageCollected<ReadableStreamBytesConsumer>(script_state, reader);
Persistent<MockClient> client = MockClient::Create();
consumer->SetClient(client);
Checkpoint checkpoint;
InSequence s;
EXPECT_CALL(checkpoint, Call(1));
EXPECT_CALL(checkpoint, Call(2));
EXPECT_CALL(checkpoint, Call(3));
EXPECT_CALL(*client, OnStateChange());
EXPECT_CALL(checkpoint, Call(4));
const char* buffer = nullptr;
size_t available = 0;
checkpoint.Call(1);
test::RunPendingTasks();
checkpoint.Call(2);
EXPECT_EQ(PublicState::kReadableOrWaiting, consumer->GetPublicState());
EXPECT_EQ(Result::kShouldWait, consumer->BeginRead(&buffer, &available));
checkpoint.Call(3);
test::RunPendingTasks();
checkpoint.Call(4);
EXPECT_EQ(PublicState::kErrored, consumer->GetPublicState());
EXPECT_EQ(Result::kError, consumer->BeginRead(&buffer, &available));
}
TEST(ReadableStreamBytesConsumerTest, EnqueueNull) {
V8TestingScope scope;
ScriptState* script_state = scope.GetScriptState();
ExceptionState& exception_state = scope.GetExceptionState();
auto* underlying_source =
MakeGarbageCollected<TestUnderlyingSource>(script_state);
auto* stream = ReadableStream::CreateWithCountQueueingStrategy(
script_state, underlying_source, 0);
underlying_source->Enqueue(
ScriptValue(script_state, v8::Null(script_state->GetIsolate())));
underlying_source->Close();
ScriptValue reader = stream->getReader(script_state, exception_state);
ASSERT_FALSE(reader.IsEmpty());
ASSERT_FALSE(exception_state.HadException());
Persistent<BytesConsumer> consumer =
MakeGarbageCollected<ReadableStreamBytesConsumer>(script_state, reader);
Persistent<MockClient> client = MockClient::Create();
consumer->SetClient(client);
Checkpoint checkpoint;
InSequence s;
EXPECT_CALL(checkpoint, Call(1));
EXPECT_CALL(checkpoint, Call(2));
EXPECT_CALL(checkpoint, Call(3));
EXPECT_CALL(*client, OnStateChange());
EXPECT_CALL(checkpoint, Call(4));
const char* buffer = nullptr;
size_t available = 0;
checkpoint.Call(1);
test::RunPendingTasks();
checkpoint.Call(2);
EXPECT_EQ(PublicState::kReadableOrWaiting, consumer->GetPublicState());
EXPECT_EQ(Result::kShouldWait, consumer->BeginRead(&buffer, &available));
checkpoint.Call(3);
test::RunPendingTasks();
checkpoint.Call(4);
EXPECT_EQ(PublicState::kErrored, consumer->GetPublicState());
EXPECT_EQ(Result::kError, consumer->BeginRead(&buffer, &available));
}
TEST(ReadableStreamBytesConsumerTest, EnqueueString) {
V8TestingScope scope;
ScriptState* script_state = scope.GetScriptState();
ExceptionState& exception_state = scope.GetExceptionState();
auto* underlying_source =
MakeGarbageCollected<TestUnderlyingSource>(script_state);
auto* stream = ReadableStream::CreateWithCountQueueingStrategy(
script_state, underlying_source, 0);
underlying_source->Enqueue(
ScriptValue(script_state, V8String(script_state->GetIsolate(), "hello")));
underlying_source->Close();
ScriptValue reader = stream->getReader(script_state, exception_state);
ASSERT_FALSE(reader.IsEmpty());
ASSERT_FALSE(exception_state.HadException());
Persistent<BytesConsumer> consumer =
MakeGarbageCollected<ReadableStreamBytesConsumer>(script_state, reader);
Persistent<MockClient> client = MockClient::Create();
consumer->SetClient(client);
Checkpoint checkpoint;
InSequence s;
EXPECT_CALL(checkpoint, Call(1));
EXPECT_CALL(checkpoint, Call(2));
EXPECT_CALL(checkpoint, Call(3));
EXPECT_CALL(*client, OnStateChange());
EXPECT_CALL(checkpoint, Call(4));
const char* buffer = nullptr;
size_t available = 0;
checkpoint.Call(1);
test::RunPendingTasks();
checkpoint.Call(2);
EXPECT_EQ(PublicState::kReadableOrWaiting, consumer->GetPublicState());
EXPECT_EQ(Result::kShouldWait, consumer->BeginRead(&buffer, &available));
checkpoint.Call(3);
test::RunPendingTasks();
checkpoint.Call(4);
EXPECT_EQ(PublicState::kErrored, consumer->GetPublicState());
EXPECT_EQ(Result::kError, consumer->BeginRead(&buffer, &available));
}
} // namespace
} // namespace blink