2019-10-18 23:21:02 +02:00
|
|
|
//
|
|
|
|
// FeedlySyncAllOperation.swift
|
|
|
|
// Account
|
|
|
|
//
|
|
|
|
// Created by Kiel Gillard on 19/9/19.
|
|
|
|
// Copyright © 2019 Ranchero Software, LLC. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import os.log
|
|
|
|
import SyncDatabase
|
|
|
|
|
|
|
|
/// Single responsibility is to compose the operations necessary to get the entire set of articles, feeds and folders with the statuses the user expects between now and a certain date in the past.
|
|
|
|
final class FeedlySyncAllOperation: FeedlyOperation {
|
|
|
|
private let operationQueue: OperationQueue
|
|
|
|
private let log: OSLog
|
|
|
|
let syncUUID: UUID
|
|
|
|
|
|
|
|
var syncCompletionHandler: ((Result<Void, Error>) -> ())?
|
|
|
|
|
|
|
|
init(account: Account, credentials: Credentials, lastSuccessfulFetchStartDate: Date?, markArticlesService: FeedlyMarkArticlesService, getUnreadService: FeedlyGetStreamIdsService, getCollectionsService: FeedlyGetCollectionsService, getStreamContentsService: FeedlyGetStreamContentsService, getStarredArticlesService: FeedlyGetStreamContentsService, database: SyncDatabase, log: OSLog) {
|
|
|
|
self.syncUUID = UUID()
|
|
|
|
self.log = log
|
|
|
|
self.operationQueue = OperationQueue()
|
|
|
|
self.operationQueue.isSuspended = true
|
|
|
|
|
|
|
|
super.init()
|
|
|
|
|
|
|
|
// Send any read/unread/starred article statuses to Feedly before anything else.
|
|
|
|
let sendArticleStatuses = FeedlySendArticleStatusesOperation(database: database, service: markArticlesService, log: log)
|
|
|
|
sendArticleStatuses.delegate = self
|
|
|
|
self.operationQueue.addOperation(sendArticleStatuses)
|
|
|
|
|
|
|
|
// Get each page of unread article ids in the global.all stream for the last 31 days (nil = Feedly API default).
|
|
|
|
let getUnread = FeedlySyncUnreadStatusesOperation(account: account, credentials: credentials, service: getUnreadService, newerThan: nil, log: log)
|
|
|
|
getUnread.delegate = self
|
|
|
|
getUnread.addDependency(sendArticleStatuses)
|
|
|
|
self.operationQueue.addOperation(getUnread)
|
|
|
|
|
|
|
|
// Get all the Collections the user has.
|
|
|
|
let getCollections = FeedlyGetCollectionsOperation(service: getCollectionsService, log: log)
|
|
|
|
getCollections.delegate = self
|
|
|
|
getCollections.addDependency(sendArticleStatuses)
|
|
|
|
self.operationQueue.addOperation(getCollections)
|
|
|
|
|
|
|
|
// Ensure a folder exists for each Collection, removing Folders without a corresponding Collection.
|
|
|
|
let mirrorCollectionsAsFolders = FeedlyMirrorCollectionsAsFoldersOperation(account: account, collectionsProvider: getCollections, log: log)
|
|
|
|
mirrorCollectionsAsFolders.delegate = self
|
|
|
|
mirrorCollectionsAsFolders.addDependency(getCollections)
|
|
|
|
self.operationQueue.addOperation(mirrorCollectionsAsFolders)
|
|
|
|
|
|
|
|
// Ensure feeds are created and grouped by their folders.
|
|
|
|
let createFeedsOperation = FeedlyCreateFeedsForCollectionFoldersOperation(account: account, feedsAndFoldersProvider: mirrorCollectionsAsFolders, log: log)
|
|
|
|
createFeedsOperation.delegate = self
|
|
|
|
createFeedsOperation.addDependency(mirrorCollectionsAsFolders)
|
|
|
|
self.operationQueue.addOperation(createFeedsOperation)
|
|
|
|
|
|
|
|
// Get each page of the global.all stream until we get either the content from the last sync or the last 31 days.
|
|
|
|
let getStreamContents = FeedlySyncStreamContentsOperation(account: account, credentials: credentials, service: getStreamContentsService, newerThan: lastSuccessfulFetchStartDate, log: log)
|
|
|
|
getStreamContents.delegate = self
|
|
|
|
getStreamContents.addDependency(getCollections)
|
|
|
|
getStreamContents.addDependency(getUnread)
|
|
|
|
getStreamContents.addDependency(createFeedsOperation)
|
|
|
|
self.operationQueue.addOperation(getStreamContents)
|
|
|
|
|
|
|
|
// Get each and every starred article.
|
|
|
|
let syncStarred = FeedlySyncStarredArticlesOperation(account: account, credentials: credentials, service: getStarredArticlesService, log: log)
|
|
|
|
syncStarred.addDependency(createFeedsOperation)
|
|
|
|
self.operationQueue.addOperation(syncStarred)
|
|
|
|
|
|
|
|
// Once this operation's dependencies, their dependencies etc finish, we can finish.
|
|
|
|
let finishOperation = FeedlyCheckpointOperation()
|
|
|
|
finishOperation.checkpointDelegate = self
|
|
|
|
finishOperation.addDependency(getStreamContents)
|
|
|
|
finishOperation.addDependency(syncStarred)
|
|
|
|
|
|
|
|
self.operationQueue.addOperation(finishOperation)
|
|
|
|
}
|
|
|
|
|
|
|
|
convenience init(account: Account, credentials: Credentials, caller: FeedlyAPICaller, database: SyncDatabase, lastSuccessfulFetchStartDate: Date?, log: OSLog) {
|
|
|
|
|
|
|
|
let newerThan: Date? = {
|
|
|
|
if let date = lastSuccessfulFetchStartDate {
|
|
|
|
return date
|
|
|
|
} else {
|
|
|
|
return Calendar.current.date(byAdding: .day, value: -31, to: Date())
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
self.init(account: account, credentials: credentials, lastSuccessfulFetchStartDate: newerThan, markArticlesService: caller, getUnreadService: caller, getCollectionsService: caller, getStreamContentsService: caller, getStarredArticlesService: caller, database: database, log: log)
|
|
|
|
}
|
|
|
|
|
|
|
|
override func cancel() {
|
|
|
|
os_log(.debug, log: log, "Cancelling sync %{public}@", syncUUID.uuidString)
|
|
|
|
self.operationQueue.cancelAllOperations()
|
|
|
|
|
2019-11-14 21:59:44 +01:00
|
|
|
super.cancel()
|
2019-10-18 23:21:02 +02:00
|
|
|
|
2019-11-14 21:59:44 +01:00
|
|
|
didFinish()
|
2019-11-14 23:32:02 +01:00
|
|
|
|
|
|
|
// Operation should silently cancel.
|
|
|
|
syncCompletionHandler = nil
|
2019-10-18 23:21:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
override func main() {
|
|
|
|
guard !isCancelled else {
|
2019-11-06 01:22:10 +01:00
|
|
|
// override of cancel calls didFinish().
|
2019-10-18 23:21:02 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
os_log(.debug, log: log, "Starting sync %{public}@", syncUUID.uuidString)
|
|
|
|
operationQueue.isSuspended = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension FeedlySyncAllOperation: FeedlyCheckpointOperationDelegate {
|
|
|
|
|
|
|
|
func feedlyCheckpointOperationDidReachCheckpoint(_ operation: FeedlyCheckpointOperation) {
|
|
|
|
assert(Thread.isMainThread)
|
|
|
|
os_log(.debug, log: self.log, "Sync completed: %{public}@", syncUUID.uuidString)
|
|
|
|
|
|
|
|
syncCompletionHandler?(.success(()))
|
|
|
|
syncCompletionHandler = nil
|
|
|
|
|
|
|
|
didFinish()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension FeedlySyncAllOperation: FeedlyOperationDelegate {
|
|
|
|
|
|
|
|
func feedlyOperation(_ operation: FeedlyOperation, didFailWith error: Error) {
|
|
|
|
assert(Thread.isMainThread)
|
|
|
|
os_log(.debug, log: log, "%{public}@ failed with error: %{public}@.", operation, error.localizedDescription)
|
|
|
|
|
|
|
|
syncCompletionHandler?(.failure(error))
|
|
|
|
syncCompletionHandler = nil
|
|
|
|
|
|
|
|
cancel()
|
|
|
|
}
|
|
|
|
}
|