2019-09-13 01:05:29 +02:00
|
|
|
|
//
|
|
|
|
|
// OPMLFile.swift
|
|
|
|
|
// Account
|
|
|
|
|
//
|
|
|
|
|
// Created by Maurice Parker on 9/12/19.
|
|
|
|
|
// Copyright © 2019 Ranchero Software, LLC. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
|
import os.log
|
|
|
|
|
import RSCore
|
|
|
|
|
import RSParser
|
|
|
|
|
|
|
|
|
|
final class OPMLFile: NSObject, NSFilePresenter {
|
|
|
|
|
|
2019-09-13 21:56:24 +02:00
|
|
|
|
private var log = OSLog(subsystem: Bundle.main.bundleIdentifier!, category: "opmlFile")
|
2019-09-13 01:05:29 +02:00
|
|
|
|
|
2019-09-13 17:15:11 +02:00
|
|
|
|
private var isDirty = false {
|
|
|
|
|
didSet {
|
|
|
|
|
queueSaveToDiskIfNeeded()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private var isLoading = false
|
2019-09-13 17:48:04 +02:00
|
|
|
|
private let fileURL: URL
|
2019-09-13 01:05:29 +02:00
|
|
|
|
private let account: Account
|
|
|
|
|
private let operationQueue: OperationQueue
|
|
|
|
|
|
|
|
|
|
var presentedItemURL: URL? {
|
2019-09-13 17:48:04 +02:00
|
|
|
|
return fileURL
|
2019-09-13 01:05:29 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var presentedItemOperationQueue: OperationQueue {
|
|
|
|
|
return operationQueue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
init(filename: String, account: Account) {
|
2019-09-13 17:48:04 +02:00
|
|
|
|
self.fileURL = URL(fileURLWithPath: filename)
|
2019-09-13 01:05:29 +02:00
|
|
|
|
self.account = account
|
|
|
|
|
operationQueue = OperationQueue()
|
|
|
|
|
operationQueue.maxConcurrentOperationCount = 1
|
2019-09-13 01:41:42 +02:00
|
|
|
|
|
|
|
|
|
super.init()
|
|
|
|
|
|
|
|
|
|
NSFileCoordinator.addFilePresenter(self)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func presentedItemDidChange() {
|
|
|
|
|
DispatchQueue.main.async {
|
|
|
|
|
self.reload()
|
|
|
|
|
}
|
2019-09-13 01:05:29 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-09-13 17:15:11 +02:00
|
|
|
|
func markAsDirty() {
|
|
|
|
|
if !isLoading {
|
|
|
|
|
isDirty = true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func queueSaveToDiskIfNeeded() {
|
|
|
|
|
Account.saveQueue.add(self, #selector(saveToDiskIfNeeded))
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-13 01:05:29 +02:00
|
|
|
|
func load() {
|
2019-09-13 17:15:11 +02:00
|
|
|
|
isLoading = true
|
2019-09-13 01:41:42 +02:00
|
|
|
|
guard let opmlItems = parsedOPMLItems() else { return }
|
|
|
|
|
BatchUpdate.shared.perform {
|
|
|
|
|
account.loadOPMLItems(opmlItems, parentFolder: nil)
|
|
|
|
|
}
|
2019-09-13 17:15:11 +02:00
|
|
|
|
isLoading = false
|
2019-09-13 01:41:42 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-09-13 17:15:11 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private extension OPMLFile {
|
|
|
|
|
|
|
|
|
|
@objc func saveToDiskIfNeeded() {
|
|
|
|
|
if isDirty && !account.isDeleted {
|
|
|
|
|
isDirty = false
|
|
|
|
|
save()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-13 01:41:42 +02:00
|
|
|
|
func save() {
|
|
|
|
|
let opmlDocumentString = opmlDocument()
|
2019-09-13 17:48:04 +02:00
|
|
|
|
|
|
|
|
|
let errorPointer: NSErrorPointer = nil
|
|
|
|
|
let fileCoordinator = NSFileCoordinator(filePresenter: self)
|
|
|
|
|
|
|
|
|
|
fileCoordinator.coordinate(writingItemAt: fileURL, options: .forReplacing, error: errorPointer, byAccessor: { writeURL in
|
|
|
|
|
do {
|
|
|
|
|
try opmlDocumentString.write(to: writeURL, atomically: true, encoding: .utf8)
|
|
|
|
|
} catch let error as NSError {
|
|
|
|
|
os_log(.error, log: log, "OPML save to disk failed: %@.", error.localizedDescription)
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
if let error = errorPointer?.pointee {
|
|
|
|
|
os_log(.error, log: log, "OPML save to disk coordination failed: %@.", error.localizedDescription)
|
2019-09-13 01:41:42 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func reload() {
|
2019-09-13 17:15:11 +02:00
|
|
|
|
isLoading = true
|
2019-09-13 01:41:42 +02:00
|
|
|
|
guard let opmlItems = parsedOPMLItems() else { return }
|
|
|
|
|
BatchUpdate.shared.perform {
|
|
|
|
|
account.topLevelFeeds.removeAll()
|
|
|
|
|
account.loadOPMLItems(opmlItems, parentFolder: nil)
|
|
|
|
|
}
|
2019-09-13 17:15:11 +02:00
|
|
|
|
isLoading = false
|
2019-09-13 01:41:42 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func parsedOPMLItems() -> [RSOPMLItem]? {
|
2019-09-13 17:48:04 +02:00
|
|
|
|
|
|
|
|
|
var fileData: Data? = nil
|
|
|
|
|
let errorPointer: NSErrorPointer = nil
|
|
|
|
|
let fileCoordinator = NSFileCoordinator(filePresenter: self)
|
2019-09-13 01:41:42 +02:00
|
|
|
|
|
2019-09-13 17:48:04 +02:00
|
|
|
|
fileCoordinator.coordinate(readingItemAt: fileURL, options: [], error: errorPointer, byAccessor: { readURL in
|
|
|
|
|
do {
|
|
|
|
|
fileData = try Data(contentsOf: readURL)
|
|
|
|
|
} catch {
|
|
|
|
|
// Commented out because it’s not an error on first run.
|
|
|
|
|
// TODO: make it so we know if it’s first run or not.
|
|
|
|
|
//NSApplication.shared.presentError(error)
|
|
|
|
|
os_log(.error, log: log, "OPML read from disk failed: %@.", error.localizedDescription)
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
if let error = errorPointer?.pointee {
|
|
|
|
|
os_log(.error, log: log, "OPML read from disk coordination failed: %@.", error.localizedDescription)
|
2019-09-13 01:05:29 +02:00
|
|
|
|
}
|
2019-09-13 17:48:04 +02:00
|
|
|
|
|
2019-09-13 01:05:29 +02:00
|
|
|
|
guard let opmlData = fileData else {
|
2019-09-13 01:41:42 +02:00
|
|
|
|
return nil
|
2019-09-13 01:05:29 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-09-13 17:48:04 +02:00
|
|
|
|
let parserData = ParserData(url: fileURL.absoluteString, data: opmlData)
|
2019-09-13 01:05:29 +02:00
|
|
|
|
var opmlDocument: RSOPMLDocument?
|
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
opmlDocument = try RSOPMLParser.parseOPML(with: parserData)
|
|
|
|
|
} catch {
|
|
|
|
|
os_log(.error, log: log, "OPML Import failed: %@.", error.localizedDescription)
|
2019-09-13 01:41:42 +02:00
|
|
|
|
return nil
|
2019-09-13 01:05:29 +02:00
|
|
|
|
}
|
2019-09-13 01:41:42 +02:00
|
|
|
|
|
|
|
|
|
return opmlDocument?.children
|
|
|
|
|
|
2019-09-13 01:05:29 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func opmlDocument() -> String {
|
|
|
|
|
let escapedTitle = account.nameForDisplay.rs_stringByEscapingSpecialXMLCharacters()
|
|
|
|
|
let openingText =
|
|
|
|
|
"""
|
|
|
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
|
|
|
<!-- OPML generated by NetNewsWire -->
|
|
|
|
|
<opml version="1.1">
|
|
|
|
|
<head>
|
|
|
|
|
<title>\(escapedTitle)</title>
|
|
|
|
|
</head>
|
|
|
|
|
<body>
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
let middleText = account.OPMLString(indentLevel: 0)
|
|
|
|
|
|
|
|
|
|
let closingText =
|
|
|
|
|
"""
|
|
|
|
|
</body>
|
|
|
|
|
</opml>
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
let opml = openingText + middleText + closingText
|
|
|
|
|
return opml
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|