Fix numerous concurrency warnings.
This commit is contained in:
parent
4b0e7addc9
commit
d1dc4cceec
@ -22,8 +22,7 @@ public struct AccountSyncError {
|
|||||||
init(account: Account, error: Error) {
|
init(account: Account, error: Error) {
|
||||||
self.account = account
|
self.account = account
|
||||||
self.error = error
|
self.error = error
|
||||||
os_log(.error, log: AccountSyncError.log, "%@", error.localizedDescription)
|
os_log(.error, log: Self.log, "%@", error.localizedDescription)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,16 +36,16 @@ class CloudKitRemoteNotificationOperation: MainThreadOperation {
|
|||||||
self.operationDelegate?.operationDidComplete(self)
|
self.operationDelegate?.operationDidComplete(self)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
os_log(.debug, log: log, "Processing remote notification...")
|
os_log(.debug, log: log, "Processing remote notification...")
|
||||||
|
|
||||||
accountZone.receiveRemoteNotification(userInfo: userInfo) {
|
Task { @MainActor in
|
||||||
articlesZone.receiveRemoteNotification(userInfo: self.userInfo) {
|
|
||||||
os_log(.debug, log: self.log, "Done processing remote notification.")
|
await accountZone.receiveRemoteNotification(userInfo: userInfo)
|
||||||
self.operationDelegate?.operationDidComplete(self)
|
await articlesZone.receiveRemoteNotification(userInfo: self.userInfo)
|
||||||
}
|
|
||||||
|
os_log(.debug, log: self.log, "Done processing remote notification.")
|
||||||
|
self.operationDelegate?.operationDidComplete(self)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -804,7 +804,7 @@ extension FeedlyAPICaller: FeedlyMarkArticlesService {
|
|||||||
var entryIDs: [String]
|
var entryIDs: [String]
|
||||||
}
|
}
|
||||||
|
|
||||||
func mark(_ articleIDs: Set<String>, as action: FeedlyMarkAction, completion: @escaping (Result<Void, Error>) -> ()) {
|
func mark(_ articleIDs: Set<String>, as action: FeedlyMarkAction, completion: @escaping @Sendable (Result<Void, Error>) -> ()) {
|
||||||
guard !isSuspended else {
|
guard !isSuspended else {
|
||||||
return DispatchQueue.main.async {
|
return DispatchQueue.main.async {
|
||||||
completion(.failure(TransportError.suspended))
|
completion(.failure(TransportError.suspended))
|
||||||
|
@ -13,8 +13,9 @@ import Core
|
|||||||
import Feedly
|
import Feedly
|
||||||
|
|
||||||
public protocol OAuthAccountAuthorizationOperationDelegate: AnyObject {
|
public protocol OAuthAccountAuthorizationOperationDelegate: AnyObject {
|
||||||
func oauthAccountAuthorizationOperation(_ operation: OAuthAccountAuthorizationOperation, didCreate account: Account)
|
|
||||||
func oauthAccountAuthorizationOperation(_ operation: OAuthAccountAuthorizationOperation, didFailWith error: Error)
|
@MainActor func oauthAccountAuthorizationOperation(_ operation: OAuthAccountAuthorizationOperation, didCreate account: Account)
|
||||||
|
@MainActor func oauthAccountAuthorizationOperation(_ operation: OAuthAccountAuthorizationOperation, didFailWith error: Error)
|
||||||
}
|
}
|
||||||
|
|
||||||
public enum OAuthAccountAuthorizationOperationError: LocalizedError {
|
public enum OAuthAccountAuthorizationOperationError: LocalizedError {
|
||||||
|
@ -98,7 +98,7 @@ public struct OAuthAuthorizationErrorResponse: Error {
|
|||||||
|
|
||||||
/// Error values as enumerated in section 4.1.2.1 of the OAuth 2.0 Authorization Framework.
|
/// Error values as enumerated in section 4.1.2.1 of the OAuth 2.0 Authorization Framework.
|
||||||
/// https://tools.ietf.org/html/rfc6749#section-4.1.2.1
|
/// https://tools.ietf.org/html/rfc6749#section-4.1.2.1
|
||||||
public enum OAuthAuthorizationError: String {
|
public enum OAuthAuthorizationError: String, Sendable {
|
||||||
case invalidRequest = "invalid_request"
|
case invalidRequest = "invalid_request"
|
||||||
case unauthorizedClient = "unauthorized_client"
|
case unauthorizedClient = "unauthorized_client"
|
||||||
case accessDenied = "access_denied"
|
case accessDenied = "access_denied"
|
||||||
|
@ -119,6 +119,15 @@ public extension CloudKitZone {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func receiveRemoteNotification(userInfo: [AnyHashable : Any]) async {
|
||||||
|
|
||||||
|
await withCheckedContinuation { continuation in
|
||||||
|
self.receiveRemoteNotification(userInfo: userInfo) {
|
||||||
|
continuation.resume()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func receiveRemoteNotification(userInfo: [AnyHashable : Any], completion: @escaping () -> Void) {
|
func receiveRemoteNotification(userInfo: [AnyHashable : Any], completion: @escaping () -> Void) {
|
||||||
let note = CKRecordZoneNotification(fromRemoteNotificationDictionary: userInfo)
|
let note = CKRecordZoneNotification(fromRemoteNotificationDictionary: userInfo)
|
||||||
guard note?.recordZoneID?.zoneName == zoneID.zoneName else {
|
guard note?.recordZoneID?.zoneName == zoneID.zoneName else {
|
||||||
|
@ -128,8 +128,11 @@ import AppKit
|
|||||||
func relaunchFromURL(_ appURL: URL) {
|
func relaunchFromURL(_ appURL: URL) {
|
||||||
// Relaunching is best achieved by requesting that the system launch the app
|
// Relaunching is best achieved by requesting that the system launch the app
|
||||||
// at the given URL with the "new instance" option to prevent it simply reactivating us.
|
// at the given URL with the "new instance" option to prevent it simply reactivating us.
|
||||||
let _ = try? NSWorkspace.shared.launchApplication(at: appURL, options: .newInstance, configuration: [:])
|
let configuration = NSWorkspace.OpenConfiguration()
|
||||||
NSApp.terminate(self)
|
configuration.createsNewApplicationInstance = true
|
||||||
|
NSWorkspace.shared.openApplication(at: appURL, configuration: configuration) { _,_ in
|
||||||
|
NSApp.terminate(self)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func defaultHandler() {
|
func defaultHandler() {
|
||||||
|
@ -32,5 +32,6 @@ public enum FeedlyMarkAction: String, Sendable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public protocol FeedlyMarkArticlesService: AnyObject {
|
public protocol FeedlyMarkArticlesService: AnyObject {
|
||||||
func mark(_ articleIDs: Set<String>, as action: FeedlyMarkAction, completion: @escaping (Result<Void, Error>) -> ())
|
|
||||||
|
func mark(_ articleIDs: Set<String>, as action: FeedlyMarkAction, completion: @escaping @Sendable (Result<Void, Error>) -> ())
|
||||||
}
|
}
|
||||||
|
@ -54,10 +54,13 @@ struct Browser {
|
|||||||
|
|
||||||
NSWorkspace.shared.open(preparedURL, configuration: configuration) { (runningApplication, error) in
|
NSWorkspace.shared.open(preparedURL, configuration: configuration) { (runningApplication, error) in
|
||||||
guard error != nil else { return }
|
guard error != nil else { return }
|
||||||
if let defaultBrowser = defaultBrowser {
|
|
||||||
defaultBrowser.openURL(url, inBackground: inBackground)
|
Task { @MainActor in
|
||||||
} else {
|
if let defaultBrowser = defaultBrowser {
|
||||||
MacWebBrowser.openURL(url, inBackground: inBackground)
|
defaultBrowser.openURL(url, inBackground: inBackground)
|
||||||
|
} else {
|
||||||
|
MacWebBrowser.openURL(url, inBackground: inBackground)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -10,15 +10,15 @@ import Foundation
|
|||||||
import WebKit
|
import WebKit
|
||||||
import Articles
|
import Articles
|
||||||
|
|
||||||
class DetailIconSchemeHandler: NSObject, WKURLSchemeHandler {
|
final class DetailIconSchemeHandler: NSObject, WKURLSchemeHandler {
|
||||||
|
|
||||||
var currentArticle: Article?
|
var currentArticle: Article?
|
||||||
|
|
||||||
func webView(_ webView: WKWebView, start urlSchemeTask: WKURLSchemeTask) {
|
func webView(_ webView: WKWebView, start urlSchemeTask: WKURLSchemeTask) {
|
||||||
|
|
||||||
Task { @MainActor in
|
Task { @MainActor in
|
||||||
|
|
||||||
guard let responseURL = urlSchemeTask.request.url, let iconImage = self.currentArticle?.iconImage() else {
|
guard let responseURL = urlSchemeTask.request.url, let iconImage = self.currentArticle?.iconImage() else {
|
||||||
urlSchemeTask.didFailWithError(URLError(.fileDoesNotExist))
|
urlSchemeTask.didFailWithError(URLError(.fileDoesNotExist))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -168,29 +168,33 @@ extension Feed: PasteboardWriterOwner {
|
|||||||
|
|
||||||
// MARK: - NSPasteboardWriting
|
// MARK: - NSPasteboardWriting
|
||||||
|
|
||||||
func writableTypes(for pasteboard: NSPasteboard) -> [NSPasteboard.PasteboardType] {
|
nonisolated func writableTypes(for pasteboard: NSPasteboard) -> [NSPasteboard.PasteboardType] {
|
||||||
|
|
||||||
return [FeedPasteboardWriter.feedUTIType, .URL, .string, FeedPasteboardWriter.feedUTIInternalType]
|
MainActor.assumeIsolated {
|
||||||
|
return [FeedPasteboardWriter.feedUTIType, .URL, .string, FeedPasteboardWriter.feedUTIInternalType]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func pasteboardPropertyList(forType type: NSPasteboard.PasteboardType) -> Any? {
|
nonisolated func pasteboardPropertyList(forType type: NSPasteboard.PasteboardType) -> Any? {
|
||||||
|
|
||||||
let plist: Any?
|
MainActor.assumeIsolated {
|
||||||
|
let plist: Any?
|
||||||
switch type {
|
|
||||||
case .string:
|
switch type {
|
||||||
plist = feed.nameForDisplay
|
case .string:
|
||||||
case .URL:
|
plist = feed.nameForDisplay
|
||||||
plist = feed.url
|
case .URL:
|
||||||
case FeedPasteboardWriter.feedUTIType:
|
plist = feed.url
|
||||||
plist = exportDictionary
|
case FeedPasteboardWriter.feedUTIType:
|
||||||
case FeedPasteboardWriter.feedUTIInternalType:
|
plist = exportDictionary
|
||||||
plist = internalDictionary
|
case FeedPasteboardWriter.feedUTIInternalType:
|
||||||
default:
|
plist = internalDictionary
|
||||||
plist = nil
|
default:
|
||||||
|
plist = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return plist
|
||||||
}
|
}
|
||||||
|
|
||||||
return plist
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -104,25 +104,29 @@ extension Folder: PasteboardWriterOwner {
|
|||||||
|
|
||||||
// MARK: - NSPasteboardWriting
|
// MARK: - NSPasteboardWriting
|
||||||
|
|
||||||
func writableTypes(for pasteboard: NSPasteboard) -> [NSPasteboard.PasteboardType] {
|
nonisolated func writableTypes(for pasteboard: NSPasteboard) -> [NSPasteboard.PasteboardType] {
|
||||||
|
|
||||||
return [.string, FolderPasteboardWriter.folderUTIInternalType]
|
MainActor.assumeIsolated {
|
||||||
|
return [.string, FolderPasteboardWriter.folderUTIInternalType]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func pasteboardPropertyList(forType type: NSPasteboard.PasteboardType) -> Any? {
|
nonisolated func pasteboardPropertyList(forType type: NSPasteboard.PasteboardType) -> Any? {
|
||||||
|
|
||||||
let plist: Any?
|
MainActor.assumeIsolated {
|
||||||
|
let plist: Any?
|
||||||
switch type {
|
|
||||||
case .string:
|
switch type {
|
||||||
plist = folder.nameForDisplay
|
case .string:
|
||||||
case FolderPasteboardWriter.folderUTIInternalType:
|
plist = folder.nameForDisplay
|
||||||
plist = internalDictionary
|
case FolderPasteboardWriter.folderUTIInternalType:
|
||||||
default:
|
plist = internalDictionary
|
||||||
plist = nil
|
default:
|
||||||
|
plist = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return plist
|
||||||
}
|
}
|
||||||
|
|
||||||
return plist
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2350,8 +2350,8 @@
|
|||||||
849C64611ED37A5D003D8FC0 /* Products */,
|
849C64611ED37A5D003D8FC0 /* Products */,
|
||||||
51C452B22265141B00C03939 /* Frameworks */,
|
51C452B22265141B00C03939 /* Frameworks */,
|
||||||
84A699132BC34E8500605AB8 /* ArticleExtractor */,
|
84A699132BC34E8500605AB8 /* ArticleExtractor */,
|
||||||
51CD32C624D2DEF9009ABAEF /* Account */,
|
|
||||||
84FB9FAE2BC3494B00B7AFC3 /* FeedFinder */,
|
84FB9FAE2BC3494B00B7AFC3 /* FeedFinder */,
|
||||||
|
51CD32C624D2DEF9009ABAEF /* Account */,
|
||||||
84A699182BC3524C00605AB8 /* LocalAccount */,
|
84A699182BC3524C00605AB8 /* LocalAccount */,
|
||||||
84FB9FAD2BC344F800B7AFC3 /* Feedbin */,
|
84FB9FAD2BC344F800B7AFC3 /* Feedbin */,
|
||||||
84A699192BC36EDB00605AB8 /* Feedly */,
|
84A699192BC36EDB00605AB8 /* Feedly */,
|
||||||
|
@ -20,23 +20,25 @@ import Account
|
|||||||
|
|
||||||
// MARK: - NSPasteboardWriting
|
// MARK: - NSPasteboardWriting
|
||||||
|
|
||||||
func writableTypes(for pasteboard: NSPasteboard) -> [NSPasteboard.PasteboardType] {
|
nonisolated func writableTypes(for pasteboard: NSPasteboard) -> [NSPasteboard.PasteboardType] {
|
||||||
|
|
||||||
return [.string]
|
return [.string]
|
||||||
}
|
}
|
||||||
|
|
||||||
func pasteboardPropertyList(forType type: NSPasteboard.PasteboardType) -> Any? {
|
nonisolated func pasteboardPropertyList(forType type: NSPasteboard.PasteboardType) -> Any? {
|
||||||
|
|
||||||
let plist: Any?
|
MainActor.assumeIsolated {
|
||||||
|
let plist: Any?
|
||||||
switch type {
|
|
||||||
case .string:
|
switch type {
|
||||||
plist = smartFeed.nameForDisplay
|
case .string:
|
||||||
default:
|
plist = smartFeed.nameForDisplay
|
||||||
plist = nil
|
default:
|
||||||
|
plist = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return plist
|
||||||
}
|
}
|
||||||
|
|
||||||
return plist
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -11,10 +11,9 @@ import Account
|
|||||||
import Articles
|
import Articles
|
||||||
import UserNotifications
|
import UserNotifications
|
||||||
|
|
||||||
final class UserNotificationManager: NSObject {
|
final class UserNotificationManager: Sendable {
|
||||||
|
|
||||||
override init() {
|
init() {
|
||||||
super.init()
|
|
||||||
NotificationCenter.default.addObserver(self, selector: #selector(accountDidDownloadArticles(_:)), name: .AccountDidDownloadArticles, object: nil)
|
NotificationCenter.default.addObserver(self, selector: #selector(accountDidDownloadArticles(_:)), name: .AccountDidDownloadArticles, object: nil)
|
||||||
NotificationCenter.default.addObserver(self, selector: #selector(statusesDidChange(_:)), name: .StatusesDidChange, object: nil)
|
NotificationCenter.default.addObserver(self, selector: #selector(statusesDidChange(_:)), name: .StatusesDidChange, object: nil)
|
||||||
registerCategoriesAndActions()
|
registerCategoriesAndActions()
|
||||||
@ -103,5 +102,4 @@ private extension UserNotificationManager {
|
|||||||
|
|
||||||
UNUserNotificationCenter.current().setNotificationCategories([newArticleCategory])
|
UNUserNotificationCenter.current().setNotificationCategories([newArticleCategory])
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user