Skip to content

Commit

Permalink
feat: don't emit sessionTimeout once result.dematerialize is called (#…
Browse files Browse the repository at this point in the history
…149)

* feat: don't emit sessionTimeout once result.dematerialize is called

* chore: bump to 2.7.0
  • Loading branch information
CAMOBAP authored Apr 18, 2024
1 parent c585511 commit 76dbfff
Show file tree
Hide file tree
Showing 9 changed files with 140 additions and 54 deletions.
5 changes: 5 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,8 @@
# 2.7.0

- Fix: don't emit `.sessionTimeout` error once `HCaptchaResult.dematerialize` called ([#129](https://github.com/hCaptcha/HCaptcha-ios-sdk/issues/129))
- Fix: keep `.sessionTimeout` as only retriable error to avoid stack overflow

# 2.6.3

- Feature: PrivacyInfo.xcprivacy was added ([#128](https://github.com/hCaptcha/HCaptcha-ios-sdk/issues/128))
Expand Down
6 changes: 4 additions & 2 deletions Example/HCaptcha_Tests/Core/HCaptchaResult__Tests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,8 @@ import XCTest
class HCaptchaResult__Tests: XCTestCase {
func test__Get_Token() {
let token = UUID().uuidString
let result = HCaptchaResult(token: token)
let manager = HCaptchaWebViewManager()
let result = HCaptchaResult(manager, token: token)

do {
let value = try result.dematerialize()
Expand All @@ -26,7 +27,8 @@ class HCaptchaResult__Tests: XCTestCase {

func test__Get_Token__Error() {
let error = HCaptchaError.random()
let result = HCaptchaResult(error: error)
let manager = HCaptchaWebViewManager()
let result = HCaptchaResult(manager, error: error)

do {
_ = try result.dematerialize()
Expand Down
50 changes: 47 additions & 3 deletions Example/HCaptcha_Tests/Core/HCaptchaWebViewManager__Tests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -286,6 +286,41 @@ class HCaptchaWebViewManager__Tests: XCTestCase {
waitForExpectations(timeout: 10)
}

func test__Reset_After_Stop() {
let exp0 = expectation(description: "stop loading")
let exp1 = expectation(description: "configureWebView called")
let exp2 = expectation(description: "token recieved")

// Stop
let manager = HCaptchaWebViewManager(messageBody: "{token: \"some_token\"}")
manager.stop()
manager.configureWebView { _ in
XCTFail("should not ask to configure the webview")
}

manager.validate(on: presenterView) { _ in
XCTFail("should not validate")
}

DispatchQueue.main.asyncAfter(deadline: .now() + 2) {
exp0.fulfill()
}

manager.reset()

manager.configureWebView { _ in
exp1.fulfill()
}

manager.validate(on: presenterView) { result in
let token = try? result.dematerialize()
XCTAssertEqual("some_token", token)
exp2.fulfill()
}

waitForExpectations(timeout: 10)
}

// MARK: Setup

func test__Key_Setup() {
Expand Down Expand Up @@ -383,6 +418,7 @@ class HCaptchaWebViewManager__Tests: XCTestCase {
var exp0Count = 0
let exp1 = expectation(description: "should call onEvent")
let exp2 = expectation(description: "fail on first execution")
let exp3 = expectation(description: "hcaptcha opened")
var result: HCaptchaResult?

// Validate
Expand All @@ -395,9 +431,17 @@ class HCaptchaWebViewManager__Tests: XCTestCase {
}

manager.onEvent = { (event, error) in
XCTAssertEqual(.error, event)
XCTAssertEqual(HCaptchaError.sessionTimeout, error as? HCaptchaError)
exp1.fulfill()
XCTAssertTrue([.error, .open].contains(event))
switch event {
case .error:
XCTAssertEqual(.error, event)
XCTAssertEqual(HCaptchaError.sessionTimeout, error as? HCaptchaError)
exp1.fulfill()
case .open:
exp3.fulfill()
default:
XCTFail("Unexpected event \(event)")
}
}

// Error
Expand Down
6 changes: 3 additions & 3 deletions Example/Podfile.lock
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
PODS:
- AppSwizzle (1.3.1)
- HCaptcha/Core (2.6.3)
- HCaptcha/RxSwift (2.6.3):
- HCaptcha/Core (2.7.0)
- HCaptcha/RxSwift (2.7.0):
- HCaptcha/Core
- RxSwift (~> 6.2.0)
- RxBlocking (6.2.0):
Expand Down Expand Up @@ -37,7 +37,7 @@ EXTERNAL SOURCES:

SPEC CHECKSUMS:
AppSwizzle: db36e436f56110d93e5ae0147683435df593cabc
HCaptcha: 2f61ea26717a21d17ef4b323eb3248cce80be648
HCaptcha: 0e52ee5303201606c66ce141444ca8c8083013d5
RxBlocking: 0b29f7d2079109a8de49c411381bed7c33ef1eeb
RxCocoa: 4baf94bb35f2c0ab31bc0cb9f1900155f646ba42
RxRelay: e72dbfd157807478401ef1982e1c61c945c94b2f
Expand Down
4 changes: 4 additions & 0 deletions HCaptcha-Carthage.xcodeproj/project.pbxproj
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
/* Begin PBXBuildFile section */
1F833B7F271DC69C00E4DAB2 /* RxSwift.xcframework in Frameworks */ = {isa = PBXBuildFile; fileRef = 1F833B7E271DC69C00E4DAB2 /* RxSwift.xcframework */; };
1F833B80271DC69C00E4DAB2 /* RxSwift.xcframework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 1F833B7E271DC69C00E4DAB2 /* RxSwift.xcframework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; };
E614B37B2BCEBF3400CEA791 /* HCaptchaWebViewManager+WKNavigationDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = E614B37A2BCEBF3400CEA791 /* HCaptchaWebViewManager+WKNavigationDelegate.swift */; };
E634944A2828856300130AC5 /* HCaptchaEvent.swift in Sources */ = {isa = PBXBuildFile; fileRef = E63494492828856300130AC5 /* HCaptchaEvent.swift */; };
E64C60082B7DFBE900D203F4 /* PrivacyInfo.xcprivacy in Resources */ = {isa = PBXBuildFile; fileRef = E64C60072B7DFBE900D203F4 /* PrivacyInfo.xcprivacy */; };
E65145E327786BDB0079668A /* HCaptchaConfig.swift in Sources */ = {isa = PBXBuildFile; fileRef = E65145E227786BDB0079668A /* HCaptchaConfig.swift */; };
Expand Down Expand Up @@ -54,6 +55,7 @@

/* Begin PBXFileReference section */
1F833B7E271DC69C00E4DAB2 /* RxSwift.xcframework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.xcframework; name = RxSwift.xcframework; path = Carthage/Build/RxSwift.xcframework; sourceTree = "<group>"; };
E614B37A2BCEBF3400CEA791 /* HCaptchaWebViewManager+WKNavigationDelegate.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = "HCaptchaWebViewManager+WKNavigationDelegate.swift"; sourceTree = "<group>"; };
E63494492828856300130AC5 /* HCaptchaEvent.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = HCaptchaEvent.swift; sourceTree = "<group>"; };
E64C60072B7DFBE900D203F4 /* PrivacyInfo.xcprivacy */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xml; path = PrivacyInfo.xcprivacy; sourceTree = "<group>"; };
E65145E227786BDB0079668A /* HCaptchaConfig.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = HCaptchaConfig.swift; sourceTree = "<group>"; };
Expand Down Expand Up @@ -180,6 +182,7 @@
F24EA1D81F9683F5001DEC17 /* HCaptchaDecoder.swift */,
E6DB9EA727B15954008F0327 /* HCaptchaDebugInfo.swift */,
F24EA1D91F9683F5001DEC17 /* HCaptchaWebViewManager.swift */,
E614B37A2BCEBF3400CEA791 /* HCaptchaWebViewManager+WKNavigationDelegate.swift */,
F24EA1DA1F9683F5001DEC17 /* String+Dict.swift */,
F24EA1DB1F9683F5001DEC17 /* HCaptchaError.swift */,
F2AE8613204F3B41002E28D7 /* HCaptchaResult.swift */,
Expand Down Expand Up @@ -333,6 +336,7 @@
F24EA1E41F968403001DEC17 /* String+Dict.swift in Sources */,
F231B39A1FEC51C800F82943 /* DispatchQueue+Throttle.swift in Sources */,
E6DB9EA827B15954008F0327 /* HCaptchaDebugInfo.swift in Sources */,
E614B37B2BCEBF3400CEA791 /* HCaptchaWebViewManager+WKNavigationDelegate.swift in Sources */,
E683772129053E560021BFD7 /* HCaptchaURLOpener.swift in Sources */,
);
runOnlyForDeploymentPostprocessing = 0;
Expand Down
2 changes: 1 addition & 1 deletion HCaptcha.podspec
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
Pod::Spec.new do |s|
s.name = 'HCaptcha'
s.version = '2.6.3'
s.version = '2.7.0'
s.summary = 'HCaptcha for iOS'
s.swift_version = '5.0'

Expand Down
8 changes: 7 additions & 1 deletion HCaptcha/Classes/HCaptchaResult.swift
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,11 @@ public class HCaptchaResult: NSObject {
/// Result error
let error: HCaptchaError?

public init (token: String? = nil, error: HCaptchaError? = nil) {
/// Manager
let manager: HCaptchaWebViewManager

internal init (_ manager: HCaptchaWebViewManager, token: String? = nil, error: HCaptchaError? = nil) {
self.manager = manager
self.token = token
self.error = error
}
Expand All @@ -35,6 +39,8 @@ public class HCaptchaResult: NSObject {
*/
@objc
public func dematerialize() throws -> String {
manager.resultHandled = true

if let token = self.token {
return token
}
Expand Down
45 changes: 45 additions & 0 deletions HCaptcha/Classes/HCaptchaWebViewManager+WKNavigationDelegate.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
//
// HCaptchaWebViewManager+WKNavigationDelegate.swift
// HCaptcha
//
// Copyright © 2024 HCaptcha. All rights reserved.
//

import Foundation
import WebKit

extension HCaptchaWebViewManager: WKNavigationDelegate {
func webView(_ webView: WKWebView, decidePolicyFor navigationAction: WKNavigationAction,
decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) {
if navigationAction.targetFrame == nil, let url = navigationAction.request.url, urlOpener.canOpenURL(url) {
urlOpener.openURL(url)
}
decisionHandler(WKNavigationActionPolicy.allow)
}

/// Tells the delegate that an error occurred during navigation.
func webView(_ webView: WKWebView, didFail navigation: WKNavigation!, withError error: Error) {
Log.debug("WebViewManager.webViewDidFail with \(error)")
completion?(HCaptchaResult(self, error: .unexpected(error)))
}

/// Tells the delegate that an error occurred during the early navigation process.
func webView(_ webView: WKWebView, didFailProvisionalNavigation navigation: WKNavigation!, withError error: Error) {
Log.debug("WebViewManager.webViewDidFailProvisionalNavigation with \(error)")
completion?(HCaptchaResult(self, error: .unexpected(error)))
}

/// Tells the delegate that the web view’s content process was terminated.
func webViewWebContentProcessDidTerminate(_ webView: WKWebView) {
Log.debug("WebViewManager.webViewWebContentProcessDidTerminate")
let kHCaptchaErrorWebViewProcessDidTerminate = -1
let kHCaptchaErrorDomain = "com.hcaptcha.sdk-ios"
let error = NSError(domain: kHCaptchaErrorDomain,
code: kHCaptchaErrorWebViewProcessDidTerminate,
userInfo: [
NSLocalizedDescriptionKey: "WebView web content process did terminate",
NSLocalizedRecoverySuggestionErrorKey: "Call HCaptcha.reset()"])
completion?(HCaptchaResult(self, error: .unexpected(error)))
didFinishLoading = false
}
}
68 changes: 24 additions & 44 deletions HCaptcha/Classes/HCaptchaWebViewManager.swift
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,9 @@ internal class HCaptchaWebViewManager: NSObject {
public var shouldSkipForTests = false
#endif

/// True if validation token was dematerialized
internal var resultHandled: Bool = false

/// Sends the result message
var completion: ((HCaptchaResult) -> Void)?

Expand Down Expand Up @@ -66,7 +69,7 @@ internal class HCaptchaWebViewManager: NSObject {
fileprivate var decoder: HCaptchaDecoder!

/// Indicates if the script has already been loaded by the `webView`
fileprivate var didFinishLoading = false {
internal var didFinishLoading = false {
didSet {
if didFinishLoading {
onDidFinishLoading?()
Expand Down Expand Up @@ -113,7 +116,7 @@ internal class HCaptchaWebViewManager: NSObject {
}()

/// Responsible for external link handling
fileprivate let urlOpener: HCaptchaURLOpener
internal let urlOpener: HCaptchaURLOpener

/**
- parameters:
Expand Down Expand Up @@ -162,9 +165,11 @@ internal class HCaptchaWebViewManager: NSObject {
*/
func validate(on view: UIView) {
Log.debug("WebViewManager.validate on: \(view)")
resultHandled = false

#if DEBUG
guard !shouldSkipForTests else {
completion?(HCaptchaResult(token: ""))
completion?(HCaptchaResult(self, token: ""))
return
}
#endif
Expand All @@ -181,6 +186,7 @@ internal class HCaptchaWebViewManager: NSObject {
Log.debug("WebViewManager.stop")
stopInitWebViewConfiguration = true
webView.stopLoading()
resultHandled = true
}

/**
Expand All @@ -192,6 +198,7 @@ internal class HCaptchaWebViewManager: NSObject {
Log.debug("WebViewManager.reset")
configureWebViewDispatchToken = UUID()
stopInitWebViewConfiguration = false
resultHandled = false
if didFinishLoading {
executeJS(command: .reset)
didFinishLoading = false
Expand Down Expand Up @@ -228,8 +235,15 @@ fileprivate extension HCaptchaWebViewManager {
*/
func handle(result: HCaptchaDecoder.Result) {
Log.debug("WebViewManager.handleResult: \(result)")

guard !resultHandled else {
Log.debug("WebViewManager.handleResult skip as handled")
return
}

switch result {
case .token(let token): completion?(HCaptchaResult(token: token))
case .token(let token):
completion?(HCaptchaResult(self, token: token))
case .error(let error):
handle(error: error)
onEvent?(.error, error)
Expand All @@ -249,11 +263,11 @@ fileprivate extension HCaptchaWebViewManager {
reset()
validate(on: view)
} else {
completion?(HCaptchaResult(error: error))
completion?(HCaptchaResult(self, error: error))
}
} else {
if let completion = completion {
completion(HCaptchaResult(error: error))
completion(HCaptchaResult(self, error: error))
} else {
lastError = error
}
Expand Down Expand Up @@ -338,9 +352,11 @@ fileprivate extension HCaptchaWebViewManager {
guard didLoad else {
if let error = lastError {
DispatchQueue.main.async { [weak self] in
guard let self = self else { return }
Log.debug("WebViewManager complete with pendingError: \(error)")
self?.completion?(HCaptchaResult(error: error))
self?.lastError = nil

self.completion?(HCaptchaResult(self, error: error))
self.lastError = nil
}
if error == .networkError {
Log.debug("WebViewManager reloads html after \(error) error")
Expand All @@ -360,39 +376,3 @@ fileprivate extension HCaptchaWebViewManager {
executeJS(command: command, didLoad: self.didFinishLoading)
}
}

extension HCaptchaWebViewManager: WKNavigationDelegate {
func webView(_ webView: WKWebView, decidePolicyFor navigationAction: WKNavigationAction,
decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) {
if navigationAction.targetFrame == nil, let url = navigationAction.request.url, urlOpener.canOpenURL(url) {
urlOpener.openURL(url)
}
decisionHandler(WKNavigationActionPolicy.allow)
}

/// Tells the delegate that an error occurred during navigation.
func webView(_ webView: WKWebView, didFail navigation: WKNavigation!, withError error: Error) {
Log.debug("WebViewManager.webViewDidFail with \(error)")
completion?(HCaptchaResult(error: .unexpected(error)))
}

/// Tells the delegate that an error occurred during the early navigation process.
func webView(_ webView: WKWebView, didFailProvisionalNavigation navigation: WKNavigation!, withError error: Error) {
Log.debug("WebViewManager.webViewDidFailProvisionalNavigation with \(error)")
completion?(HCaptchaResult(error: .unexpected(error)))
}

/// Tells the delegate that the web view’s content process was terminated.
func webViewWebContentProcessDidTerminate(_ webView: WKWebView) {
Log.debug("WebViewManager.webViewWebContentProcessDidTerminate")
let kHCaptchaErrorWebViewProcessDidTerminate = -1
let kHCaptchaErrorDomain = "com.hcaptcha.sdk-ios"
let error = NSError(domain: kHCaptchaErrorDomain,
code: kHCaptchaErrorWebViewProcessDidTerminate,
userInfo: [
NSLocalizedDescriptionKey: "WebView web content process did terminate",
NSLocalizedRecoverySuggestionErrorKey: "Call HCaptcha.reset()"])
completion?(HCaptchaResult(error: .unexpected(error)))
didFinishLoading = false
}
}

0 comments on commit 76dbfff

Please sign in to comment.