// Copyright © 2020 Metabolist. All rights reserved. import Foundation import Combine struct IdentitiesService { let mostRecentlyUsedIdentityID: AnyPublisher private let identityDatabase: IdentityDatabase private let environment: AppEnvironment init(identityDatabase: IdentityDatabase, environment: AppEnvironment) { self.identityDatabase = identityDatabase self.environment = environment mostRecentlyUsedIdentityID = identityDatabase.mostRecentlyUsedIdentityIDObservation() .replaceError(with: nil) .eraseToAnyPublisher() } } extension IdentitiesService { func identityService(id: UUID) throws -> IdentityService { try IdentityService(identityID: id, identityDatabase: identityDatabase, environment: environment) } func createIdentity(id: UUID, instanceURL: URL) -> AnyPublisher { identityDatabase.createIdentity(id: id, url: instanceURL) } func authorizeIdentity(id: UUID, instanceURL: URL) -> AnyPublisher { let secretsService = SecretsService(identityID: id, keychainService: environment.keychainServiceType) let authenticationService = AuthenticationService(environment: environment) return authenticationService.authorizeApp(instanceURL: instanceURL) .tryMap { appAuthorization -> (URL, AppAuthorization) in try secretsService.set(appAuthorization.clientId, forItem: .clientID) try secretsService.set(appAuthorization.clientSecret, forItem: .clientSecret) return (instanceURL, appAuthorization) } .flatMap(authenticationService.authenticate(instanceURL:appAuthorization:)) .tryMap { try secretsService.set($0.accessToken, forItem: .accessToken) } .ignoreOutput() .eraseToAnyPublisher() } func deleteIdentity(_ identity: Identity) -> AnyPublisher { let secretsService = SecretsService(identityID: identity.id, keychainService: environment.keychainServiceType) let networkClient = MastodonClient(environment: environment) networkClient.instanceURL = identity.url return identityDatabase.deleteIdentity(id: identity.id) .tryMap { _ in DeletionEndpoint.oauthRevoke( token: try secretsService.item(.accessToken), clientID: try secretsService.item(.clientID), clientSecret: try secretsService.item(.clientSecret)) } .flatMap(networkClient.request) .tryMap { _ in try secretsService.deleteAllItems() } .ignoreOutput() .eraseToAnyPublisher() } func updatePushSubscriptions(deviceToken: String) -> AnyPublisher { identityDatabase.identitiesWithOutdatedDeviceTokens(deviceToken: deviceToken) .tryMap { identities -> [AnyPublisher] in try identities.map { try identityService(id: $0.id) .createPushSubscription(deviceToken: deviceToken, alerts: $0.pushSubscriptionAlerts) .catch { _ in Empty() } // don't want to disrupt pipeline .eraseToAnyPublisher() } } .map(Publishers.MergeMany.init) .ignoreOutput() .eraseToAnyPublisher() } }