2017-05-23 22:24:42 +02:00
|
|
|
//
|
|
|
|
// LocalAccountRefresher.swift
|
2019-07-09 07:58:19 +02:00
|
|
|
// NetNewsWire
|
2017-05-23 22:24:42 +02:00
|
|
|
//
|
|
|
|
// Created by Brent Simmons on 9/6/16.
|
|
|
|
// Copyright © 2016 Ranchero Software, LLC. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import RSCore
|
2017-07-02 02:22:19 +02:00
|
|
|
import RSParser
|
2017-05-23 22:24:42 +02:00
|
|
|
import RSWeb
|
2018-07-24 03:29:08 +02:00
|
|
|
import Articles
|
2020-04-11 00:23:39 +02:00
|
|
|
import ArticlesDatabase
|
2017-05-23 22:24:42 +02:00
|
|
|
|
2020-04-10 18:20:35 +02:00
|
|
|
protocol LocalAccountRefresherDelegate {
|
2020-05-08 16:28:11 +02:00
|
|
|
func localAccountRefresher(_ refresher: LocalAccountRefresher, requestCompletedFor: WebFeed, articleChanges: ArticleChanges?)
|
2020-04-10 18:20:35 +02:00
|
|
|
}
|
|
|
|
|
2017-10-07 21:40:14 +02:00
|
|
|
final class LocalAccountRefresher {
|
2017-05-23 22:24:42 +02:00
|
|
|
|
2020-05-08 09:32:13 +02:00
|
|
|
private var completion: (() -> Void)? = nil
|
2019-12-07 00:06:54 +01:00
|
|
|
private var isSuspended = false
|
2020-04-10 18:20:35 +02:00
|
|
|
var delegate: LocalAccountRefresherDelegate?
|
2019-10-02 23:41:32 +02:00
|
|
|
|
2019-04-26 21:21:17 +02:00
|
|
|
private lazy var downloadSession: DownloadSession = {
|
|
|
|
return DownloadSession(delegate: self)
|
2017-05-23 22:24:42 +02:00
|
|
|
}()
|
|
|
|
|
2020-05-08 09:32:13 +02:00
|
|
|
public func refreshFeeds(_ feeds: Set<WebFeed>, completion: (() -> Void)? = nil) {
|
2020-04-24 02:13:57 +02:00
|
|
|
guard !feeds.isEmpty else {
|
2020-05-08 09:32:13 +02:00
|
|
|
completion?()
|
2020-04-24 02:13:57 +02:00
|
|
|
return
|
|
|
|
}
|
2020-04-26 03:20:56 +02:00
|
|
|
self.completion = completion
|
2019-04-26 21:21:17 +02:00
|
|
|
downloadSession.downloadObjects(feeds as NSSet)
|
2017-05-23 22:24:42 +02:00
|
|
|
}
|
2019-11-05 03:41:08 +01:00
|
|
|
|
2019-12-07 00:06:54 +01:00
|
|
|
public func suspend() {
|
2019-11-05 03:41:08 +01:00
|
|
|
downloadSession.cancelAll()
|
2019-12-07 00:06:54 +01:00
|
|
|
isSuspended = true
|
|
|
|
}
|
|
|
|
|
|
|
|
public func resume() {
|
|
|
|
isSuspended = false
|
2019-11-05 03:41:08 +01:00
|
|
|
}
|
|
|
|
|
2017-10-07 21:40:14 +02:00
|
|
|
}
|
2017-05-23 22:24:42 +02:00
|
|
|
|
2017-10-07 21:40:14 +02:00
|
|
|
// MARK: - DownloadSessionDelegate
|
|
|
|
|
|
|
|
extension LocalAccountRefresher: DownloadSessionDelegate {
|
|
|
|
|
|
|
|
func downloadSession(_ downloadSession: DownloadSession, requestForRepresentedObject representedObject: AnyObject) -> URLRequest? {
|
2019-11-15 03:11:41 +01:00
|
|
|
guard let feed = representedObject as? WebFeed else {
|
2017-05-23 22:24:42 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
guard let url = URL(string: feed.url) else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-28 05:58:59 +01:00
|
|
|
var request = URLRequest(url: url)
|
2017-09-27 06:43:40 +02:00
|
|
|
if let conditionalGetInfo = feed.conditionalGetInfo {
|
2020-01-28 05:58:59 +01:00
|
|
|
conditionalGetInfo.addRequestHeadersToURLRequest(&request)
|
2017-05-23 22:24:42 +02:00
|
|
|
}
|
|
|
|
|
2020-01-28 05:58:59 +01:00
|
|
|
return request
|
2017-05-23 22:24:42 +02:00
|
|
|
}
|
|
|
|
|
2019-12-08 00:26:38 +01:00
|
|
|
func downloadSession(_ downloadSession: DownloadSession, downloadDidCompleteForRepresentedObject representedObject: AnyObject, response: URLResponse?, data: Data, error: NSError?, completion: @escaping () -> Void) {
|
2020-04-02 19:00:10 +02:00
|
|
|
let feed = representedObject as! WebFeed
|
|
|
|
|
|
|
|
guard !data.isEmpty, !isSuspended else {
|
2019-12-08 18:00:20 +01:00
|
|
|
completion()
|
2020-05-08 16:28:11 +02:00
|
|
|
delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2017-05-23 22:24:42 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if let error = error {
|
|
|
|
print("Error downloading \(feed.url) - \(error)")
|
2019-12-08 18:00:20 +01:00
|
|
|
completion()
|
2020-05-08 16:28:11 +02:00
|
|
|
delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2017-05-23 22:24:42 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-01-18 08:00:56 +01:00
|
|
|
let dataHash = data.md5String
|
2017-05-23 22:24:42 +02:00
|
|
|
if dataHash == feed.contentHash {
|
2019-12-08 18:00:20 +01:00
|
|
|
completion()
|
2020-05-08 16:28:11 +02:00
|
|
|
delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2017-05-23 22:24:42 +02:00
|
|
|
return
|
|
|
|
}
|
2017-10-04 22:28:48 +02:00
|
|
|
|
2017-07-02 02:22:19 +02:00
|
|
|
let parserData = ParserData(url: feed.url, data: data)
|
|
|
|
FeedParser.parse(parserData) { (parsedFeed, error) in
|
2020-04-02 19:00:10 +02:00
|
|
|
|
2017-09-17 21:08:50 +02:00
|
|
|
guard let account = feed.account, let parsedFeed = parsedFeed, error == nil else {
|
2020-04-02 19:00:10 +02:00
|
|
|
completion()
|
2020-05-08 16:28:11 +02:00
|
|
|
self.delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2017-05-23 22:24:42 +02:00
|
|
|
return
|
|
|
|
}
|
2020-04-02 19:00:10 +02:00
|
|
|
|
2020-04-11 00:23:39 +02:00
|
|
|
account.update(feed, with: parsedFeed) { result in
|
2020-04-24 02:13:57 +02:00
|
|
|
if case .success(let articleChanges) = result {
|
2020-05-08 16:28:11 +02:00
|
|
|
if let httpResponse = response as? HTTPURLResponse {
|
|
|
|
feed.conditionalGetInfo = HTTPConditionalGetInfo(urlResponse: httpResponse)
|
2019-12-17 07:45:59 +01:00
|
|
|
}
|
2020-05-08 16:28:11 +02:00
|
|
|
feed.contentHash = dataHash
|
|
|
|
completion()
|
|
|
|
self.delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: articleChanges)
|
2020-04-14 08:25:35 +02:00
|
|
|
} else {
|
|
|
|
completion()
|
2020-05-08 16:28:11 +02:00
|
|
|
self.delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2017-05-23 22:24:42 +02:00
|
|
|
}
|
|
|
|
}
|
2020-04-02 19:00:10 +02:00
|
|
|
|
2017-05-23 22:24:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-07 21:40:14 +02:00
|
|
|
func downloadSession(_ downloadSession: DownloadSession, shouldContinueAfterReceivingData data: Data, representedObject: AnyObject) -> Bool {
|
2020-04-04 00:29:26 +02:00
|
|
|
let feed = representedObject as! WebFeed
|
|
|
|
guard !isSuspended else {
|
2020-05-08 16:28:11 +02:00
|
|
|
delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2017-05-23 22:24:42 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if data.isEmpty {
|
|
|
|
return true
|
|
|
|
}
|
2019-12-07 00:06:54 +01:00
|
|
|
|
2017-05-26 22:07:55 +02:00
|
|
|
if data.isDefinitelyNotFeed() {
|
2020-05-08 16:28:11 +02:00
|
|
|
delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2017-05-23 22:24:42 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if data.count > 4096 {
|
2017-07-02 02:22:19 +02:00
|
|
|
let parserData = ParserData(url: feed.url, data: data)
|
2020-04-02 19:25:23 +02:00
|
|
|
if FeedParser.mightBeAbleToParseBasedOnPartialData(parserData) {
|
|
|
|
return true
|
|
|
|
} else {
|
2020-05-08 16:28:11 +02:00
|
|
|
delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2020-04-02 19:25:23 +02:00
|
|
|
return false
|
|
|
|
}
|
2017-05-23 22:24:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2019-04-26 21:21:17 +02:00
|
|
|
func downloadSession(_ downloadSession: DownloadSession, didReceiveUnexpectedResponse response: URLResponse, representedObject: AnyObject) {
|
2020-04-02 19:25:23 +02:00
|
|
|
let feed = representedObject as! WebFeed
|
2020-05-08 16:28:11 +02:00
|
|
|
delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2019-04-25 14:13:14 +02:00
|
|
|
}
|
2019-04-26 21:21:17 +02:00
|
|
|
|
2019-04-25 14:13:14 +02:00
|
|
|
func downloadSession(_ downloadSession: DownloadSession, didReceiveNotModifiedResponse: URLResponse, representedObject: AnyObject) {
|
2020-04-02 19:00:10 +02:00
|
|
|
let feed = representedObject as! WebFeed
|
2020-05-08 16:28:11 +02:00
|
|
|
delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2019-04-25 14:13:14 +02:00
|
|
|
}
|
2019-10-02 23:41:32 +02:00
|
|
|
|
2020-04-10 18:20:35 +02:00
|
|
|
func downloadSession(_ downloadSession: DownloadSession, didDiscardDuplicateRepresentedObject representedObject: AnyObject) {
|
|
|
|
let feed = representedObject as! WebFeed
|
2020-05-08 16:28:11 +02:00
|
|
|
delegate?.localAccountRefresher(self, requestCompletedFor: feed, articleChanges: nil)
|
2020-04-10 18:20:35 +02:00
|
|
|
}
|
|
|
|
|
2019-10-02 23:41:32 +02:00
|
|
|
func downloadSessionDidCompleteDownloadObjects(_ downloadSession: DownloadSession) {
|
2020-05-08 09:32:13 +02:00
|
|
|
completion?()
|
2020-04-26 03:20:56 +02:00
|
|
|
completion = nil
|
2019-10-02 23:41:32 +02:00
|
|
|
}
|
|
|
|
|
2017-05-26 22:07:55 +02:00
|
|
|
}
|
2017-05-23 22:24:42 +02:00
|
|
|
|
2017-10-07 21:40:14 +02:00
|
|
|
// MARK: - Utility
|
|
|
|
|
2017-05-26 22:07:55 +02:00
|
|
|
private extension Data {
|
|
|
|
|
|
|
|
func isDefinitelyNotFeed() -> Bool {
|
|
|
|
// We only detect a few image types for now. This should get fleshed-out at some later date.
|
2020-01-18 08:00:56 +01:00
|
|
|
return self.isImage
|
2017-05-26 22:07:55 +02:00
|
|
|
}
|
2017-05-23 22:24:42 +02:00
|
|
|
}
|