metatext/Services/AuthenticationService.swift

114 lines
4.2 KiB
Swift
Raw Normal View History

2020-08-09 05:37:04 +00:00
// Copyright © 2020 Metabolist. All rights reserved.
import Foundation
import Combine
2020-08-30 23:33:11 +00:00
import Mastodon
2020-08-09 05:37:04 +00:00
struct AuthenticationService {
2020-08-30 23:59:49 +00:00
private let networkClient: APIClient
2020-08-12 09:01:21 +00:00
private let webAuthSessionType: WebAuthSession.Type
2020-08-09 05:37:04 +00:00
private let webAuthSessionContextProvider = WebAuthSessionContextProvider()
init(environment: AppEnvironment) {
2020-08-30 23:59:49 +00:00
networkClient = APIClient(session: environment.session)
webAuthSessionType = environment.webAuthSessionType
2020-08-09 05:37:04 +00:00
}
}
extension AuthenticationService {
func authorizeApp(instanceURL: URL) -> AnyPublisher<AppAuthorization, Error> {
2020-08-09 05:37:04 +00:00
let endpoint = AppAuthorizationEndpoint.apps(
clientName: OAuth.clientName,
redirectURI: OAuth.callbackURL.absoluteString,
scopes: OAuth.scopes,
website: OAuth.website)
2020-08-30 23:59:49 +00:00
let target = Target(baseURL: instanceURL, endpoint: endpoint, accessToken: nil)
2020-08-09 05:37:04 +00:00
return networkClient.request(target)
}
func authenticate(instanceURL: URL, appAuthorization: AppAuthorization) -> AnyPublisher<AccessToken, Error> {
guard let authorizationURL = authorizationURL(
instanceURL: instanceURL,
clientID: appAuthorization.clientId) else {
return Fail(error: URLError(.badURL)).eraseToAnyPublisher()
}
2020-08-09 05:37:04 +00:00
return webAuthSessionType.publisher(
url: authorizationURL,
callbackURLScheme: OAuth.callbackURLScheme,
presentationContextProvider: webAuthSessionContextProvider)
.tryCatch { error -> AnyPublisher<URL?, Error> in
if (error as? WebAuthSessionError)?.code == .canceledLogin {
return Just(nil).setFailureType(to: Error.self).eraseToAnyPublisher()
}
2020-08-09 05:37:04 +00:00
throw error
2020-08-09 05:37:04 +00:00
}
.compactMap { $0 }
.tryMap { url -> String in
guard let queryItems = URLComponents(url: url, resolvingAgainstBaseURL: true)?.queryItems,
let code = queryItems.first(where: {
$0.name == OAuth.codeCallbackQueryItemName
})?.value
else { throw OAuthError.codeNotFound }
return code
}
.flatMap { code -> AnyPublisher<AccessToken, Error> in
let endpoint = AccessTokenEndpoint.oauthToken(
clientID: appAuthorization.clientId,
clientSecret: appAuthorization.clientSecret,
code: code,
grantType: OAuth.grantType,
scopes: OAuth.scopes,
redirectURI: OAuth.callbackURL.absoluteString)
2020-08-30 23:59:49 +00:00
let target = Target(baseURL: instanceURL, endpoint: endpoint, accessToken: nil)
return networkClient.request(target)
}
.eraseToAnyPublisher()
2020-08-09 05:37:04 +00:00
}
}
private extension AuthenticationService {
struct OAuth {
static let clientName = "Metatext"
static let scopes = "read write follow push"
static let codeCallbackQueryItemName = "code"
static let grantType = "authorization_code"
static let callbackURLScheme = "metatext"
static let callbackURL = URL(string: "\(callbackURLScheme)://oauth.callback")!
static let website = URL(string: "https://metabolist.com/metatext")!
2020-08-09 05:37:04 +00:00
}
enum OAuthError {
case codeNotFound
2020-08-09 05:37:04 +00:00
}
private func authorizationURL(instanceURL: URL, clientID: String) -> URL? {
guard var authorizationURLComponents = URLComponents(url: instanceURL, resolvingAgainstBaseURL: true) else {
return nil
2020-08-09 05:37:04 +00:00
}
authorizationURLComponents.path = "/oauth/authorize"
authorizationURLComponents.queryItems = [
"client_id": clientID,
"scope": OAuth.scopes,
"response_type": "code",
"redirect_uri": OAuth.callbackURL.absoluteString
].map { URLQueryItem(name: $0, value: $1) }
2020-08-09 05:37:04 +00:00
return authorizationURLComponents.url
}
}
2020-08-09 05:37:04 +00:00
extension AuthenticationService.OAuthError: LocalizedError {
var errorDescription: String? {
switch self {
case .codeNotFound:
return NSLocalizedString("oauth.error.code-not-found", comment: "")
2020-08-09 05:37:04 +00:00
}
}
}