我从UITextField
检索了用户名和密码,现在我想为Moya用基本身份验证执行的每个请求设置用户名和密码。
我该怎么做?
答案 0 :(得分:0)
涵盖基本身份验证的文档为here
以下是您需要的部件
HTTP auth是HTTP协议本身内置的用户名/密码质询。如果您需要使用HTTP身份验证,则可以在初始化提供程序时提供CredentialsPlugin。
let provider = MoyaProvider<YourAPI>(plugins: [CredentialsPlugin { _ -> URLCredential? in
return URLCredential(user: "user", password: "passwd", persistence: .none)
}
])
此特定示例显示了使用HTTP来验证每个请求,这通常是不必要的。这可能是一个更好的主意:
let provider = MoyaProvider<YourAPI>(plugins: [CredentialsPlugin { target -> URLCredential? in
switch target {
case .targetThatNeedsAuthentication:
return URLCredential(user: "user", password: "passwd", persistence: .none)
default:
return nil
}
}
])
答案 1 :(得分:0)
此解决方案不适用于我。
问题在于URLCredential
仅用于身份验证质询,而不用于对请求进行预授权。因此,如果您的API需要带有Base64“ username:password”参数的Authorization标头,则此方法将无效。关于它的工作方式似乎有很多困惑,请参见Alamofire上的类似问题。
我通过以下方法解决了这个问题:
解决方案1
lazy var provider: MoyaProvider<LoginService> = {
let endpointClosure = { (target: LoginService) -> Endpoint<LoginService> in
let defaultEndpoint = MoyaProvider.defaultEndpointMapping(for: target)
switch target {
case .login(let username, let password):
return defaultEndpoint.adding(newHTTPHeaderFields: ["Authorization": "Basic " + "\(username):\(password)".data(using: .nonLossyASCII)!.base64EncodedString(options: []))
}
}
return MoyaProvider(endpointClosure: endpointClosure)
}()
以下是使用Moya插件方法的其他解决方案:
解决方案2
class AuthProvider {
static let basicAuthPlugin: PluginType = AccessTokenPlugin(tokenClosure: { () -> String in
guard let loginData = String(format: "\("username"):\("password")").data(using: .utf8) else { return "" }
return loginData.base64EncodedString()
})
}
let AuthAPIProvider = MoyaProvider<AuthAPI>(plugins: [AuthProvider.basicAuthPlugin])
enum AuthAPI {
// ... methods
}
extension AuthAPI: AccessTokenAuthorizable {
var authorizationType: AuthorizationType {
return .basic
}
}