forked from zelo72/mastodon-ios
168 lines
7.4 KiB
Swift
168 lines
7.4 KiB
Swift
//
|
|
// APIService+Favorite.swift
|
|
// Mastodon
|
|
//
|
|
// Created by sxiaojian on 2021/2/8.
|
|
//
|
|
|
|
|
|
import Foundation
|
|
import Combine
|
|
import MastodonSDK
|
|
import CoreData
|
|
import CoreDataStack
|
|
import CommonOSLog
|
|
|
|
extension APIService {
|
|
|
|
// make local state change only
|
|
func like(
|
|
tootObjectID: NSManagedObjectID,
|
|
mastodonUserObjectID: NSManagedObjectID,
|
|
favoriteKind: Mastodon.API.Favorites.FavoriteKind
|
|
) -> AnyPublisher<Toot.ID, Error> {
|
|
var _targetTootID: Toot.ID?
|
|
let managedObjectContext = backgroundManagedObjectContext
|
|
return managedObjectContext.performChanges {
|
|
let toot = managedObjectContext.object(with: tootObjectID) as! Toot
|
|
let mastodonUser = managedObjectContext.object(with: mastodonUserObjectID) as! MastodonUser
|
|
let targetToot = toot.reblog ?? toot
|
|
let targetTootID = targetToot.id
|
|
_targetTootID = targetTootID
|
|
|
|
targetToot.update(liked: favoriteKind == .create, mastodonUser: mastodonUser)
|
|
|
|
}
|
|
.tryMap { result in
|
|
switch result {
|
|
case .success:
|
|
guard let targetTootID = _targetTootID else {
|
|
throw APIError.implicit(.badRequest)
|
|
}
|
|
return targetTootID
|
|
|
|
case .failure(let error):
|
|
assertionFailure(error.localizedDescription)
|
|
throw error
|
|
}
|
|
}
|
|
.eraseToAnyPublisher()
|
|
}
|
|
|
|
// send favorite request to remote
|
|
func like(
|
|
statusID: Mastodon.Entity.Status.ID,
|
|
favoriteKind: Mastodon.API.Favorites.FavoriteKind,
|
|
mastodonAuthenticationBox: AuthenticationService.MastodonAuthenticationBox
|
|
) -> AnyPublisher<Mastodon.Response.Content<Mastodon.Entity.Status>, Error> {
|
|
let authorization = mastodonAuthenticationBox.userAuthorization
|
|
let requestMastodonUserID = mastodonAuthenticationBox.userID
|
|
return Mastodon.API.Favorites.favorites(domain: mastodonAuthenticationBox.domain, statusID: statusID, session: session, authorization: authorization, favoriteKind: favoriteKind)
|
|
.map { response -> AnyPublisher<Mastodon.Response.Content<Mastodon.Entity.Status>, Error> in
|
|
let log = OSLog.api
|
|
let entity = response.value
|
|
let managedObjectContext = self.backgroundManagedObjectContext
|
|
|
|
return managedObjectContext.performChanges {
|
|
let _requestMastodonUser: MastodonUser? = {
|
|
let request = MastodonUser.sortedFetchRequest
|
|
request.predicate = MastodonUser.predicate(domain: mastodonAuthenticationBox.domain, id: requestMastodonUserID)
|
|
request.fetchLimit = 1
|
|
request.returnsObjectsAsFaults = false
|
|
do {
|
|
return try managedObjectContext.fetch(request).first
|
|
} catch {
|
|
assertionFailure(error.localizedDescription)
|
|
return nil
|
|
}
|
|
}()
|
|
let _oldToot: Toot? = {
|
|
let request = Toot.sortedFetchRequest
|
|
request.predicate = Toot.predicate(domain: mastodonAuthenticationBox.domain, id: statusID)
|
|
request.fetchLimit = 1
|
|
request.returnsObjectsAsFaults = false
|
|
request.relationshipKeyPathsForPrefetching = [#keyPath(Toot.reblog)]
|
|
do {
|
|
return try managedObjectContext.fetch(request).first
|
|
} catch {
|
|
assertionFailure(error.localizedDescription)
|
|
return nil
|
|
}
|
|
}()
|
|
|
|
guard let requestMastodonUser = _requestMastodonUser,
|
|
let oldToot = _oldToot else {
|
|
assertionFailure()
|
|
return
|
|
}
|
|
APIService.CoreData.merge(toot: oldToot, entity: entity, requestMastodonUser: requestMastodonUser, domain: mastodonAuthenticationBox.domain, networkDate: response.networkDate)
|
|
if favoriteKind == .destroy {
|
|
oldToot.update(favouritesCount: NSNumber(value: max(0, oldToot.favouritesCount.intValue - 1)))
|
|
}
|
|
os_log(.info, log: log, "%{public}s[%{public}ld], %{public}s: did update toot %{public}s like status to: %{public}s. now %ld likes", ((#file as NSString).lastPathComponent), #line, #function, entity.id, entity.favourited.flatMap { $0 ? "like" : "unlike" } ?? "<nil>", entity.favouritesCount )
|
|
}
|
|
.setFailureType(to: Error.self)
|
|
.tryMap { result -> Mastodon.Response.Content<Mastodon.Entity.Status> in
|
|
switch result {
|
|
case .success:
|
|
return response
|
|
case .failure(let error):
|
|
throw error
|
|
}
|
|
}
|
|
.eraseToAnyPublisher()
|
|
}
|
|
.switchToLatest()
|
|
.handleEvents(receiveCompletion: { completion in
|
|
switch completion {
|
|
case .failure(let error):
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: error:", ((#file as NSString).lastPathComponent), #line, #function)
|
|
debugPrint(error)
|
|
case .finished:
|
|
break
|
|
}
|
|
})
|
|
.eraseToAnyPublisher()
|
|
}
|
|
|
|
}
|
|
|
|
extension APIService {
|
|
func likeList(
|
|
limit: Int = onceRequestTootMaxCount,
|
|
userID: String,
|
|
maxID: String? = nil,
|
|
mastodonAuthenticationBox: AuthenticationService.MastodonAuthenticationBox
|
|
) -> AnyPublisher<Mastodon.Response.Content<[Mastodon.Entity.Status]>, Error> {
|
|
|
|
let requestMastodonUserID = mastodonAuthenticationBox.userID
|
|
let query = Mastodon.API.Favorites.ListQuery(limit: limit, minID: nil, maxID: maxID)
|
|
return Mastodon.API.Favorites.favoritedStatus(domain: mastodonAuthenticationBox.domain, session: session, authorization: mastodonAuthenticationBox.userAuthorization, query: query)
|
|
.map { response -> AnyPublisher<Mastodon.Response.Content<[Mastodon.Entity.Status]>, Error> in
|
|
let log = OSLog.api
|
|
|
|
return APIService.Persist.persistTimeline(
|
|
managedObjectContext: self.backgroundManagedObjectContext,
|
|
domain: mastodonAuthenticationBox.domain,
|
|
query: query,
|
|
response: response,
|
|
persistType: .likeList,
|
|
requestMastodonUserID: requestMastodonUserID,
|
|
log: log
|
|
)
|
|
.setFailureType(to: Error.self)
|
|
.tryMap { result -> Mastodon.Response.Content<[Mastodon.Entity.Status]> in
|
|
switch result {
|
|
case .success:
|
|
return response
|
|
case .failure(let error):
|
|
throw error
|
|
}
|
|
}
|
|
.eraseToAnyPublisher()
|
|
}
|
|
.switchToLatest()
|
|
.eraseToAnyPublisher()
|
|
}
|
|
}
|