2020-08-07 21:57:18 +00:00
|
|
|
// Copyright © 2020 Metabolist. All rights reserved.
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import Combine
|
|
|
|
|
2020-08-08 23:08:47 +00:00
|
|
|
class IdentityService {
|
2020-08-07 21:57:18 +00:00
|
|
|
@Published var identity: Identity
|
|
|
|
let observationErrors: AnyPublisher<Error, Never>
|
|
|
|
|
|
|
|
private let networkClient: MastodonClient
|
|
|
|
private let appEnvironment: AppEnvironment
|
|
|
|
private let observationErrorsInput = PassthroughSubject<Error, Never>()
|
|
|
|
private var cancellables = Set<AnyCancellable>()
|
|
|
|
|
2020-08-07 23:19:13 +00:00
|
|
|
init(identityID: UUID, appEnvironment: AppEnvironment) throws {
|
2020-08-07 21:57:18 +00:00
|
|
|
self.appEnvironment = appEnvironment
|
|
|
|
observationErrors = observationErrorsInput.eraseToAnyPublisher()
|
|
|
|
networkClient = MastodonClient(configuration: appEnvironment.URLSessionConfiguration)
|
|
|
|
networkClient.accessToken = try appEnvironment.secrets.item(.accessToken, forIdentityID: identityID)
|
|
|
|
|
|
|
|
let observation = appEnvironment.identityDatabase.identityObservation(id: identityID).share()
|
|
|
|
|
|
|
|
var initialIdentity: Identity?
|
|
|
|
|
|
|
|
observation.first().sink(
|
|
|
|
receiveCompletion: { _ in },
|
|
|
|
receiveValue: { initialIdentity = $0 })
|
|
|
|
.store(in: &cancellables)
|
|
|
|
|
|
|
|
guard let identity = initialIdentity else { throw IdentityDatabaseError.identityNotFound }
|
|
|
|
|
|
|
|
self.identity = identity
|
|
|
|
networkClient.instanceURL = identity.url
|
|
|
|
|
|
|
|
observation.catch { [weak self] error -> Empty<Identity, Never> in
|
|
|
|
self?.observationErrorsInput.send(error)
|
|
|
|
|
|
|
|
return Empty()
|
|
|
|
}
|
|
|
|
.assign(to: &$identity)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-08 23:08:47 +00:00
|
|
|
extension IdentityService {
|
2020-08-07 21:57:18 +00:00
|
|
|
var isAuthorized: Bool { networkClient.accessToken != nil }
|
|
|
|
|
|
|
|
func verifyCredentials() -> AnyPublisher<Void, Error> {
|
|
|
|
networkClient.request(AccountEndpoint.verifyCredentials)
|
|
|
|
.continuingIfWeakReferenceIsStillAlive(to: self)
|
|
|
|
.map { ($0, $1.identity.id) }
|
|
|
|
.flatMap(appEnvironment.identityDatabase.updateAccount)
|
|
|
|
.eraseToAnyPublisher()
|
|
|
|
}
|
|
|
|
|
|
|
|
func refreshServerPreferences() -> AnyPublisher<Void, Error> {
|
|
|
|
networkClient.request(PreferencesEndpoint.preferences)
|
|
|
|
.continuingIfWeakReferenceIsStillAlive(to: self)
|
|
|
|
.map { ($1.identity.preferences.updated(from: $0), $1.identity.id) }
|
|
|
|
.flatMap(appEnvironment.identityDatabase.updatePreferences)
|
|
|
|
.eraseToAnyPublisher()
|
|
|
|
}
|
|
|
|
|
|
|
|
func refreshInstance() -> AnyPublisher<Void, Error> {
|
|
|
|
networkClient.request(InstanceEndpoint.instance)
|
|
|
|
.continuingIfWeakReferenceIsStillAlive(to: self)
|
|
|
|
.map { ($0, $1.identity.id) }
|
|
|
|
.flatMap(appEnvironment.identityDatabase.updateInstance)
|
|
|
|
.eraseToAnyPublisher()
|
|
|
|
}
|
|
|
|
|
|
|
|
func identitiesObservation() -> AnyPublisher<[Identity], Error> {
|
2020-08-08 07:43:06 +00:00
|
|
|
appEnvironment.identityDatabase.identitiesObservation()
|
2020-08-07 21:57:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func recentIdentitiesObservation() -> AnyPublisher<[Identity], Error> {
|
|
|
|
appEnvironment.identityDatabase.recentIdentitiesObservation(excluding: identity.id)
|
|
|
|
}
|
|
|
|
|
|
|
|
func updatePreferences(_ preferences: Identity.Preferences) -> AnyPublisher<Void, Error> {
|
|
|
|
appEnvironment.identityDatabase.updatePreferences(preferences, forIdentityID: identity.id)
|
|
|
|
}
|
|
|
|
}
|