// Copyright © 2020 Metabolist. All rights reserved. import Foundation import Combine class IdentityService { @Published private(set) var identity: Identity let observationErrors: AnyPublisher private let identityDatabase: IdentityDatabase private let environment: AppEnvironment private let networkClient: MastodonClient private let secretsService: SecretsService private let observationErrorsInput = PassthroughSubject() init(identityID: UUID, identityDatabase: IdentityDatabase, environment: AppEnvironment) throws { self.identityDatabase = identityDatabase self.environment = environment observationErrors = observationErrorsInput.eraseToAnyPublisher() let observation = identityDatabase.identityObservation(id: identityID).share() var initialIdentity: Identity? _ = observation.first().sink( receiveCompletion: { _ in }, receiveValue: { initialIdentity = $0 }) guard let identity = initialIdentity else { throw IdentityDatabaseError.identityNotFound } self.identity = identity secretsService = SecretsService( identityID: identityID, keychainServiceType: environment.keychainServiceType) networkClient = MastodonClient(session: environment.session) networkClient.instanceURL = identity.url networkClient.accessToken = try secretsService.item(.accessToken) observation.catch { [weak self] error -> Empty in self?.observationErrorsInput.send(error) return Empty() } .assign(to: &$identity) } } extension IdentityService { var isAuthorized: Bool { networkClient.accessToken != nil } func updateLastUse() -> AnyPublisher { identityDatabase.updateLastUsedAt(identityID: identity.id) } func verifyCredentials() -> AnyPublisher { networkClient.request(AccountEndpoint.verifyCredentials) .zip(Just(identity.id).first().setFailureType(to: Error.self)) .flatMap(identityDatabase.updateAccount) .eraseToAnyPublisher() } func refreshServerPreferences() -> AnyPublisher { networkClient.request(PreferencesEndpoint.preferences) .zip(Just(self).first().setFailureType(to: Error.self)) .map { ($1.identity.preferences.updated(from: $0), $1.identity.id) } .flatMap(identityDatabase.updatePreferences) .eraseToAnyPublisher() } func refreshInstance() -> AnyPublisher { networkClient.request(InstanceEndpoint.instance) .zip(Just(identity.id).first().setFailureType(to: Error.self)) .flatMap(identityDatabase.updateInstance) .eraseToAnyPublisher() } func identitiesObservation() -> AnyPublisher<[Identity], Error> { identityDatabase.identitiesObservation() } func recentIdentitiesObservation() -> AnyPublisher<[Identity], Error> { identityDatabase.recentIdentitiesObservation(excluding: identity.id) } func updatePreferences(_ preferences: Identity.Preferences) -> AnyPublisher { identityDatabase.updatePreferences(preferences, forIdentityID: identity.id) } func createPushSubscription(deviceToken: String, alerts: PushSubscription.Alerts) -> AnyPublisher { let publicKey: String let auth: String do { publicKey = try secretsService.generatePushKeyAndReturnPublicKey().base64EncodedString() auth = try secretsService.generatePushAuth().base64EncodedString() } catch { return Fail(error: error).eraseToAnyPublisher() } let identityID = identity.id let endpoint = Self.pushSubscriptionEndpointURL .appendingPathComponent(deviceToken) .appendingPathComponent(identityID.uuidString) return networkClient.request( PushSubscriptionEndpoint.create( endpoint: endpoint, publicKey: publicKey, auth: auth, alerts: alerts)) .map { (deviceToken, $0.alerts, identityID) } .flatMap(identityDatabase.updatePushSubscription(deviceToken:alerts:forIdentityID:)) .eraseToAnyPublisher() } } private extension IdentityService { #if DEBUG static let pushSubscriptionEndpointURL = URL(string: "https://metatext-apns.metabolist.com/push?sandbox=true")! #else static let pushSubscriptionEndpointURL = URL(string: "https://metatext-apns.metabolist.com/push")! #endif }