ios - 使用 NSURLSession 的 HTTP 基本身份验证
<p><p>我正在尝试使用 <code>NSURLSession</code> 实现 HTTP 基本身份验证,但遇到了几个问题。请在回答之前阅读整个问题,我怀疑这是与其他问题的重复。</p>
<p>根据我运行的测试,<code>NSURLSession</code> 的行为如下:</p>
<ul>
<li>第一个请求总是不带 <code>Authorization</code>header 。</li>
<li>如果第一个请求失败并返回 <code>401 Unauthorized</code> 响应和 <code>WWW-Authenticate Basic realm=...</code>header ,则会自动重试。</li>
<li>在重试请求之前, session 将尝试通过查看 session 配置的 <code>NSURLCredentialStorage</code> 或调用 <code>URLSession:task:didReceiveChallenge:completionHandler:</code> 来获取凭据委托(delegate)方法(或两者)。</li>
<li>如果可以获取凭据,则使用正确的 <code>Authorization</code>header 重试请求。如果不是,则在没有 header 的情况下重试(这很奇怪,因为在这种情况下,这是完全相同的请求)。</li>
<li>如果第二个请求成功,则任务会透明地报告为成功,并且您甚至不会收到请求已尝试两次的通知。如果不是,则报告第二个请求失败(但不是第一个)。</li>
</ul>
<p>我对这种行为的问题是我通过多部分请求将大文件上传到我的服务器,所以当请求被尝试两次时,整个 <code>POST</code> 正文被发送两次,这是一个可怕的开销。</p>
<p>我曾尝试手动将 <code>Authorization</code>header 添加到 session 配置的 <code>httpAdditionalHeaders</code> 中,但只有在 <em>before</em> 设置了属性时才有效>session 已创建。之后尝试修改 <code>session.configuration.httpAdditionalHeaders</code> 不起作用。此外,文档明确指出不应手动设置 <code>Authorization</code>header 。</p>
<hr/>
<p><strong>所以我的问题是</strong>:如果我需要在获得凭据之前启动 session ,并且如果我想确保始终使用正确的 <code>Authorization</code> 发出请求第一次报头,怎么办?</p>
<hr/>
<p>这是我用于测试的代码示例。你可以用它重现我上面描述的所有行为。</p>
<p>请注意,为了能够看到双重请求,您需要使用自己的 http 服务器并记录请求或通过记录所有请求的代理连接(我为此使用了 <a href="https://www.charlesproxy.com" rel="noreferrer noopener nofollow">Charles Proxy</a>)</p>
<pre><code>class URLSessionTest: NSObject, URLSessionDelegate
{
static let shared = URLSessionTest()
func start()
{
let requestURL = URL(string: "https://httpbin.org/basic-auth/username/password")!
let credential = URLCredential(user: "username", password: "password", persistence: .forSession)
let protectionSpace = URLProtectionSpace(host: "httpbin.org", port: 443, protocol: NSURLProtectionSpaceHTTPS, realm: "Fake Realm", authenticationMethod: NSURLAuthenticationMethodHTTPBasic)
let useHTTPHeader = false
let useCredentials = true
let useCustomCredentialsStorage = false
let useDelegateMethods = true
let sessionConfiguration = URLSessionConfiguration.default
if (useHTTPHeader) {
let authData = "\(credential.user!):\(credential.password!)".data(using: .utf8)!
let authValue = "Basic " + authData.base64EncodedString()
sessionConfiguration.httpAdditionalHeaders = ["Authorization": authValue]
}
if (useCredentials) {
if (useCustomCredentialsStorage) {
let urlCredentialStorage = URLCredentialStorage()
urlCredentialStorage.set(credential, for: protectionSpace)
sessionConfiguration.urlCredentialStorage = urlCredentialStorage
} else {
sessionConfiguration.urlCredentialStorage?.set(credential, for: protectionSpace)
}
}
let delegate = useDelegateMethods ? self : nil
let session = URLSession(configuration: sessionConfiguration, delegate: delegate, delegateQueue: nil)
self.makeBasicAuthTest(url: requestURL, session: session) {
self.makeBasicAuthTest(url: requestURL, session: session) {
DispatchQueue.main.asyncAfter(deadline: .now() + 61.0) {
self.makeBasicAuthTest(url: requestURL, session: session) {}
}
}
}
}
func makeBasicAuthTest(url: URL, session: URLSession, completion: @escaping () -> Void)
{
let task = session.dataTask(with: url) { (data, response, error) in
if let response = response {
print("response : \(response)")
}
if let data = data {
if let json = try? JSONSerialization.jsonObject(with: data, options: .allowFragments) {
print("json : \(json)")
} else if data.count > 0, let string = String(data: data, encoding: .utf8) {
print("string : \(string)")
} else {
print("data : \(data)")
}
}
if let error = error {
print("error : \(error)")
}
print()
DispatchQueue.main.async(execute: completion)
}
task.resume()
}
@objc(URLSession:didReceiveChallenge:completionHandler:)
func urlSession(_ session: URLSession, didReceive challenge: URLAuthenticationChallenge, completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Swift.Void)
{
print("Session authenticationMethod: \(challenge.protectionSpace.authenticationMethod)")
if (challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodHTTPBasic) {
let credential = URLCredential(user: "username", password: "password", persistence: .forSession)
completionHandler(.useCredential, credential)
} else {
completionHandler(.performDefaultHandling, nil)
}
}
@objc(URLSession:task:didReceiveChallenge:completionHandler:)
func urlSession(_ session: URLSession, task: URLSessionTask, didReceive challenge: URLAuthenticationChallenge, completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Swift.Void)
{
print("Task authenticationMethod: \(challenge.protectionSpace.authenticationMethod)")
if (challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodHTTPBasic) {
let credential = URLCredential(user: "username", password: "password", persistence: .forSession)
completionHandler(.useCredential, credential)
} else {
completionHandler(.performDefaultHandling, nil)
}
}
}
</code></pre>
<p><em>注意 1</em>:当向同一个端点连续发出多个请求时,我上面描述的行为只涉及第一个请求。第一次使用正确的 <code>Authorization</code>header 尝试后续请求。但是,如果您等待一段时间(大约 1 分钟), session 将恢复到默认行为(第一次请求尝试了两次)。</p>
<p><em>注 2</em>:这没有直接关系,但使用自定义 <code>NSURLCredentialStorage</code> 为 session 配置的 <code>urlCredentialStorage</code> 似乎没有工作。只有使用默认值(根据文档,它是共享的 <code>NSURLCredentialStorage</code>)才有效。</p>
<p><em>注意 3</em>:我尝试过使用 <code>Alamofire</code>,但由于它基于 <code>NSURLSession</code>,因此其行为方式完全相同。</p></p>
<br><hr><h1><strong>Best Answer-推荐答案</ strong></h1><br>
<p><p>如果可能,服务器应该在客户端完成发送正文之前很久就响应错误。但是,在许多高级服务器端语言中,这很困难,并且无法保证即使您这样做也会停止上传。</p>
<p>真正的问题是您正在使用单个 POST 请求执行大型上传。这会使身份验证出现问题,并且如果连接在上传中途断开,还会阻止任何有用的继续上传。分 block 上传基本上可以解决您的所有问题:</p>
<ul>
<li><p>对于您的第一个请求,仅发送适合的数量而不添加额外的以太网数据包,即计算您的典型 header 大小,以 1500 字节为模,添加几十个字节以获得良好的度量,从 1500 中减去,并为您的第一个 block 硬编码该大小。最多浪费了几个包。</p></li>
<li><p>对于后续的 block ,增大大小。</p></li>
<li><p>当请求失败时,询问服务器它得到了多少,然后从上传中断的地方重试。</p></li>
<li><p>上传完成后向服务器发出请求。</p></li>
<li><p>使用 cron 作业或其他方式定期清除服务器端的部分上传。</p></li>
</ul>
<p>也就是说,如果您无法控制服务器端,通常的解决方法是在您的 POST 请求之前发送经过身份验证的 GET 请求。这可以最大限度地减少浪费的数据包,同时只要网络可靠,大部分时间仍然可以正常工作。</p></p>
<p style="font-size: 20px;">关于ios - 使用 NSURLSession 的 HTTP 基本身份验证,我们在Stack Overflow上找到一个类似的问题:
<a href="https://stackoverflow.com/questions/42824063/" rel="noreferrer noopener nofollow" style="color: red;">
https://stackoverflow.com/questions/42824063/
</a>
</p>
页:
[1]