gpt4 book ai didi

swift - SafariViewController : How to grab OAuth token from URL?

转载 作者:IT王子 更新时间:2023-10-29 05:14:21 26 4
gpt4 key购买 nike

尝试将 Facebook OAuth 与 SafariViewController 结合使用。首先,我用 SafariViewController 打开 authURL,如果用户在 Safari 上登录 Facebook,它将重定向他们并返回一个带有特定服务 token 的 OAuth URL,例如Instagram

响应:https://www.facebook.com/connect/login_success.html#access_token=BLAHTOKENRESPONSE&expires_in=5114338

当 SafariViewController 重定向后,我想获取响应 URL 并存储它以便获取 token 。这是我的代码:

import SafariServices

let kSafariViewControllerCloseNotification = "kSafariViewControllerCloseNotification"

import UIKit

// facebook OAuth URL for service
let authURL = NSURL(string: "https://www.facebook.com/dialog/oauth?client_id=3627644767&redirect_uri=https://www.facebook.com/connect/login_success.html&scope=basic_info,email,public_profile,user_about_me,user_activities,user_birthday,user_education_history,user_friends,user_interests,user_likes,user_location,user_photos,user_relationship_details&response_type=token")

class ViewController: UIViewController, SFSafariViewControllerDelegate {

var safariVC: SFSafariViewController?
@IBOutlet weak var loginButton: UIButton!

@IBAction func loginButtonTapped(sender: UIButton) {
safariVC = SFSafariViewController(URL: authURL!)
safariVC!.delegate = self
self.presentViewController(safariVC!, animated: true, completion: nil)
}

override func viewDidLoad() {
super.viewDidLoad()
// not firing the safariLogin function below
NSNotificationCenter.defaultCenter().addObserver(self, selector: #selector(ViewController.safariLogin(_:)), name: kSafariViewControllerCloseNotification, object: nil)
}

func safariLogin(notification: NSNotification) {
print("Safari Login call")
// get the url form the auth callback
let url = notification.object as! NSURL
print(url)
self.safariVC!.dismissViewControllerAnimated(true, completion: nil)
}

func application(application: UIApplication, openURL url: NSURL, sourceApplication: String?, annotation: AnyObject) -> Bool {
print("application call")
// just making sure we send the notification when the URL is opened in SFSafariViewController
if (sourceApplication == "com.application.SafariViewTest") {
NSNotificationCenter.defaultCenter().postNotificationName(kSafariViewControllerCloseNotification, object: url)
return true
}
return true
}

}

它打开 authURL 并重定向到正确的响应 URL,但观察者不会触发 safariLogin 函数来获取 URL。任何帮助将不胜感激!

非常感谢!

最佳答案

iOS 12+

iOS 12 Beta 已经弃用 SFAuthenticationSession(见下文),取而代之的是 ASWebAuthenticationSession .看起来它的使用方式完全相同,但需要新的 AuthenticationServices框架。

iOS 11

引入 iOS 11 SFAuthenticationSession这更容易处理。鉴于其性质,此 beta API 可能仍会更改,但互联网上已经有几个示例(12)。首先,您需要一个用身份验证请求的结果调用的完成处理程序:

let completion : SFAuthenticationSession.CompletionHandler = { (callBack:URL?, error:Error?) in
guard error == nil, let successURL = callBack else {
return
}

let oauthToken = NSURLComponents(string: (successURL.absoluteString))?.queryItems?.filter({$0.name == "oauth_token"}).first

// Do what you have to do...
}

然后您只需创建一个 SFAuthenticationSession 并启动它。

let authURL = "https://the.service.you/want/toAuthorizeWith?..."
let scheme = "YOURSCHEME://"
let authSession = SFAuthenticationSession(url: authURL, callbackURLScheme: scheme, completionHandler: completion)
authSession.start()

iOS 10 及之前版本

正如一些人在评论中指出的那样,接受的答案是不完整的,不能单独使用。浏览 Strawberry Code 的博客文章时,可以找到 link to the related GitHub project。 .该项目的 README.MD 解释了设置的关键部分,即将重定向 URI 添加到 Info.plist。所以整个事情是这样的:

AppDelegate.swift

func application(_ app: UIApplication, open url: URL, options: [UIApplicationOpenURLOptionsKey : Any] = [:]) -> Bool {

if let sourceApplication = options[.sourceApplication] {
if (String(describing: sourceApplication) == "com.apple.SafariViewService") {
NotificationCenter.default.post(name: Notification.Name("CallbackNotification"), object: url)
return true
}
}

return false
}

ViewController.swift

注册通知以在某个合理的地方调用您的处理程序。我建议不要在 viewDidLoad() 中这样做,而是在您实际呈现 SFSafariViewController 之前这样做。

NotificationCenter.default.addObserver(self, selector: #selector(safariLogin(_:)), name: Notification.Name("CallbackNotification"), object: nil)
let safariVC = SFSafariViewController(URL: authURL)
safariVC.delegate = self
self.present(safariVC, animated: true, completion: nil)

然后删除处理程序中的遵守:

@objc func safariLogin(_ notification : Notification) {

NotificationCenter.default.removeObserver(self, name: Notification.Name("CallbackNotification"), object: nil)

guard let url = notification.object as? URL else {
return
}

// Parse url ...

}

请记住,用户可以通过点击Done 按钮关闭SFSafariViewController,因此请务必采用SFSafariViewControllerDelegate 协议(protocol)并移除也有这样的仪式:

func safariViewControllerDidFinish(_ controller: SFSafariViewController) {
NotificationCenter.default.removeObserver(self, name: Notification.Name("CallbackNotification"), object: nil)
}

信息.plist

要使其全部正常工作,您需要将重定向 URI 方案添加到 Info.plist:

<key>CFBundleURLTypes</key>
<array>
<dict>
<key>CFBundleURLName</key>
<string>com.YOUR.BUNDLE.IDENTIFIER</string>
<key>CFBundleURLSchemes</key>
<array>
<string>YOURSCHEME</string>
</array>
</dict>
</array>

当然,YOURSCHEME 必须与您注册的重定向 URI 的方案相匹配,而您正尝试对其进行授权。

关于swift - SafariViewController : How to grab OAuth token from URL?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38818786/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com