| // 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/modules/payments/payment_request_update_event.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/script_promise_resolver.h" |
| #include "third_party/blink/renderer/bindings/core/v8/v8_binding_for_testing.h" |
| #include "third_party/blink/renderer/core/event_type_names.h" |
| #include "third_party/blink/renderer/modules/payments/payment_request.h" |
| #include "third_party/blink/renderer/modules/payments/payment_test_helper.h" |
| #include "third_party/blink/renderer/modules/payments/payment_updater.h" |
| #include "third_party/blink/renderer/platform/bindings/exception_state.h" |
| #include "third_party/blink/renderer/platform/bindings/script_state.h" |
| |
| namespace blink { |
| namespace { |
| |
| class MockPaymentUpdater : public GarbageCollectedFinalized<MockPaymentUpdater>, |
| public PaymentUpdater { |
| USING_GARBAGE_COLLECTED_MIXIN(MockPaymentUpdater); |
| WTF_MAKE_NONCOPYABLE(MockPaymentUpdater); |
| |
| public: |
| MockPaymentUpdater() = default; |
| ~MockPaymentUpdater() override = default; |
| |
| MOCK_METHOD2(OnUpdatePaymentDetails, |
| void(const AtomicString& event_type, |
| const ScriptValue& detailsScriptValue)); |
| MOCK_METHOD1(OnUpdatePaymentDetailsFailure, void(const String& error)); |
| |
| void Trace(blink::Visitor* visitor) override {} |
| }; |
| |
| TEST(PaymentRequestUpdateEventTest, OnUpdatePaymentDetailsCalled) { |
| V8TestingScope scope; |
| PaymentRequestUpdateEvent* event = PaymentRequestUpdateEvent::Create( |
| scope.GetExecutionContext(), event_type_names::kShippingaddresschange); |
| MockPaymentUpdater* updater = MakeGarbageCollected<MockPaymentUpdater>(); |
| event->SetTrusted(true); |
| event->SetPaymentDetailsUpdater(updater); |
| event->SetEventPhase(Event::kCapturingPhase); |
| ScriptPromiseResolver* payment_details = |
| ScriptPromiseResolver::Create(scope.GetScriptState()); |
| event->updateWith(scope.GetScriptState(), payment_details->Promise(), |
| scope.GetExceptionState()); |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| |
| EXPECT_CALL(*updater, |
| OnUpdatePaymentDetails(event_type_names::kShippingaddresschange, |
| testing::_)); |
| EXPECT_CALL(*updater, OnUpdatePaymentDetailsFailure(testing::_)).Times(0); |
| |
| payment_details->Resolve("foo"); |
| } |
| |
| TEST(PaymentRequestUpdateEventTest, OnUpdatePaymentDetailsFailureCalled) { |
| V8TestingScope scope; |
| PaymentRequestUpdateEvent* event = PaymentRequestUpdateEvent::Create( |
| scope.GetExecutionContext(), event_type_names::kShippingaddresschange); |
| MockPaymentUpdater* updater = MakeGarbageCollected<MockPaymentUpdater>(); |
| event->SetTrusted(true); |
| event->SetPaymentDetailsUpdater(updater); |
| event->SetEventPhase(Event::kCapturingPhase); |
| ScriptPromiseResolver* payment_details = |
| ScriptPromiseResolver::Create(scope.GetScriptState()); |
| event->updateWith(scope.GetScriptState(), payment_details->Promise(), |
| scope.GetExceptionState()); |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| |
| EXPECT_CALL(*updater, |
| OnUpdatePaymentDetails(event_type_names::kShippingaddresschange, |
| testing::_)) |
| .Times(0); |
| EXPECT_CALL(*updater, OnUpdatePaymentDetailsFailure(testing::_)); |
| |
| payment_details->Reject("oops"); |
| } |
| |
| TEST(PaymentRequestUpdateEventTest, CannotUpdateWithoutDispatching) { |
| V8TestingScope scope; |
| PaymentRequestUpdateEvent* event = PaymentRequestUpdateEvent::Create( |
| scope.GetExecutionContext(), event_type_names::kShippingaddresschange); |
| event->SetPaymentDetailsUpdater(new MockPaymentUpdater); |
| |
| event->updateWith( |
| scope.GetScriptState(), |
| ScriptPromiseResolver::Create(scope.GetScriptState())->Promise(), |
| scope.GetExceptionState()); |
| |
| EXPECT_TRUE(scope.GetExceptionState().HadException()); |
| } |
| |
| TEST(PaymentRequestUpdateEventTest, CannotUpdateTwice) { |
| V8TestingScope scope; |
| PaymentRequestUpdateEvent* event = PaymentRequestUpdateEvent::Create( |
| scope.GetExecutionContext(), event_type_names::kShippingaddresschange); |
| MockPaymentUpdater* updater = MakeGarbageCollected<MockPaymentUpdater>(); |
| event->SetTrusted(true); |
| event->SetPaymentDetailsUpdater(updater); |
| event->SetEventPhase(Event::kCapturingPhase); |
| event->updateWith( |
| scope.GetScriptState(), |
| ScriptPromiseResolver::Create(scope.GetScriptState())->Promise(), |
| scope.GetExceptionState()); |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| |
| event->updateWith( |
| scope.GetScriptState(), |
| ScriptPromiseResolver::Create(scope.GetScriptState())->Promise(), |
| scope.GetExceptionState()); |
| |
| EXPECT_TRUE(scope.GetExceptionState().HadException()); |
| } |
| |
| TEST(PaymentRequestUpdateEventTest, UpdaterNotRequired) { |
| V8TestingScope scope; |
| PaymentRequestUpdateEvent* event = PaymentRequestUpdateEvent::Create( |
| scope.GetExecutionContext(), event_type_names::kShippingaddresschange); |
| event->SetTrusted(true); |
| |
| event->updateWith( |
| scope.GetScriptState(), |
| ScriptPromiseResolver::Create(scope.GetScriptState())->Promise(), |
| scope.GetExceptionState()); |
| |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| } |
| |
| TEST(PaymentRequestUpdateEventTest, AddressChangeUpdateWithTimeout) { |
| V8TestingScope scope; |
| PaymentRequestMockFunctionScope funcs(scope.GetScriptState()); |
| MakePaymentRequestOriginSecure(scope.GetDocument()); |
| PaymentRequest* request = PaymentRequest::Create( |
| scope.GetExecutionContext(), BuildPaymentMethodDataForTest(), |
| BuildPaymentDetailsInitForTest(), scope.GetExceptionState()); |
| PaymentRequestUpdateEvent* event = PaymentRequestUpdateEvent::Create( |
| scope.GetExecutionContext(), event_type_names::kShippingaddresschange); |
| event->SetPaymentDetailsUpdater(request); |
| event->SetTrusted(true); |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| |
| String error_message; |
| request->show(scope.GetScriptState()) |
| .Then(funcs.ExpectNoCall(), funcs.ExpectCall(&error_message)); |
| |
| event->OnUpdateEventTimeoutForTesting(); |
| |
| v8::MicrotasksScope::PerformCheckpoint(scope.GetScriptState()->GetIsolate()); |
| EXPECT_EQ( |
| "AbortError: Timed out waiting for a response to a " |
| "'shippingaddresschange' event", |
| error_message); |
| |
| event->updateWith( |
| scope.GetScriptState(), |
| ScriptPromiseResolver::Create(scope.GetScriptState())->Promise(), |
| scope.GetExceptionState()); |
| |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| } |
| |
| TEST(PaymentRequestUpdateEventTest, OptionChangeUpdateWithTimeout) { |
| V8TestingScope scope; |
| PaymentRequestMockFunctionScope funcs(scope.GetScriptState()); |
| MakePaymentRequestOriginSecure(scope.GetDocument()); |
| PaymentRequest* request = PaymentRequest::Create( |
| scope.GetExecutionContext(), BuildPaymentMethodDataForTest(), |
| BuildPaymentDetailsInitForTest(), scope.GetExceptionState()); |
| PaymentRequestUpdateEvent* event = PaymentRequestUpdateEvent::Create( |
| scope.GetExecutionContext(), event_type_names::kShippingoptionchange); |
| event->SetTrusted(true); |
| event->SetPaymentDetailsUpdater(request); |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| |
| String error_message; |
| request->show(scope.GetScriptState()) |
| .Then(funcs.ExpectNoCall(), funcs.ExpectCall(&error_message)); |
| |
| event->OnUpdateEventTimeoutForTesting(); |
| |
| v8::MicrotasksScope::PerformCheckpoint(scope.GetScriptState()->GetIsolate()); |
| EXPECT_EQ( |
| "AbortError: Timed out waiting for a response to a " |
| "'shippingoptionchange' event", |
| error_message); |
| |
| event->updateWith( |
| scope.GetScriptState(), |
| ScriptPromiseResolver::Create(scope.GetScriptState())->Promise(), |
| scope.GetExceptionState()); |
| |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| } |
| |
| TEST(PaymentRequestUpdateEventTest, AddressChangePromiseTimeout) { |
| V8TestingScope scope; |
| PaymentRequestMockFunctionScope funcs(scope.GetScriptState()); |
| MakePaymentRequestOriginSecure(scope.GetDocument()); |
| PaymentRequest* request = PaymentRequest::Create( |
| scope.GetExecutionContext(), BuildPaymentMethodDataForTest(), |
| BuildPaymentDetailsInitForTest(), scope.GetExceptionState()); |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| PaymentRequestUpdateEvent* event = PaymentRequestUpdateEvent::Create( |
| scope.GetExecutionContext(), event_type_names::kShippingaddresschange); |
| event->SetTrusted(true); |
| event->SetPaymentDetailsUpdater(request); |
| event->SetEventPhase(Event::kCapturingPhase); |
| ScriptPromiseResolver* payment_details = |
| ScriptPromiseResolver::Create(scope.GetScriptState()); |
| String error_message; |
| request->show(scope.GetScriptState()) |
| .Then(funcs.ExpectNoCall(), funcs.ExpectCall(&error_message)); |
| event->updateWith(scope.GetScriptState(), payment_details->Promise(), |
| scope.GetExceptionState()); |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| |
| event->OnUpdateEventTimeoutForTesting(); |
| |
| v8::MicrotasksScope::PerformCheckpoint(scope.GetScriptState()->GetIsolate()); |
| EXPECT_EQ( |
| "AbortError: Timed out waiting for a response to a " |
| "'shippingaddresschange' event", |
| error_message); |
| |
| payment_details->Resolve("foo"); |
| } |
| |
| TEST(PaymentRequestUpdateEventTest, OptionChangePromiseTimeout) { |
| V8TestingScope scope; |
| PaymentRequestMockFunctionScope funcs(scope.GetScriptState()); |
| MakePaymentRequestOriginSecure(scope.GetDocument()); |
| PaymentRequest* request = PaymentRequest::Create( |
| scope.GetExecutionContext(), BuildPaymentMethodDataForTest(), |
| BuildPaymentDetailsInitForTest(), scope.GetExceptionState()); |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| PaymentRequestUpdateEvent* event = PaymentRequestUpdateEvent::Create( |
| scope.GetExecutionContext(), event_type_names::kShippingoptionchange); |
| event->SetTrusted(true); |
| event->SetPaymentDetailsUpdater(request); |
| event->SetEventPhase(Event::kCapturingPhase); |
| ScriptPromiseResolver* payment_details = |
| ScriptPromiseResolver::Create(scope.GetScriptState()); |
| String error_message; |
| request->show(scope.GetScriptState()) |
| .Then(funcs.ExpectNoCall(), funcs.ExpectCall(&error_message)); |
| event->updateWith(scope.GetScriptState(), payment_details->Promise(), |
| scope.GetExceptionState()); |
| EXPECT_FALSE(scope.GetExceptionState().HadException()); |
| |
| event->OnUpdateEventTimeoutForTesting(); |
| |
| v8::MicrotasksScope::PerformCheckpoint(scope.GetScriptState()->GetIsolate()); |
| EXPECT_EQ( |
| "AbortError: Timed out waiting for a response to a " |
| "'shippingoptionchange' event", |
| error_message); |
| |
| payment_details->Resolve("foo"); |
| } |
| |
| TEST(PaymentRequestUpdateEventTest, NotAllowUntrustedEvent) { |
| V8TestingScope scope; |
| PaymentRequestUpdateEvent* event = PaymentRequestUpdateEvent::Create( |
| scope.GetExecutionContext(), event_type_names::kShippingaddresschange); |
| event->SetTrusted(false); |
| |
| event->updateWith( |
| scope.GetScriptState(), |
| ScriptPromiseResolver::Create(scope.GetScriptState())->Promise(), |
| scope.GetExceptionState()); |
| |
| EXPECT_TRUE(scope.GetExceptionState().HadException()); |
| } |
| |
| } // namespace |
| } // namespace blink |