我有一个我想连接的API,服务器是Windows身份验证。
我正在尝试将URLSession与URLCredential一起使用委托方法
func urlSession(_ session: URLSession, didReceive challenge: URLAuthenticationChallenge, completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Swift.Void){
var disposition: URLSession.AuthChallengeDisposition = URLSession.AuthChallengeDisposition.performDefaultHandling
var credential:URLCredential?
print(challenge.protectionSpace.authenticationMethod)
if challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodServerTrust {
credential = URLCredential(trust: challenge.protectionSpace.serverTrust!)
if (credential != nil) {
disposition = URLSession.AuthChallengeDisposition.useCredential
}
else
{
disposition = URLSession.AuthChallengeDisposition.performDefaultHandling
}
}
else
{
disposition = URLSession.AuthChallengeDisposition.cancelAuthenticationChallenge
}
completionHandler(disposition, credential);
}
此代码运行两次,因为有两种身份验证方法:
NSURLAuthenticationMethodServerTrust和NSURLAuthenticationMethodNTLM在通过NSURLAuthenticationMethodServerTrust运行时一切正常,但是当它运行NSURLAuthenticationMethodNTLM时,我在这一行收到错误:
credential = URLCredential(trust: challenge.protectionSpace.serverTrust!)
这样说:
fatal error: unexpectedly found nil while unwrapping an Optional value
但仅当我从
更改此条件时if challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodServerTrust {
到
if challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodNTLM {
我做错了什么?
以下是我尝试连接到该API的方法
func loginUser(_ username: String, password: String, completion: @escaping (_ result: Bool) -> Void)
{
//Create request URL as String
let requestString = String(format:"%@", webservice) as String
//Covert URL request string to URL
guard let url = URL(string: requestString) else {
print("Error: cannot create URL")
return
}
//Convert URL to URLRequest
let urlRequest = URLRequest(url: url)
print(urlRequest)
//Add the username and password to URLCredential
credentials = URLCredential(user:username, password:password, persistence: .forSession)
print(credentials)
//Setup the URLSessionConfiguration
let config = URLSessionConfiguration.default
//Setup the URLSession
let session = URLSession(configuration: config, delegate: self, delegateQueue: nil)
//Prepare the task to get data.
let task = session.dataTask(with: urlRequest, completionHandler: { (data, response, error) in
DispatchQueue.main.async(execute: {
print(error!)
if(error == nil)
{
completion(true)
}
else
{
completion(false)
}
})
})
//Run the task to get data.
task.resume()
}
答案 0 :(得分:1)
在Apple documentation for URLProtectionSpace.serverTrust
中,它说:
如果保护空间的身份验证方法不是服务器信任,则
。
所以,你试图解开一个nil
的可选值,这当然会导致崩溃。
在您的情况下,您可能只需用(未经测试)替换整个函数:
func urlSession(_ session: URLSession, didReceive challenge: URLAuthenticationChallenge, completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Swift.Void){
var disposition: URLSession.AuthChallengeDisposition = URLSession.AuthChallengeDisposition.performDefaultHandling
print(challenge.protectionSpace.authenticationMethod)
if challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodServerTrust {
disposition = URLSession.AuthChallengeDisposition.performDefaultHandling
}
else
{
disposition = URLSession.AuthChallengeDisposition.cancelAuthenticationChallenge
}
completionHandler(disposition, credential);
}