forked from zelo72/mastodon-ios
120 lines
5.3 KiB
Swift
120 lines
5.3 KiB
Swift
//
|
|
// APIService+Notification.swift
|
|
// Mastodon
|
|
//
|
|
// Created by sxiaojian on 2021/4/13.
|
|
//
|
|
|
|
import Combine
|
|
import CoreData
|
|
import CoreDataStack
|
|
import Foundation
|
|
import MastodonSDK
|
|
import OSLog
|
|
|
|
extension APIService {
|
|
func allNotifications(
|
|
domain: String,
|
|
query: Mastodon.API.Notifications.Query,
|
|
mastodonAuthenticationBox: MastodonAuthenticationBox
|
|
) -> AnyPublisher<Mastodon.Response.Content<[Mastodon.Entity.Notification]>, Error> {
|
|
let authorization = mastodonAuthenticationBox.userAuthorization
|
|
let userID = mastodonAuthenticationBox.userID
|
|
return Mastodon.API.Notifications.getNotifications(
|
|
session: session,
|
|
domain: domain,
|
|
query: query,
|
|
authorization: authorization
|
|
)
|
|
.flatMap { response -> AnyPublisher<Mastodon.Response.Content<[Mastodon.Entity.Notification]>, Error> in
|
|
let log = OSLog.api
|
|
return self.backgroundManagedObjectContext.performChanges {
|
|
if query.maxID == nil {
|
|
let requestMastodonNotificationRequest = MastodonNotification.sortedFetchRequest
|
|
requestMastodonNotificationRequest.predicate = MastodonNotification.predicate(domain: domain, userID: userID)
|
|
let oldNotifications = self.backgroundManagedObjectContext.safeFetch(requestMastodonNotificationRequest)
|
|
oldNotifications.forEach { notification in
|
|
self.backgroundManagedObjectContext.delete(notification)
|
|
}
|
|
}
|
|
response.value.forEach { notification in
|
|
let (mastodonUser, _) = APIService.CoreData.createOrMergeMastodonUser(into: self.backgroundManagedObjectContext, for: nil, in: domain, entity: notification.account, userCache: nil, networkDate: Date(), log: log)
|
|
var status: Status?
|
|
if let statusEntity = notification.status {
|
|
let (statusInCoreData, _, _) = APIService.CoreData.createOrMergeStatus(
|
|
into: self.backgroundManagedObjectContext,
|
|
for: nil,
|
|
domain: domain,
|
|
entity: statusEntity,
|
|
statusCache: nil,
|
|
userCache: nil,
|
|
networkDate: Date(),
|
|
log: log
|
|
)
|
|
status = statusInCoreData
|
|
}
|
|
// use constrain to avoid repeated save
|
|
let property = MastodonNotification.Property(id: notification.id, typeRaw: notification.type.rawValue, account: mastodonUser, status: status, createdAt: notification.createdAt)
|
|
let notification = MastodonNotification.insert(into: self.backgroundManagedObjectContext, domain: domain, userID: userID, networkDate: response.networkDate, property: property)
|
|
os_log(.info, log: log, "%{public}s[%{public}ld], %{public}s: fetch mastodon user [%s](%s)", (#file as NSString).lastPathComponent, #line, #function, notification.typeRaw, notification.account.username)
|
|
}
|
|
}
|
|
.setFailureType(to: Error.self)
|
|
.tryMap { result -> Mastodon.Response.Content<[Mastodon.Entity.Notification]> in
|
|
switch result {
|
|
case .success:
|
|
return response
|
|
case .failure(let error):
|
|
throw error
|
|
}
|
|
}
|
|
.eraseToAnyPublisher()
|
|
}
|
|
.eraseToAnyPublisher()
|
|
}
|
|
|
|
func notification(
|
|
notificationID: Mastodon.Entity.Notification.ID,
|
|
mastodonAuthenticationBox: MastodonAuthenticationBox
|
|
) -> AnyPublisher<Mastodon.Response.Content<Mastodon.Entity.Notification>, Error> {
|
|
let domain = mastodonAuthenticationBox.domain
|
|
let authorization = mastodonAuthenticationBox.userAuthorization
|
|
|
|
return Mastodon.API.Notifications.getNotification(
|
|
session: session,
|
|
domain: domain,
|
|
notificationID: notificationID,
|
|
authorization: authorization
|
|
)
|
|
.flatMap { response -> AnyPublisher<Mastodon.Response.Content<Mastodon.Entity.Notification>, Error> in
|
|
guard let status = response.value.status else {
|
|
return Just(response)
|
|
.setFailureType(to: Error.self)
|
|
.eraseToAnyPublisher()
|
|
}
|
|
|
|
return APIService.Persist.persistStatus(
|
|
managedObjectContext: self.backgroundManagedObjectContext,
|
|
domain: domain,
|
|
query: nil,
|
|
response: response.map { _ in [status] },
|
|
persistType: .lookUp,
|
|
requestMastodonUserID: nil,
|
|
log: OSLog.api
|
|
)
|
|
.setFailureType(to: Error.self)
|
|
.tryMap { result -> Mastodon.Response.Content<Mastodon.Entity.Notification> in
|
|
switch result {
|
|
case .success:
|
|
return response
|
|
case .failure(let error):
|
|
throw error
|
|
}
|
|
}
|
|
.eraseToAnyPublisher()
|
|
}
|
|
.eraseToAnyPublisher()
|
|
}
|
|
|
|
}
|