Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

COIOS-802: Identify native redirect flow (v5) #1879

Open
wants to merge 23 commits into
base: develop
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
ceae25f
Add NativeRedirectAction
nauaros Oct 15, 2024
9bd40ca
Update unit tests
nauaros Oct 25, 2024
39675e8
Test native redirect flow when nativeRedirectData is nil
nauaros Oct 25, 2024
7efd84d
Merge branch 'develop' into COIOS-802_identify_native_redirect_flow
goergisn Oct 25, 2024
d574e11
Flag internallty nativeRedirect flow
nauaros Oct 28, 2024
a8f3f1c
Merge remote-tracking branch 'refs/remotes/origin/COIOS-802_identify_…
nauaros Oct 28, 2024
d339c28
Update unit tests
nauaros Oct 28, 2024
c5756a8
Merge branch 'develop' into COIOS-802_identify_native_redirect_flow
nauaros Oct 29, 2024
1a3c809
Expose redirect type
nauaros Oct 29, 2024
5c64d5c
Make RedirectAction value type
nauaros Oct 29, 2024
e6a8a46
Update documentation
nauaros Oct 29, 2024
ad894f8
Update documentation
nauaros Oct 29, 2024
9b827b6
Merge branch 'develop' into COIOS-802_identify_native_redirect_flow
nauaros Oct 29, 2024
68a0bb0
Refactor RedirectType decoding
nauaros Oct 31, 2024
ae3ed8d
Merge branch 'develop' into COIOS-802_identify_native_redirect_flow
nauaros Oct 31, 2024
8f36bd1
Explicitly indicate decoding value
nauaros Oct 31, 2024
63c1355
Make redirect type private
nauaros Nov 1, 2024
3d9f471
Change redirect type to be internal
nauaros Nov 4, 2024
896e151
Merge branch 'develop' into COIOS-802_identify_native_redirect_flow
goergisn Nov 14, 2024
10e1eab
Merge branch 'develop' into COIOS-802_identify_native_redirect_flow
goergisn Nov 15, 2024
6721b79
Merge branch 'develop' into COIOS-802_identify_native_redirect_flow
goergisn Nov 25, 2024
7f206e2
Merge branch 'develop' into COIOS-802_identify_native_redirect_flow
nauaros Jan 14, 2025
3f5babf
Add paymentMethodType to decoder initializer
nauaros Jan 14, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions Adyen.xcodeproj/project.pbxproj
Original file line number Diff line number Diff line change
Expand Up @@ -497,6 +497,7 @@
C9454C37276A340B0086C218 /* BACSDirectDebitPresentationDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = C9454C35276A33A00086C218 /* BACSDirectDebitPresentationDelegate.swift */; };
C9454C38276A34150086C218 /* BACSDirectDebitPresentationDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = C9454C35276A33A00086C218 /* BACSDirectDebitPresentationDelegate.swift */; };
C94632BE27BA6985003DD81F /* AnalyticsProvider.swift in Sources */ = {isa = PBXBuildFile; fileRef = C94632BD27BA6985003DD81F /* AnalyticsProvider.swift */; };
C958E9AD2CCBC63B005F2C69 /* NativeRedirectAction.swift in Sources */ = {isa = PBXBuildFile; fileRef = C958E9AC2CCBC63B005F2C69 /* NativeRedirectAction.swift */; };
C95903DE275A48D000E7D3BC /* BACSDirectDebitComponentTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = C95903DD275A48D000E7D3BC /* BACSDirectDebitComponentTests.swift */; };
C96688BF26A6FC1C00DC7297 /* AffirmComponentTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = C96688BE26A6FC1C00DC7297 /* AffirmComponentTests.swift */; };
C96E07A3283B92E300345732 /* BACSDirectDebitComponentTrackerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = C96E07A1283B92D500345732 /* BACSDirectDebitComponentTrackerTests.swift */; };
Expand Down Expand Up @@ -1808,6 +1809,7 @@
C93B01B82760B06300D311A1 /* BACSConfirmationPresenter.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BACSConfirmationPresenter.swift; sourceTree = "<group>"; };
C9454C35276A33A00086C218 /* BACSDirectDebitPresentationDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BACSDirectDebitPresentationDelegate.swift; sourceTree = "<group>"; };
C94632BD27BA6985003DD81F /* AnalyticsProvider.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AnalyticsProvider.swift; sourceTree = "<group>"; };
C958E9AC2CCBC63B005F2C69 /* NativeRedirectAction.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NativeRedirectAction.swift; sourceTree = "<group>"; };
C95903DD275A48D000E7D3BC /* BACSDirectDebitComponentTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BACSDirectDebitComponentTests.swift; sourceTree = "<group>"; };
C95C89312BF63A3500C47296 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; };
C96688BE26A6FC1C00DC7297 /* AffirmComponentTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AffirmComponentTests.swift; sourceTree = "<group>"; };
Expand Down Expand Up @@ -3744,6 +3746,7 @@
5A988B7A2653F1750007F4C0 /* BoletoVoucherAction.swift */,
F99D2F6F2664EDC900BB5B2F /* AnyVoucherAction.swift */,
A02AF3E9275A3A5100E1636C /* DocumentAction.swift */,
C958E9AC2CCBC63B005F2C69 /* NativeRedirectAction.swift */,
);
path = Actions;
sourceTree = "<group>";
Expand Down Expand Up @@ -7498,6 +7501,7 @@
F96757C327CF909900A16FB6 /* AnyWeChatPaySDKActionComponent.swift in Sources */,
F9175FD22594999600D653BE /* RedirectComponent.swift in Sources */,
00165FE02C05DA8600347399 /* RedireactableAwaitAction.swift in Sources */,
C958E9AD2CCBC63B005F2C69 /* NativeRedirectAction.swift in Sources */,
F9237D3D28CB470E004F9929 /* ThreeDS2ClassicActionHandler+Initializers.swift in Sources */,
81E4231B2C74DC3600349A5E /* String+QRCode.swift in Sources */,
8182AAB72B95D3560087568E /* Twint+Injectable.swift in Sources */,
Expand Down
6 changes: 4 additions & 2 deletions AdyenActions/Actions/Action.swift
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ public enum Action: Decodable {

/// Indicates the user should be redirected to a URL.
case redirect(RedirectAction)

/// Indicates the user should be redirected to an SDK.
case sdk(SDKAction)

Expand Down Expand Up @@ -47,8 +47,10 @@ public enum Action: Decodable {
let type = try container.decode(ActionType.self, forKey: .type)

switch type {
case .redirect, .nativeRedirect:
case .redirect:
self = try .redirect(RedirectAction(from: decoder))
case .nativeRedirect:
self = try .redirect(NativeRedirectAction(from: decoder))
case .threeDS2Fingerprint:
self = try .threeDS2Fingerprint(ThreeDS2FingerprintAction(from: decoder))
case .threeDS2Challenge:
Expand Down
41 changes: 41 additions & 0 deletions AdyenActions/Actions/NativeRedirectAction.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
//
// Copyright (c) 2024 Adyen N.V.
//
// This file is open source and available under the MIT license. See the LICENSE file for more info.
//

import Foundation

/// Describes an action in which the user is redirected to an app.
public class NativeRedirectAction: RedirectAction {

/// Native redirect data.
public let nativeRedirectData: String?

/// Initializes a native redirect action.
///
/// - Parameters:
/// - url: The URL to which to redirect the user.
/// - paymentData: The server-generated payment data that should be submitted to the `/payments/details` endpoint.
/// - nativeRedirectData: Native redirect data.
public init(url: URL, paymentData: String?, nativeRedirectData: String? = nil) {
self.nativeRedirectData = nativeRedirectData
super.init(url: url, paymentData: paymentData)
}

public required init(from decoder: any Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
let url = try container.decode(URL.self, forKey: CodingKeys.url)
let paymentData = try container.decodeIfPresent(String.self, forKey: CodingKeys.paymentData)
self.nativeRedirectData = try container.decodeIfPresent(String.self, forKey: CodingKeys.nativeRedirectData)
super.init(url: url, paymentData: paymentData)
}

// MARK: - CodingKeys

private enum CodingKeys: CodingKey {
case url
case paymentData
case nativeRedirectData
}
}
15 changes: 5 additions & 10 deletions AdyenActions/Actions/RedirectAction.swift
Original file line number Diff line number Diff line change
Expand Up @@ -7,26 +7,21 @@
import Foundation

/// Describes an action in which the user is redirected to a URL.
public struct RedirectAction: Decodable {
public class RedirectAction: Decodable {
nauaros marked this conversation as resolved.
Show resolved Hide resolved

/// The URL to which to redirect the user.
public let url: URL

/// The server-generated payment data that should be submitted to the `/payments/details` endpoint.
public let paymentData: String?

/// Native redirect data.
public let nativeRedirectData: String?
nauaros marked this conversation as resolved.
Show resolved Hide resolved


/// Initializes a redirect action.
///
/// - Parameters:
/// - url: The URL to which to redirect the user.
/// - paymentData: The server-generated payment data that should be submitted to the `/payments/details` endpoint.
/// - nativeRedirectData: Native redirect data.
public init(url: URL, paymentData: String?, nativeRedirectData: String? = nil) {
public init(url: URL, paymentData: String?) {
self.url = url
self.paymentData = paymentData
self.nativeRedirectData = nativeRedirectData
}
}
11 changes: 8 additions & 3 deletions AdyenActions/Components/Redirect/RedirectComponent.swift
Original file line number Diff line number Diff line change
Expand Up @@ -172,14 +172,19 @@ public final class RedirectComponent: ActionComponent {
}

private func didOpen(url returnURL: URL, _ action: RedirectAction) throws {
if let redirectStateData = action.nativeRedirectData {
try handleNativeMobileRedirect(withReturnURL: returnURL, redirectStateData: redirectStateData, action)
if let nativeRedirectAction = action as? NativeRedirectAction {
let redirectStateData = nativeRedirectAction.nativeRedirectData
try handleNativeMobileRedirect(
withReturnURL: returnURL,
redirectStateData: redirectStateData,
action
)
} else {
try notifyDelegateDidProvide(redirectDetails: RedirectDetails(returnURL: returnURL), action)
}
}

private func handleNativeMobileRedirect(withReturnURL returnURL: URL, redirectStateData: String, _ action: RedirectAction) throws {
private func handleNativeMobileRedirect(withReturnURL returnURL: URL, redirectStateData: String?, _ action: RedirectAction) throws {
guard let queryString = returnURL.query else {
throw Error.invalidRedirectParameters
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -283,7 +283,7 @@ class RedirectComponentTests: XCTestCase {
}
delegate.onDidFail = { _, _ in XCTFail("Should not call onDidFail") }

let action = RedirectAction(url: URL(string: "https://google.com")!, paymentData: nil, nativeRedirectData: "test_nativeRedirectData")
let action = NativeRedirectAction(url: URL(string: "https://google.com")!, paymentData: nil, nativeRedirectData: "test_nativeRedirectData")
sut.handle(action)
XCTAssertTrue(RedirectComponent.applicationDidOpen(from: URL(string: "url://?queryParam=value")!))

Expand Down Expand Up @@ -311,7 +311,7 @@ class RedirectComponentTests: XCTestCase {
XCTFail("Should not call onDidProvide")
}

let action = RedirectAction(url: URL(string: "https://google.com")!, paymentData: nil, nativeRedirectData: "test_nativeRedirectData")
let action = NativeRedirectAction(url: URL(string: "https://google.com")!, paymentData: nil, nativeRedirectData: "test_nativeRedirectData")
sut.handle(action)
XCTAssertFalse(RedirectComponent.applicationDidOpen(from: URL(string: "url://")!))

Expand Down Expand Up @@ -343,10 +343,44 @@ class RedirectComponentTests: XCTestCase {
redirectExpectation.fulfill()
}

let action = RedirectAction(url: URL(string: "https://google.com")!, paymentData: nil, nativeRedirectData: "test_nativeRedirectData")
let action = NativeRedirectAction(url: URL(string: "https://google.com")!, paymentData: nil, nativeRedirectData: "test_nativeRedirectData")
sut.handle(action)
XCTAssertTrue(RedirectComponent.applicationDidOpen(from: URL(string: "url://?queryParam=value")!))

waitForExpectations(timeout: 10)
}

func testNativeRedirectWithNativeRedirectDataNilShouldPerformNativeRedirectResultRequest() {
// Given
let apiClient = APIClientMock()
let sut = RedirectComponent(context: Dummy.context, apiClient: apiClient.retryAPIClient(with: SimpleScheduler(maximumCount: 2)))
apiClient.mockedResults = [.success(try! RedirectDetails(returnURL: URL(string: "url://?redirectResult=test_redirectResult")!))]

let appLauncher = AppLauncherMock()
sut.appLauncher = appLauncher
let appLauncherExpectation = expectation(description: "Expect appLauncher.openUniversalAppUrl() to be called")
appLauncher.onOpenUniversalAppUrl = { url, completion in
XCTAssertEqual(url, URL(string: "https://google.com")!)
completion?(true)
appLauncherExpectation.fulfill()
}

let delegate = ActionComponentDelegateMock()
sut.delegate = delegate
let redirectExpectation = expectation(description: "Expect redirect to be proccessed")
delegate.onDidProvide = { data, component in
XCTAssertTrue(component === sut)
XCTAssertNotNil(data.details)
redirectExpectation.fulfill()
}
delegate.onDidFail = { _, _ in XCTFail("Should not call onDidFail") }

// When
let action = NativeRedirectAction(url: URL(string: "https://google.com")!, paymentData: nil, nativeRedirectData: nil)
sut.handle(action)

// Then
XCTAssertTrue(RedirectComponent.applicationDidOpen(from: URL(string: "url://?queryParam=value")!))
waitForExpectations(timeout: 10)
}
}
Loading