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

Merged
merged 26 commits into from
Jan 22, 2025
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
Show all changes
26 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
8528392
Remove decoder initializer
nauaros Jan 15, 2025
1271dbd
Merge branch 'develop' into COIOS-802_identify_native_redirect_flow
nauaros Jan 21, 2025
ec3a9de
Remove enum string values. Not needed
nauaros Jan 21, 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
2 changes: 1 addition & 1 deletion 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
43 changes: 37 additions & 6 deletions AdyenActions/Actions/RedirectAction.swift
Original file line number Diff line number Diff line change
Expand Up @@ -7,26 +7,57 @@
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

private enum Constants {
static let nativeRedirectType = "nativeRedirect"
}

/// 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


internal let isNaviteRedirect: Bool

/// 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?,
nativeRedirectData: String? = nil
) {
self.url = url
self.paymentData = paymentData
self.nativeRedirectData = nativeRedirectData
self.isNaviteRedirect = nativeRedirectData != nil

}

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

let redirectType = try container.decode(String.self, forKey: .type)
self.isNaviteRedirect = redirectType == Constants.nativeRedirectType
}

// MARK: - Private

private enum CodingKeys: CodingKey {
case url
case paymentData
case nativeRedirectData
case type
}
}
10 changes: 7 additions & 3 deletions AdyenActions/Components/Redirect/RedirectComponent.swift
Original file line number Diff line number Diff line change
Expand Up @@ -172,14 +172,18 @@ 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 action.isNaviteRedirect {
try handleNativeMobileRedirect(
withReturnURL: returnURL,
redirectStateData: action.nativeRedirectData,
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)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ class ThreeDS2FingerprintSubmitterTests: XCTestCase {
let apiClient = APIClientMock()
let sut = ThreeDS2FingerprintSubmitter(apiContext: Dummy.apiContext, apiClient: apiClient)

let mockedRedirectAction = RedirectAction(url: URL(string: "https://www.adyen.com")!, paymentData: "data")
let mockedRedirectAction = RedirectAction(url: URL(string: "https://www.adyen.com")!, paymentData: "data", nativeRedirectData: <#String?#>)
let mockedAction = Action.redirect(mockedRedirectAction)
let mockedResponse = Submit3DS2FingerprintResponse(result: .action(mockedAction))
apiClient.mockedResults = [.success(mockedResponse)]
Expand Down
Loading