2019-09-19 04:56:43 +02:00
|
|
|
//
|
|
|
|
// FeedlyCreateFeedsForCollectionFoldersOperation.swift
|
|
|
|
// Account
|
|
|
|
//
|
|
|
|
// Created by Kiel Gillard on 20/9/19.
|
|
|
|
// Copyright © 2019 Ranchero Software, LLC. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import os.log
|
|
|
|
|
|
|
|
/// Single responsibility is to accurately reflect Collections and their Feeds as Folders and their Feeds.
|
2019-09-23 09:29:53 +02:00
|
|
|
final class FeedlyCreateFeedsForCollectionFoldersOperation: FeedlyOperation {
|
2019-09-19 04:56:43 +02:00
|
|
|
|
|
|
|
let account: Account
|
2019-10-10 00:15:22 +02:00
|
|
|
let feedsAndFoldersProvider: FeedlyFeedsAndFoldersProviding
|
2019-09-19 04:56:43 +02:00
|
|
|
let log: OSLog
|
|
|
|
|
2019-10-10 00:15:22 +02:00
|
|
|
init(account: Account, feedsAndFoldersProvider: FeedlyFeedsAndFoldersProviding, log: OSLog) {
|
|
|
|
self.feedsAndFoldersProvider = feedsAndFoldersProvider
|
2019-09-19 04:56:43 +02:00
|
|
|
self.account = account
|
|
|
|
self.log = log
|
|
|
|
}
|
|
|
|
|
2020-01-16 06:30:37 +01:00
|
|
|
override func run() {
|
2020-01-16 07:51:00 +01:00
|
|
|
super.run()
|
2020-01-16 06:30:37 +01:00
|
|
|
defer {
|
|
|
|
didFinish()
|
|
|
|
}
|
2019-09-19 04:56:43 +02:00
|
|
|
|
2019-10-10 00:15:22 +02:00
|
|
|
let pairs = feedsAndFoldersProvider.feedsAndFolders
|
2019-09-19 04:56:43 +02:00
|
|
|
|
2019-10-10 12:24:45 +02:00
|
|
|
let feedsBefore = Set(pairs
|
|
|
|
.map { $0.1 }
|
2019-11-15 03:11:41 +01:00
|
|
|
.flatMap { $0.topLevelWebFeeds })
|
2019-10-10 12:24:45 +02:00
|
|
|
|
2019-10-03 05:30:43 +02:00
|
|
|
// Remove feeds in a folder which are not in the corresponding collection.
|
2019-10-10 00:15:22 +02:00
|
|
|
for (collectionFeeds, folder) in pairs {
|
2019-11-15 03:11:41 +01:00
|
|
|
let feedsInFolder = folder.topLevelWebFeeds
|
2019-10-10 00:15:22 +02:00
|
|
|
let feedsInCollection = Set(collectionFeeds.map { $0.id })
|
2019-11-15 03:11:41 +01:00
|
|
|
let feedsToRemove = feedsInFolder.filter { !feedsInCollection.contains($0.webFeedID) }
|
2019-10-03 05:30:43 +02:00
|
|
|
if !feedsToRemove.isEmpty {
|
|
|
|
folder.removeFeeds(feedsToRemove)
|
2019-10-10 00:15:22 +02:00
|
|
|
// os_log(.debug, log: log, "\"%@\" - removed: %@", collection.label, feedsToRemove.map { $0.feedID }, feedsInCollection)
|
2019-10-03 05:30:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pair each Feed with its Folder.
|
2019-11-15 03:11:41 +01:00
|
|
|
var feedsAdded = Set<WebFeed>()
|
2019-10-10 12:24:45 +02:00
|
|
|
|
2019-09-19 04:56:43 +02:00
|
|
|
let feedsAndFolders = pairs
|
|
|
|
.map({ (collectionFeeds, folder) -> [(FeedlyFeed, Folder)] in
|
|
|
|
return collectionFeeds.map { feed -> (FeedlyFeed, Folder) in
|
|
|
|
return (feed, folder) // pairs a folder for every feed in parallel
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.flatMap { $0 }
|
2019-11-15 03:11:41 +01:00
|
|
|
.compactMap { (collectionFeed, folder) -> (WebFeed, Folder) in
|
2019-09-19 04:56:43 +02:00
|
|
|
|
2019-10-10 12:24:45 +02:00
|
|
|
// find an existing feed previously added to the account
|
2019-11-15 03:11:41 +01:00
|
|
|
if let feed = account.existingWebFeed(withWebFeedID: collectionFeed.id) {
|
2019-10-10 12:24:45 +02:00
|
|
|
return (feed, folder)
|
|
|
|
} else {
|
|
|
|
// find an existing feed we created below in an earlier value
|
2019-11-15 03:11:41 +01:00
|
|
|
for feed in feedsAdded where feed.webFeedID == collectionFeed.id {
|
2019-09-19 04:56:43 +02:00
|
|
|
return (feed, folder)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// no exsiting feed, create a new one
|
2019-10-03 05:30:43 +02:00
|
|
|
let id = collectionFeed.id
|
|
|
|
let url = FeedlyFeedResourceId(id: id).url
|
2019-11-15 03:11:41 +01:00
|
|
|
let feed = account.createWebFeed(with: collectionFeed.title, url: url, webFeedID: id, homePageURL: collectionFeed.website)
|
2019-09-19 04:56:43 +02:00
|
|
|
|
|
|
|
// So the same feed isn't created more than once.
|
2019-10-10 12:24:45 +02:00
|
|
|
feedsAdded.insert(feed)
|
2019-09-19 04:56:43 +02:00
|
|
|
|
|
|
|
return (feed, folder)
|
|
|
|
}
|
|
|
|
|
|
|
|
os_log(.debug, log: log, "Processing %i feeds.", feedsAndFolders.count)
|
|
|
|
feedsAndFolders.forEach { (feed, folder) in
|
|
|
|
if !folder.has(feed) {
|
2019-11-15 03:11:41 +01:00
|
|
|
folder.addWebFeed(feed)
|
2019-09-19 04:56:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-03 05:30:43 +02:00
|
|
|
// Remove feeds without folders/collections.
|
2019-09-19 04:56:43 +02:00
|
|
|
let feedsAfter = Set(feedsAndFolders.map { $0.0 })
|
|
|
|
let feedsWithoutCollections = feedsBefore.subtracting(feedsAfter)
|
2019-10-03 10:45:16 +02:00
|
|
|
account.removeFeeds(feedsWithoutCollections)
|
2019-09-19 04:56:43 +02:00
|
|
|
|
|
|
|
if !feedsWithoutCollections.isEmpty {
|
|
|
|
os_log(.debug, log: log, "Removed %i feeds", feedsWithoutCollections.count)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|