Suspend and resume account resources when moving between the background and foreground.

This commit is contained in:
Maurice Parker 2019-12-01 16:51:25 -06:00
parent 4f3f56bda5
commit d9f2e13732
3 changed files with 60 additions and 20 deletions

@ -24,6 +24,7 @@ public final class AccountManager: UnreadCountProvider {
private let defaultAccountFolderName = "OnMyMac" private let defaultAccountFolderName = "OnMyMac"
private let defaultAccountIdentifier = "OnMyMac" private let defaultAccountIdentifier = "OnMyMac"
public var isSuspended = false
public var isUnreadCountsInitialized: Bool { public var isUnreadCountsInitialized: Bool {
for account in activeAccounts { for account in activeAccounts {
if !account.isUnreadCountsInitialized { if !account.isUnreadCountsInitialized {
@ -164,10 +165,12 @@ public final class AccountManager: UnreadCountProvider {
} }
public func suspendAll() { public func suspendAll() {
isSuspended = true
accounts.forEach { $0.suspend() } accounts.forEach { $0.suspend() }
} }
public func resumeAll() { public func resumeAll() {
isSuspended = false
accounts.forEach { $0.resume() } accounts.forEach { $0.resume() }
} }

@ -46,6 +46,12 @@ final class FetchRequestOperation {
} }
} }
// The account manager may have been suspended while we were queued up
if AccountManager.shared.isSuspended {
callCompletionIfNeeded()
return
}
if isCanceled { if isCanceled {
callCompletionIfNeeded() callCompletionIfNeeded()
return return

@ -51,6 +51,18 @@ class AppDelegate: UIResponder, UIApplicationDelegate, UNUserNotificationCenterD
} }
} }
var isSyncArticleStatusRunning = false
var isWaitingForSyncTasks = false
var isAnySceneInForeground: Bool {
for scene in UIApplication.shared.connectedScenes {
if scene.activationState != .background {
return true
}
}
return false
}
override init() { override init() {
super.init() super.init()
appDelegate = self appDelegate = self
@ -108,7 +120,6 @@ class AppDelegate: UIResponder, UIApplicationDelegate, UNUserNotificationCenterD
func applicationWillTerminate(_ application: UIApplication) { func applicationWillTerminate(_ application: UIApplication) {
shuttingDown = true shuttingDown = true
AccountManager.shared.suspendAll()
} }
// MARK: Notifications // MARK: Notifications
@ -128,11 +139,15 @@ class AppDelegate: UIResponder, UIApplicationDelegate, UNUserNotificationCenterD
func prepareAccountsForBackground() { func prepareAccountsForBackground() {
syncTimer?.invalidate() syncTimer?.invalidate()
scheduleBackgroundFeedRefresh() scheduleBackgroundFeedRefresh()
waitForProgressToFinish()
syncArticleStatus() syncArticleStatus()
waitForSyncTasksToFinish()
} }
func prepareAccountsForForeground() { func prepareAccountsForForeground() {
if AccountManager.shared.isSuspended {
AccountManager.shared.resumeAll()
}
if let lastRefresh = AppDefaults.lastRefresh { if let lastRefresh = AppDefaults.lastRefresh {
if Date() > lastRefresh.addingTimeInterval(15 * 60) { if Date() > lastRefresh.addingTimeInterval(15 * 60) {
AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log) AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log)
@ -216,46 +231,58 @@ private extension AppDelegate {
// MARK: Go To Background // MARK: Go To Background
private extension AppDelegate { private extension AppDelegate {
func waitForProgressToFinish() { func waitForSyncTasksToFinish() {
let completeProcessing = { [unowned self] in guard !isAnySceneInForeground && !isWaitingForSyncTasks else { return }
UIApplication.shared.endBackgroundTask(self.waitBackgroundUpdateTask)
self.waitBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
}
self.waitBackgroundUpdateTask = UIApplication.shared.beginBackgroundTask { isWaitingForSyncTasks = true
completeProcessing()
self.waitBackgroundUpdateTask = UIApplication.shared.beginBackgroundTask { [weak self] in
guard let self = self else { return }
self.completeProcessing(true)
os_log("Accounts wait for progress terminated for running too long.", log: self.log, type: .info) os_log("Accounts wait for progress terminated for running too long.", log: self.log, type: .info)
} }
DispatchQueue.main.async { [weak self] in DispatchQueue.main.async { [weak self] in
self?.waitToComplete() { self?.waitToComplete() { [weak self] suspend in
completeProcessing() self?.completeProcessing(suspend)
} }
} }
} }
func waitToComplete(completion: @escaping () -> Void) { func waitToComplete(completion: @escaping (Bool) -> Void) {
guard UIApplication.shared.applicationState != .active else { guard !isAnySceneInForeground else {
os_log("App came back to forground, no longer waiting.", log: self.log, type: .info) os_log("App came back to forground, no longer waiting.", log: self.log, type: .info)
completion() completion(false)
return return
} }
if AccountManager.shared.refreshInProgress { if AccountManager.shared.refreshInProgress || isSyncArticleStatusRunning {
os_log("Waiting for refresh progress to finish...", log: self.log, type: .info) os_log("Waiting for sync to finish...", log: self.log, type: .info)
DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) { [weak self] in DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) { [weak self] in
self?.waitToComplete() { self?.waitToComplete(completion: completion)
completion()
}
} }
} else { } else {
os_log("Refresh progress complete.", log: self.log, type: .info) os_log("Refresh progress complete.", log: self.log, type: .info)
completion() completion(true)
} }
} }
func completeProcessing(_ suspend: Bool) {
if suspend {
AccountManager.shared.suspendAll()
}
UIApplication.shared.endBackgroundTask(self.waitBackgroundUpdateTask)
self.waitBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
isWaitingForSyncTasks = false
}
func syncArticleStatus() { func syncArticleStatus() {
guard !isSyncArticleStatusRunning else { return }
isSyncArticleStatusRunning = true
let completeProcessing = { [unowned self] in let completeProcessing = { [unowned self] in
self.isSyncArticleStatusRunning = false
UIApplication.shared.endBackgroundTask(self.syncBackgroundUpdateTask) UIApplication.shared.endBackgroundTask(self.syncBackgroundUpdateTask)
self.syncBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid self.syncBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
} }
@ -312,8 +339,12 @@ private extension AppDelegate {
os_log("Woken to perform account refresh.", log: self.log, type: .info) os_log("Woken to perform account refresh.", log: self.log, type: .info)
DispatchQueue.main.async { [weak task] in DispatchQueue.main.async { [weak task] in
if AccountManager.shared.isSuspended {
AccountManager.shared.resumeAll()
}
AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log) { AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log) {
AccountManager.shared.saveAll() AccountManager.shared.saveAll()
AccountManager.shared.suspendAll()
os_log("Account refresh operation completed.", log: self.log, type: .info) os_log("Account refresh operation completed.", log: self.log, type: .info)
task?.setTaskCompleted(success: true) task?.setTaskCompleted(success: true)
} }