我已经为自己构建了一个APIService类,处理与我的应用程序使用的API的所有交互
func callEndpointX(token: String, completion: @escaping(Result<User, APIError>) -> Void) {
guard let endpoint = URL(string: apiBaseUrl + "/endpointX") else {fatalError()}
var request = URLRequest(url: endpoint)
request.addValue("Bearer " + token, forHTTPHeaderField: "Authorization")
request.httpMethod = "GET"
let dataTask = URLSession.shared.dataTask(with: request) { data, response, error in
guard let httpResponse = response as? HTTPURLResponse, httpResponse.statusCode == 200, let jsonData = data
else { ... completion(.failure(.responseError)); return }
do {
let response = try JSONDecoder()...
completion(.success(response.detailresponse!))
} catch {
...
completion(.failure(.decodingError))
}
}
dataTask.resume()
}我面临的挑战是,我要传递的Bearer 令牌 do 过期。在出现过期令牌时,我想要做的事情是,在访问api之前,为用户静默地刷新令牌。因此,我有一个迷你函数func tokenExpired() -> Bool {...}来检查令牌是否已经过期,并且我可以以与上面func callEndpointX类似的方式访问另一个API --所有refreshToken。
我所挣扎的是如何将这些链接在一起以避免竞争条件()--即当func callEndpointX被调用时,它应该检查func tokenExpired(),并且在完成之前不要继续任何其他工作。如果过期,它应该首先执行refreshToken,然后才返回,最后执行func callEndpointX的ret。
所有这些链子搅乱了我的大脑,我希望有人能指引我到这里来。
非常感谢!
发布于 2020-03-29 17:04:20
解决方案是在完成令牌刷新之后递归调用callEndPointX。假设您的令牌请求如下:
func requestToken(completion:@escaping((String) -> Void))然后
func callEndpointX(token: String, completion: @escaping(Result<User, APIError>) -> Void) {
if tokenExpired() {
requestToken(completion: { newToken in
callEndpointX(token: newToken, completion: completion)
})
}
else {
// your code for callendpointX
}
}https://stackoverflow.com/questions/60913591
复制相似问题