2020-07-30 01:50:30 +02:00
|
|
|
// Copyright © 2020 Metabolist. All rights reserved.
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import Combine
|
|
|
|
import GRDB
|
|
|
|
|
2020-08-03 17:20:51 +02:00
|
|
|
enum IdentityDatabaseError: Error {
|
|
|
|
case identityNotFound
|
|
|
|
}
|
|
|
|
|
2020-07-30 01:50:30 +02:00
|
|
|
struct IdentityDatabase {
|
|
|
|
private let databaseQueue: DatabaseQueue
|
|
|
|
|
|
|
|
init(inMemory: Bool = false) throws {
|
|
|
|
guard
|
|
|
|
let documentsDirectory = NSSearchPathForDirectoriesInDomains(
|
|
|
|
.documentDirectory,
|
|
|
|
.userDomainMask, true)
|
|
|
|
.first
|
|
|
|
else { throw DatabaseError.documentsDirectoryNotFound }
|
|
|
|
|
|
|
|
if inMemory {
|
|
|
|
databaseQueue = DatabaseQueue()
|
|
|
|
} else {
|
|
|
|
databaseQueue = try DatabaseQueue(path: "\(documentsDirectory)/IdentityDatabase.sqlite3")
|
|
|
|
}
|
|
|
|
|
|
|
|
try Self.migrate(databaseQueue)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension IdentityDatabase {
|
2020-08-02 09:02:03 +02:00
|
|
|
func createIdentity(id: String, url: URL) -> AnyPublisher<Void, Error> {
|
2020-08-04 22:26:09 +02:00
|
|
|
databaseQueue.writePublisher(
|
|
|
|
updates: StoredIdentity(
|
|
|
|
id: id,
|
|
|
|
url: url,
|
|
|
|
lastUsedAt: Date(),
|
2020-08-07 03:41:59 +02:00
|
|
|
preferences: Identity.Preferences(),
|
2020-08-04 22:26:09 +02:00
|
|
|
instanceURI: nil).save)
|
2020-08-02 09:02:03 +02:00
|
|
|
.eraseToAnyPublisher()
|
2020-07-30 01:50:30 +02:00
|
|
|
}
|
|
|
|
|
2020-08-04 22:26:09 +02:00
|
|
|
func updateLastUsedAt(identityID: String) -> AnyPublisher<Void, Error> {
|
|
|
|
databaseQueue.writePublisher {
|
|
|
|
try StoredIdentity
|
|
|
|
.filter(Column("id") == identityID)
|
|
|
|
.updateAll($0, Column("lastUsedAt").set(to: Date()))
|
|
|
|
}
|
|
|
|
.eraseToAnyPublisher()
|
|
|
|
}
|
|
|
|
|
2020-08-02 09:02:03 +02:00
|
|
|
func updateInstance(_ instance: Instance, forIdentityID identityID: String) -> AnyPublisher<Void, Error> {
|
2020-07-30 01:50:30 +02:00
|
|
|
databaseQueue.writePublisher {
|
|
|
|
try Identity.Instance(
|
|
|
|
uri: instance.uri,
|
|
|
|
streamingAPI: instance.urls.streamingApi,
|
|
|
|
title: instance.title,
|
|
|
|
thumbnail: instance.thumbnail)
|
|
|
|
.save($0)
|
|
|
|
try StoredIdentity
|
|
|
|
.filter(Column("id") == identityID)
|
|
|
|
.updateAll($0, Column("instanceURI").set(to: instance.uri))
|
|
|
|
}
|
|
|
|
.eraseToAnyPublisher()
|
|
|
|
}
|
|
|
|
|
2020-08-02 09:02:03 +02:00
|
|
|
func updateAccount(_ account: Account, forIdentityID identityID: String) -> AnyPublisher<Void, Error> {
|
|
|
|
databaseQueue.writePublisher(
|
|
|
|
updates: Identity.Account(
|
2020-07-30 01:50:30 +02:00
|
|
|
id: account.id,
|
|
|
|
identityID: identityID,
|
|
|
|
username: account.username,
|
|
|
|
url: account.url,
|
|
|
|
avatar: account.avatar,
|
|
|
|
avatarStatic: account.avatarStatic,
|
|
|
|
header: account.header,
|
|
|
|
headerStatic: account.headerStatic)
|
2020-08-02 09:02:03 +02:00
|
|
|
.save)
|
|
|
|
.eraseToAnyPublisher()
|
2020-08-07 12:14:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func updatePreferences(_ preferences: Identity.Preferences,
|
|
|
|
forIdentityID identityID: String) -> AnyPublisher<Void, Error> {
|
|
|
|
databaseQueue.writePublisher {
|
|
|
|
let data = try StoredIdentity.databaseJSONEncoder(for: "preferences").encode(preferences)
|
|
|
|
|
|
|
|
try StoredIdentity
|
|
|
|
.filter(Column("id") == identityID)
|
|
|
|
.updateAll($0, Column("preferences").set(to: data))
|
|
|
|
}
|
|
|
|
.eraseToAnyPublisher()
|
2020-07-30 01:50:30 +02:00
|
|
|
}
|
|
|
|
|
2020-08-03 17:20:51 +02:00
|
|
|
func identityObservation(id: String) -> AnyPublisher<Identity, Error> {
|
2020-08-02 09:02:03 +02:00
|
|
|
ValueObservation.tracking(
|
|
|
|
StoredIdentity
|
|
|
|
.filter(Column("id") == id)
|
|
|
|
.including(optional: StoredIdentity.instance)
|
|
|
|
.including(optional: StoredIdentity.account)
|
|
|
|
.asRequest(of: IdentityResult.self)
|
|
|
|
.fetchOne)
|
|
|
|
.removeDuplicates()
|
|
|
|
.publisher(in: databaseQueue, scheduling: .immediate)
|
2020-08-03 17:20:51 +02:00
|
|
|
.tryMap {
|
|
|
|
guard let result = $0 else { throw IdentityDatabaseError.identityNotFound }
|
2020-08-02 09:02:03 +02:00
|
|
|
|
|
|
|
return Identity(result: result)
|
|
|
|
}
|
|
|
|
.eraseToAnyPublisher()
|
2020-07-30 01:50:30 +02:00
|
|
|
}
|
2020-08-04 22:26:09 +02:00
|
|
|
|
|
|
|
func identitiesObservation() -> AnyPublisher<[Identity], Error> {
|
|
|
|
ValueObservation.tracking(
|
|
|
|
StoredIdentity
|
|
|
|
.including(optional: StoredIdentity.instance)
|
|
|
|
.including(optional: StoredIdentity.account)
|
2020-08-05 13:48:50 +02:00
|
|
|
.asRequest(of: IdentityResult.self)
|
|
|
|
.fetchAll)
|
2020-08-04 22:26:09 +02:00
|
|
|
.removeDuplicates()
|
|
|
|
.publisher(in: databaseQueue, scheduling: .immediate)
|
|
|
|
.map { $0.map(Identity.init(result:)) }
|
|
|
|
.eraseToAnyPublisher()
|
|
|
|
}
|
|
|
|
|
2020-08-05 13:48:50 +02:00
|
|
|
func recentIdentitiesObservation(excluding: String) -> AnyPublisher<[Identity], Error> {
|
|
|
|
ValueObservation.tracking(
|
|
|
|
StoredIdentity
|
|
|
|
.filter(Column("id") != excluding)
|
|
|
|
.order(Column("lastUsedAt").desc)
|
|
|
|
.limit(10)
|
|
|
|
.including(optional: StoredIdentity.instance)
|
|
|
|
.including(optional: StoredIdentity.account)
|
|
|
|
.asRequest(of: IdentityResult.self)
|
|
|
|
.fetchAll)
|
2020-08-04 22:26:09 +02:00
|
|
|
.removeDuplicates()
|
|
|
|
.publisher(in: databaseQueue, scheduling: .immediate)
|
2020-08-05 13:48:50 +02:00
|
|
|
.map { $0.map(Identity.init(result:)) }
|
2020-08-04 22:26:09 +02:00
|
|
|
.eraseToAnyPublisher()
|
|
|
|
}
|
|
|
|
|
|
|
|
var mostRecentlyUsedIdentityID: String? {
|
|
|
|
try? databaseQueue.read(StoredIdentity.select(Column("id")).order(Column("lastUsedAt").desc).fetchOne)
|
|
|
|
}
|
2020-07-30 01:50:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private extension IdentityDatabase {
|
|
|
|
private static func migrate(_ writer: DatabaseWriter) throws {
|
|
|
|
var migrator = DatabaseMigrator()
|
|
|
|
|
|
|
|
migrator.registerMigration("createIdentities") { db in
|
|
|
|
try db.create(table: "instance", ifNotExists: true) { t in
|
|
|
|
t.column("uri", .text).notNull().primaryKey(onConflict: .replace)
|
|
|
|
t.column("streamingAPI", .text)
|
|
|
|
t.column("title", .text)
|
|
|
|
t.column("thumbnail", .text)
|
|
|
|
}
|
|
|
|
|
|
|
|
try db.create(table: "storedIdentity", ifNotExists: true) { t in
|
|
|
|
t.column("id", .text).notNull().primaryKey(onConflict: .replace)
|
|
|
|
t.column("url", .text).notNull()
|
2020-08-04 22:26:09 +02:00
|
|
|
t.column("lastUsedAt", .datetime).notNull()
|
2020-07-30 01:50:30 +02:00
|
|
|
t.column("instanceURI", .text)
|
|
|
|
.indexed()
|
|
|
|
.references("instance", column: "uri")
|
2020-08-07 03:41:59 +02:00
|
|
|
t.column("preferences", .blob).notNull()
|
2020-07-30 01:50:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
try db.create(table: "account", ifNotExists: true) { t in
|
|
|
|
t.column("id", .text).notNull().primaryKey(onConflict: .replace)
|
|
|
|
t.column("identityID", .text)
|
|
|
|
.notNull()
|
|
|
|
.indexed()
|
|
|
|
.references("storedIdentity", column: "id", onDelete: .cascade)
|
|
|
|
t.column("username", .text).notNull()
|
|
|
|
t.column("url", .text).notNull()
|
|
|
|
t.column("avatar", .text).notNull()
|
|
|
|
t.column("avatarStatic", .text).notNull()
|
|
|
|
t.column("header", .text).notNull()
|
|
|
|
t.column("headerStatic", .text).notNull()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
try migrator.migrate(writer)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private struct StoredIdentity: Codable, Hashable, TableRecord, FetchableRecord, PersistableRecord {
|
|
|
|
let id: String
|
|
|
|
let url: URL
|
2020-08-04 22:26:09 +02:00
|
|
|
let lastUsedAt: Date
|
2020-08-07 03:41:59 +02:00
|
|
|
let preferences: Identity.Preferences
|
2020-07-30 01:50:30 +02:00
|
|
|
let instanceURI: String?
|
|
|
|
}
|
|
|
|
|
|
|
|
extension StoredIdentity {
|
|
|
|
static let instance = belongsTo(Identity.Instance.self, key: "instance")
|
|
|
|
static let account = hasOne(Identity.Account.self, key: "account")
|
|
|
|
|
|
|
|
var instance: QueryInterfaceRequest<Identity.Instance> {
|
|
|
|
request(for: Self.instance)
|
|
|
|
}
|
|
|
|
|
|
|
|
var account: QueryInterfaceRequest<Identity.Account> {
|
|
|
|
request(for: Self.account)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private struct IdentityResult: Codable, Hashable, FetchableRecord {
|
|
|
|
let identity: StoredIdentity
|
|
|
|
let instance: Identity.Instance?
|
|
|
|
let account: Identity.Account?
|
|
|
|
}
|
|
|
|
|
|
|
|
private extension Identity {
|
|
|
|
init(result: IdentityResult) {
|
2020-08-04 22:26:09 +02:00
|
|
|
self.init(
|
|
|
|
id: result.identity.id,
|
|
|
|
url: result.identity.url,
|
|
|
|
lastUsedAt: result.identity.lastUsedAt,
|
2020-08-07 03:41:59 +02:00
|
|
|
preferences: result.identity.preferences,
|
2020-08-04 22:26:09 +02:00
|
|
|
instance: result.instance,
|
|
|
|
account: result.account)
|
2020-07-30 01:50:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension Identity.Instance: TableRecord, FetchableRecord, PersistableRecord {}
|
|
|
|
|
|
|
|
extension Identity.Account: TableRecord, FetchableRecord, PersistableRecord {}
|