2021-02-03 11:52:47 +01:00
|
|
|
//
|
|
|
|
// APIService+HomeTimeline.swift
|
|
|
|
// Mastodon
|
|
|
|
//
|
|
|
|
// Created by MainasuK Cirno on 2021/2/3.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import Combine
|
|
|
|
import CoreData
|
|
|
|
import CoreDataStack
|
2021-02-04 07:45:44 +01:00
|
|
|
import CommonOSLog
|
2021-02-03 11:52:47 +01:00
|
|
|
import DateToolsSwift
|
|
|
|
import MastodonSDK
|
|
|
|
|
|
|
|
extension APIService {
|
|
|
|
|
|
|
|
func homeTimeline(
|
|
|
|
sinceID: Mastodon.Entity.Status.ID? = nil,
|
|
|
|
maxID: Mastodon.Entity.Status.ID? = nil,
|
2021-04-01 08:39:15 +02:00
|
|
|
limit: Int = onceRequestStatusMaxCount,
|
2021-02-04 07:45:44 +01:00
|
|
|
local: Bool? = nil,
|
2022-01-27 14:23:39 +01:00
|
|
|
authenticationBox: MastodonAuthenticationBox
|
|
|
|
) async throws -> Mastodon.Response.Content<[Mastodon.Entity.Status]> {
|
|
|
|
let domain = authenticationBox.domain
|
|
|
|
let authorization = authenticationBox.userAuthorization
|
2021-02-03 11:52:47 +01:00
|
|
|
let query = Mastodon.API.Timeline.HomeTimelineQuery(
|
|
|
|
maxID: maxID,
|
|
|
|
sinceID: sinceID,
|
|
|
|
minID: nil, // prefer sinceID
|
|
|
|
limit: limit,
|
2021-02-04 07:45:44 +01:00
|
|
|
local: local
|
2021-02-03 11:52:47 +01:00
|
|
|
)
|
|
|
|
|
2022-01-27 14:23:39 +01:00
|
|
|
let response = try await Mastodon.API.Timeline.home(
|
2021-02-03 11:52:47 +01:00
|
|
|
session: session,
|
|
|
|
domain: domain,
|
|
|
|
query: query,
|
|
|
|
authorization: authorization
|
2022-01-27 14:23:39 +01:00
|
|
|
).singleOutput()
|
|
|
|
|
|
|
|
let managedObjectContext = self.backgroundManagedObjectContext
|
|
|
|
try await managedObjectContext.performChanges {
|
|
|
|
guard let me = authenticationBox.authenticationRecord.object(in: managedObjectContext)?.user else {
|
|
|
|
assertionFailure()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// persist status
|
|
|
|
var statuses: [Status] = []
|
|
|
|
for entity in response.value {
|
|
|
|
let result = Persistence.Status.createOrMerge(
|
|
|
|
in: managedObjectContext,
|
|
|
|
context: Persistence.Status.PersistContext(
|
|
|
|
domain: domain,
|
|
|
|
entity: entity,
|
|
|
|
me: me,
|
|
|
|
statusCache: nil, // TODO: add cache
|
|
|
|
userCache: nil, // TODO: add cache
|
|
|
|
networkDate: response.networkDate
|
|
|
|
)
|
|
|
|
)
|
|
|
|
statuses.append(result.status)
|
|
|
|
}
|
|
|
|
|
|
|
|
// locate anchor status
|
|
|
|
let anchorStatus: Status? = {
|
|
|
|
guard let maxID = maxID else { return nil }
|
|
|
|
let request = Status.sortedFetchRequest
|
|
|
|
request.predicate = Status.predicate(domain: domain, id: maxID)
|
|
|
|
request.fetchLimit = 1
|
|
|
|
return try? managedObjectContext.fetch(request).first
|
|
|
|
}()
|
|
|
|
|
|
|
|
// update hasMore flag for anchor status
|
|
|
|
let acct = Feed.Acct.mastodon(domain: authenticationBox.domain, userID: authenticationBox.userID)
|
|
|
|
if let anchorStatus = anchorStatus,
|
|
|
|
let feed = anchorStatus.feed(kind: .home, acct: acct) {
|
|
|
|
feed.update(hasMore: false)
|
|
|
|
}
|
|
|
|
|
|
|
|
// persist Feed relationship
|
|
|
|
let sortedStatuses = statuses.sorted(by: { $0.createdAt < $1.createdAt })
|
|
|
|
let oldestStatus = sortedStatuses.first
|
|
|
|
for status in sortedStatuses {
|
|
|
|
let _feed = status.feed(kind: .home, acct: acct)
|
|
|
|
if let feed = _feed {
|
|
|
|
feed.update(updatedAt: response.networkDate)
|
|
|
|
} else {
|
|
|
|
let feedProperty = Feed.Property(
|
|
|
|
acct: acct,
|
|
|
|
kind: .home,
|
|
|
|
hasMore: false,
|
|
|
|
createdAt: status.createdAt,
|
|
|
|
updatedAt: response.networkDate
|
|
|
|
)
|
|
|
|
let feed = Feed.insert(into: managedObjectContext, property: feedProperty)
|
|
|
|
status.attach(feed: feed)
|
|
|
|
|
|
|
|
// set hasMore on oldest status if is new feed
|
|
|
|
if status === oldestStatus {
|
|
|
|
feed.update(hasMore: true)
|
|
|
|
}
|
2021-02-03 11:52:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-01-27 14:23:39 +01:00
|
|
|
|
|
|
|
return response
|
2021-02-03 11:52:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|