2022-05-01 18:30:30 +02:00
|
|
|
import Foundation
|
2023-04-11 18:18:31 +02:00
|
|
|
import CBORCoding
|
2022-05-01 18:30:30 +02:00
|
|
|
|
2023-04-11 18:18:31 +02:00
|
|
|
protocol HistoryManagerProtocol {
|
|
|
|
|
|
|
|
func loadEntries() -> [HistoryItem]
|
|
|
|
|
|
|
|
func save(item: HistoryItem) throws
|
|
|
|
}
|
|
|
|
|
|
|
|
final class HistoryManager: HistoryManagerProtocol {
|
|
|
|
|
|
|
|
private let encoder = CBOREncoder(dateEncodingStrategy: .secondsSince1970)
|
2022-05-01 18:30:30 +02:00
|
|
|
|
|
|
|
private var fm: FileManager {
|
|
|
|
.default
|
|
|
|
}
|
|
|
|
|
2023-04-11 18:18:31 +02:00
|
|
|
static var documentDirectory: URL {
|
|
|
|
try! FileManager.default.url(
|
2022-05-01 18:30:30 +02:00
|
|
|
for: .documentDirectory,
|
|
|
|
in: .userDomainMask,
|
|
|
|
appropriateFor: nil, create: true)
|
|
|
|
}
|
|
|
|
|
2023-04-11 18:18:31 +02:00
|
|
|
private let fileUrl: URL
|
|
|
|
|
|
|
|
init() {
|
|
|
|
self.fileUrl = HistoryManager.documentDirectory.appendingPathComponent("history2.bin")
|
2022-05-01 18:30:30 +02:00
|
|
|
}
|
2023-04-11 18:18:31 +02:00
|
|
|
|
2022-05-01 18:30:30 +02:00
|
|
|
func loadEntries() -> [HistoryItem] {
|
2023-04-11 18:18:31 +02:00
|
|
|
guard fm.fileExists(atPath: fileUrl.path) else {
|
2022-05-01 18:30:30 +02:00
|
|
|
print("No history data found")
|
|
|
|
return []
|
|
|
|
}
|
|
|
|
let content: Data
|
|
|
|
do {
|
2023-04-11 18:18:31 +02:00
|
|
|
content = try Data(contentsOf: fileUrl)
|
2022-05-01 18:30:30 +02:00
|
|
|
} catch {
|
|
|
|
print("Failed to read history data: \(error)")
|
|
|
|
return []
|
|
|
|
}
|
2023-04-11 18:18:31 +02:00
|
|
|
let decoder = CBORDecoder()
|
2022-05-01 18:30:30 +02:00
|
|
|
var index = 0
|
|
|
|
var entries = [HistoryItem]()
|
|
|
|
while index < content.count {
|
2023-04-11 18:18:31 +02:00
|
|
|
let length = Int(content[index])
|
|
|
|
index += 1
|
|
|
|
if index + length > content.count {
|
|
|
|
print("Missing bytes in history file: needed \(length), has only \(content.count - index)")
|
|
|
|
return entries
|
|
|
|
}
|
|
|
|
let entryData = content[index..<index+length]
|
|
|
|
index += length
|
|
|
|
do {
|
|
|
|
let entry: HistoryItem = try decoder.decode(from: entryData)
|
|
|
|
entries.append(entry)
|
|
|
|
} catch {
|
|
|
|
print("Failed to decode history (index: \(index), length \(length)): \(error)")
|
2022-05-01 18:30:30 +02:00
|
|
|
return entries
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return entries.sorted().reversed()
|
|
|
|
}
|
|
|
|
|
|
|
|
func save(item: HistoryItem) throws {
|
2023-04-11 18:18:31 +02:00
|
|
|
let entryData = try encoder.encode(item)
|
|
|
|
let data = Data([UInt8(entryData.count)]) + entryData
|
|
|
|
guard fm.fileExists(atPath: fileUrl.path) else {
|
|
|
|
try data.write(to: fileUrl)
|
|
|
|
print("First history item written (\(data[0]))")
|
2022-05-01 18:30:30 +02:00
|
|
|
return
|
|
|
|
}
|
2023-04-11 18:18:31 +02:00
|
|
|
let handle = try FileHandle(forWritingTo: fileUrl)
|
2022-05-01 18:30:30 +02:00
|
|
|
try handle.seekToEnd()
|
|
|
|
try handle.write(contentsOf: data)
|
|
|
|
try handle.close()
|
2023-04-11 18:18:31 +02:00
|
|
|
print("History item written (\(data[0]))")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final class HistoryManagerMock: HistoryManagerProtocol {
|
|
|
|
|
|
|
|
func loadEntries() -> [HistoryItem] {
|
|
|
|
[.mock]
|
|
|
|
}
|
|
|
|
|
|
|
|
func save(item: HistoryItem) throws {
|
|
|
|
|
2022-05-01 18:30:30 +02:00
|
|
|
}
|
|
|
|
}
|