blob: 9b4d9b9f5828636fee7ae965ad5dfa3917f8ff40 [file] [log] [blame]
// Copyright 2019 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/streams/stream_promise_resolver.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/blink/renderer/bindings/core/v8/script_promise.h"
#include "third_party/blink/renderer/bindings/core/v8/v8_binding_for_testing.h"
#include "third_party/blink/renderer/core/streams/promise_handler.h"
#include "third_party/blink/renderer/platform/bindings/v8_binding.h"
#include "third_party/blink/renderer/platform/heap/heap.h"
#include "third_party/blink/renderer/platform/heap/visitor.h"
namespace blink {
namespace {
TEST(StreamPromiseResolverTest, Construct) {
V8TestingScope scope;
auto* promise =
MakeGarbageCollected<StreamPromiseResolver>(scope.GetScriptState());
EXPECT_TRUE(promise->V8Promise(scope.GetIsolate())->IsPromise());
EXPECT_FALSE(promise->IsSettled());
}
TEST(StreamPromiseResolverTest, Resolve) {
V8TestingScope scope;
auto* isolate = scope.GetIsolate();
auto* promise =
MakeGarbageCollected<StreamPromiseResolver>(scope.GetScriptState());
promise->Resolve(scope.GetScriptState(), v8::Null(isolate));
ASSERT_EQ(promise->State(isolate), v8::Promise::kFulfilled);
EXPECT_TRUE(promise->V8Promise(isolate)->Result()->IsNull());
EXPECT_TRUE(promise->IsSettled());
}
TEST(StreamPromiseResolverTest, ResolveWithUndefined) {
V8TestingScope scope;
auto* isolate = scope.GetIsolate();
auto* promise =
MakeGarbageCollected<StreamPromiseResolver>(scope.GetScriptState());
promise->ResolveWithUndefined(scope.GetScriptState());
ASSERT_EQ(promise->State(isolate), v8::Promise::kFulfilled);
EXPECT_TRUE(promise->V8Promise(isolate)->Result()->IsUndefined());
EXPECT_TRUE(promise->IsSettled());
}
TEST(StreamPromiseResolverTest, Reject) {
V8TestingScope scope;
auto* isolate = scope.GetIsolate();
auto* promise =
MakeGarbageCollected<StreamPromiseResolver>(scope.GetScriptState());
promise->Reject(scope.GetScriptState(), v8::Number::New(isolate, 2));
ASSERT_EQ(promise->State(isolate), v8::Promise::kRejected);
auto result = promise->V8Promise(isolate)->Result();
ASSERT_TRUE(result->IsNumber());
EXPECT_EQ(result.As<v8::Number>()->Value(), 2.0);
EXPECT_TRUE(promise->IsSettled());
}
TEST(StreamPromiseResolverTest, RejectDoesNothingAfterResolve) {
V8TestingScope scope;
auto* isolate = scope.GetIsolate();
auto* promise =
MakeGarbageCollected<StreamPromiseResolver>(scope.GetScriptState());
promise->Resolve(scope.GetScriptState(), v8::Undefined(isolate));
promise->Reject(scope.GetScriptState(), v8::Null(isolate));
ASSERT_EQ(promise->State(isolate), v8::Promise::kFulfilled);
EXPECT_TRUE(promise->V8Promise(isolate)->Result()->IsUndefined());
}
TEST(StreamPromiseResolverTest, ResolveDoesNothingAfterReject) {
V8TestingScope scope;
auto* isolate = scope.GetIsolate();
auto* promise =
MakeGarbageCollected<StreamPromiseResolver>(scope.GetScriptState());
promise->Reject(scope.GetScriptState(), v8::Null(isolate));
promise->Resolve(scope.GetScriptState(), v8::Undefined(isolate));
ASSERT_EQ(promise->State(isolate), v8::Promise::kRejected);
EXPECT_TRUE(promise->V8Promise(isolate)->Result()->IsNull());
}
TEST(StreamPromiseResolverTest, ResolveDoesNothingInsideResolve) {
V8TestingScope scope;
auto* isolate = scope.GetIsolate();
auto* promise =
MakeGarbageCollected<StreamPromiseResolver>(scope.GetScriptState());
// Create an object equivalent to
// value = {
// get then() {
// resolvePromise(promise, undefined);
// runMicrotasks();
// }
// }
class ThenGetter final : public ScriptFunction {
public:
static v8::Local<v8::Function> Create(ScriptState* script_state,
StreamPromiseResolver* promise) {
return MakeGarbageCollected<ThenGetter>(script_state, promise)
->BindToV8Function();
}
ThenGetter(ScriptState* script_state, StreamPromiseResolver* promise)
: ScriptFunction(script_state), promise_(promise) {}
void Trace(Visitor* visitor) const override {
visitor->Trace(promise_);
ScriptFunction::Trace(visitor);
}
private:
void CallRaw(const v8::FunctionCallbackInfo<v8::Value>&) override {
auto* isolate = GetScriptState()->GetIsolate();
EXPECT_TRUE(promise_->IsSettled());
promise_->Resolve(GetScriptState(), v8::Undefined(isolate));
v8::MicrotasksScope::PerformCheckpoint(isolate);
}
Member<StreamPromiseResolver> promise_;
};
auto value = v8::Object::New(isolate);
v8::PropertyDescriptor property_descriptor(
ThenGetter::Create(scope.GetScriptState(), promise),
v8::Undefined(isolate));
const auto then = V8String(isolate, "then");
value->DefineProperty(scope.GetContext(), then, property_descriptor).Check();
// Resolving with |value| will call the "then" getter synchronously.
promise->Resolve(scope.GetScriptState(), value);
ASSERT_EQ(promise->State(isolate), v8::Promise::kFulfilled);
EXPECT_EQ(promise->V8Promise(isolate)->Result(), value);
}
TEST(StreamPromiseResolverTest, GetScriptPromise) {
V8TestingScope scope;
auto* promise =
MakeGarbageCollected<StreamPromiseResolver>(scope.GetScriptState());
ScriptPromise script_promise =
promise->GetScriptPromise(scope.GetScriptState());
EXPECT_FALSE(script_promise.IsEmpty());
}
TEST(StreamPromiseResolverTest, MarkAsHandled) {
V8TestingScope scope;
auto* promise =
MakeGarbageCollected<StreamPromiseResolver>(scope.GetScriptState());
auto* isolate = scope.GetIsolate();
promise->MarkAsHandled(isolate);
EXPECT_TRUE(promise->V8Promise(isolate)->HasHandler());
}
TEST(StreamPromiseResolverTest, CreateResolved) {
V8TestingScope scope;
auto* isolate = scope.GetIsolate();
auto* promise = StreamPromiseResolver::CreateResolved(scope.GetScriptState(),
v8::Null(isolate));
ASSERT_EQ(promise->State(isolate), v8::Promise::kFulfilled);
EXPECT_TRUE(promise->V8Promise(isolate)->Result()->IsNull());
EXPECT_TRUE(promise->IsSettled());
}
TEST(StreamPromiseResolverTest, CreateResolvedWithUndefined) {
V8TestingScope scope;
auto* isolate = scope.GetIsolate();
auto* promise = StreamPromiseResolver::CreateResolvedWithUndefined(
scope.GetScriptState());
ASSERT_EQ(promise->State(isolate), v8::Promise::kFulfilled);
EXPECT_TRUE(promise->V8Promise(isolate)->Result()->IsUndefined());
EXPECT_TRUE(promise->IsSettled());
}
TEST(StreamPromiseResolverTest, CreateRejected) {
V8TestingScope scope;
auto* isolate = scope.GetIsolate();
auto* promise = StreamPromiseResolver::CreateRejected(scope.GetScriptState(),
v8::Null(isolate));
ASSERT_EQ(promise->State(isolate), v8::Promise::kRejected);
EXPECT_TRUE(promise->V8Promise(isolate)->Result()->IsNull());
EXPECT_TRUE(promise->IsSettled());
}
} // namespace
} // namespace blink