mirror of
https://github.com/advplyr/audiobookshelf-app.git
synced 2025-07-12 23:14:48 +02:00
333 lines
16 KiB
Swift
333 lines
16 KiB
Swift
//
|
|
// AbsDownloader.swift
|
|
// App
|
|
//
|
|
// Created by advplyr on 5/13/22.
|
|
//
|
|
|
|
import Foundation
|
|
import Capacitor
|
|
import RealmSwift
|
|
|
|
@objc(AbsDownloader)
|
|
public class AbsDownloader: CAPPlugin, URLSessionDownloadDelegate {
|
|
|
|
static let downloadsDirectory = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask)[0]
|
|
|
|
typealias DownloadProgressHandler = (_ downloadItem: DownloadItem, _ downloadItemPart: inout DownloadItemPart) throws -> Void
|
|
|
|
private lazy var session: URLSession = {
|
|
let queue = OperationQueue()
|
|
queue.maxConcurrentOperationCount = 5
|
|
return URLSession(configuration: .default, delegate: self, delegateQueue: queue)
|
|
}()
|
|
private let progressStatusQueue = DispatchQueue(label: "progress-status-queue", attributes: .concurrent)
|
|
private var downloadItemProgress = [String: DownloadItem]()
|
|
private var isMonitoringProgress = false
|
|
|
|
public func urlSession(_ session: URLSession, downloadTask: URLSessionDownloadTask, didFinishDownloadingTo location: URL) {
|
|
handleDownloadTaskUpdate(downloadTask: downloadTask) { downloadItem, downloadItemPart in
|
|
downloadItemPart.progress = 1
|
|
downloadItemPart.completed = true
|
|
|
|
do {
|
|
// Move the downloaded file into place
|
|
guard let destinationUrl = downloadItemPart.destinationURL else {
|
|
throw LibraryItemDownloadError.downloadItemPartDestinationUrlNotDefined
|
|
}
|
|
try? FileManager.default.removeItem(at: destinationUrl)
|
|
try FileManager.default.moveItem(at: location, to: destinationUrl)
|
|
downloadItemPart.moved = true
|
|
} catch {
|
|
downloadItemPart.failed = true
|
|
throw error
|
|
}
|
|
}
|
|
}
|
|
|
|
public func urlSession(_ session: URLSession, task: URLSessionTask, didCompleteWithError error: Error?) {
|
|
handleDownloadTaskUpdate(downloadTask: task) { downloadItem, downloadItemPart in
|
|
if let error = error {
|
|
downloadItemPart.completed = true
|
|
downloadItemPart.failed = true
|
|
throw error
|
|
}
|
|
}
|
|
}
|
|
|
|
public func urlSession(_ session: URLSession, downloadTask: URLSessionDownloadTask, didWriteData bytesWritten: Int64, totalBytesWritten: Int64, totalBytesExpectedToWrite: Int64) {
|
|
handleDownloadTaskUpdate(downloadTask: downloadTask) { downloadItem, downloadItemPart in
|
|
// Calculate the download percentage
|
|
let percentDownloaded = (Double(totalBytesWritten) / Double(totalBytesExpectedToWrite)) * 100
|
|
// Only update the progress if we received accurate progress data
|
|
if percentDownloaded >= 0.0 && percentDownloaded <= 100.0 {
|
|
downloadItemPart.progress = percentDownloaded
|
|
}
|
|
}
|
|
}
|
|
|
|
private func handleDownloadTaskUpdate(downloadTask: URLSessionTask, progressHandler: DownloadProgressHandler) {
|
|
do {
|
|
guard let downloadItemPartId = downloadTask.taskDescription else { throw LibraryItemDownloadError.noTaskDescription }
|
|
NSLog("Received download update for \(downloadItemPartId)")
|
|
|
|
// Find the download item
|
|
let downloadItem = Database.shared.getDownloadItem(downloadItemPartId: downloadItemPartId)
|
|
guard var downloadItem = downloadItem else { throw LibraryItemDownloadError.downloadItemNotFound }
|
|
self.progressStatusQueue.sync {
|
|
downloadItem = self.downloadItemProgress[downloadItem.id!] ?? downloadItem
|
|
}
|
|
|
|
// Find the download item part
|
|
let partIndex = downloadItem.downloadItemParts.firstIndex(where: { $0.id == downloadItemPartId })
|
|
guard let partIndex = partIndex else { throw LibraryItemDownloadError.downloadItemPartNotFound }
|
|
|
|
// Call the progress handler
|
|
do {
|
|
try progressHandler(downloadItem, &downloadItem.downloadItemParts[partIndex])
|
|
} catch {
|
|
NSLog("Error while processing progress")
|
|
debugPrint(error)
|
|
}
|
|
|
|
// Update the progress
|
|
self.progressStatusQueue.async(flags: .barrier) {
|
|
self.downloadItemProgress.updateValue(downloadItem, forKey: downloadItem.id!)
|
|
}
|
|
self.notifyDownloadProgress()
|
|
} catch {
|
|
NSLog("DownloadItemError")
|
|
debugPrint(error)
|
|
}
|
|
}
|
|
|
|
// We want to handle updating the UI in the background and throttled so we don't overload the UI with progress updates
|
|
private func notifyDownloadProgress() {
|
|
if !self.isMonitoringProgress {
|
|
self.isMonitoringProgress = true
|
|
DispatchQueue.global(qos: .userInteractive).async {
|
|
NSLog("Starting monitoring download progress...")
|
|
|
|
// Fetch active downloads in a thread-safe way
|
|
func fetchActiveDownloads() -> [String: DownloadItem]? {
|
|
self.progressStatusQueue.sync { self.downloadItemProgress }
|
|
}
|
|
|
|
// Remove a completed download item in a thread-safe way
|
|
func handleDoneDownloadItem(_ item: DownloadItem) {
|
|
self.progressStatusQueue.async(flags: .barrier) {
|
|
self.downloadItemProgress.removeValue(forKey: item.id!)
|
|
}
|
|
Database.shared.removeDownloadItem(item)
|
|
self.handleDownloadTaskCompleteFromDownloadItem(item)
|
|
}
|
|
|
|
// While there are active download items, emit status updates
|
|
while !(fetchActiveDownloads()?.isEmpty ?? false) {
|
|
if let activeDownloads = fetchActiveDownloads() {
|
|
for item in activeDownloads.values {
|
|
try? self.notifyListeners("onItemDownloadUpdate", data: item.asDictionary())
|
|
if item.isDoneDownloading() { handleDoneDownloadItem(item) }
|
|
}
|
|
}
|
|
|
|
// Wait 200ms before reporting status again
|
|
Thread.sleep(forTimeInterval: TimeInterval(0.2))
|
|
}
|
|
|
|
NSLog("Finished monitoring download progress...")
|
|
self.isMonitoringProgress = false
|
|
}
|
|
}
|
|
}
|
|
|
|
private func handleDownloadTaskCompleteFromDownloadItem(_ downloadItem: DownloadItem) {
|
|
var statusNotification = [String: Any]()
|
|
statusNotification["libraryItemId"] = downloadItem.libraryItemId
|
|
|
|
if ( downloadItem.didDownloadSuccessfully() ) {
|
|
ApiClient.getLibraryItemWithProgress(libraryItemId: downloadItem.libraryItemId!, episodeId: downloadItem.episodeId) { libraryItem in
|
|
guard let libraryItem = libraryItem else { NSLog("LibraryItem not found"); return }
|
|
let localDirectory = libraryItem.id
|
|
var coverFile: String?
|
|
|
|
// Assemble the local library item
|
|
let files = downloadItem.downloadItemParts.enumerated().compactMap { _, part -> LocalFile? in
|
|
if part.filename == "cover.jpg" {
|
|
coverFile = part.destinationUri
|
|
return nil
|
|
} else {
|
|
return LocalFile(libraryItem.id, part.filename!, part.mimeType()!, part.destinationUri!, fileSize: Int(part.destinationURL!.fileSize))
|
|
}
|
|
}
|
|
let localLibraryItem = LocalLibraryItem(libraryItem, localUrl: localDirectory, server: Store.serverConfig!, files: files, coverPath: coverFile)
|
|
|
|
Database.shared.saveLocalLibraryItem(localLibraryItem: localLibraryItem)
|
|
statusNotification["localLibraryItem"] = try? localLibraryItem.asDictionary()
|
|
|
|
if let progress = libraryItem.userMediaProgress {
|
|
// TODO: Handle podcast
|
|
let localMediaProgress = LocalMediaProgress(localLibraryItem: localLibraryItem, episode: nil, progress: progress)
|
|
Database.shared.saveLocalMediaProgress(localMediaProgress)
|
|
statusNotification["localMediaProgress"] = try? localMediaProgress.asDictionary()
|
|
}
|
|
|
|
self.notifyListeners("onItemDownloadComplete", data: statusNotification)
|
|
}
|
|
} else {
|
|
self.notifyListeners("onItemDownloadComplete", data: statusNotification)
|
|
}
|
|
}
|
|
|
|
@objc func downloadLibraryItem(_ call: CAPPluginCall) {
|
|
let libraryItemId = call.getString("libraryItemId")
|
|
var episodeId = call.getString("episodeId")
|
|
if ( episodeId == "null" ) { episodeId = nil }
|
|
|
|
NSLog("Download library item \(libraryItemId ?? "N/A") / episode \(episodeId ?? "N/A")")
|
|
guard let libraryItemId = libraryItemId else { return call.resolve(["error": "libraryItemId not specified"]) }
|
|
|
|
ApiClient.getLibraryItemWithProgress(libraryItemId: libraryItemId, episodeId: episodeId) { libraryItem in
|
|
if let libraryItem = libraryItem {
|
|
NSLog("Got library item from server \(libraryItem.id)")
|
|
do {
|
|
if let episodeId = episodeId {
|
|
// Download a podcast episode
|
|
guard libraryItem.mediaType == "podcast" else { throw LibraryItemDownloadError.libraryItemNotPodcast }
|
|
let episode = libraryItem.media?.episodes.enumerated().first(where: { $1.id == episodeId })?.element
|
|
guard let episode = episode else { throw LibraryItemDownloadError.podcastEpisodeNotFound }
|
|
try self.startLibraryItemDownload(libraryItem, episode: episode)
|
|
} else {
|
|
// Download a book
|
|
try self.startLibraryItemDownload(libraryItem)
|
|
}
|
|
call.resolve()
|
|
} catch {
|
|
debugPrint(error)
|
|
call.resolve(["error": "Failed to download"])
|
|
}
|
|
} else {
|
|
call.resolve(["error": "Server request failed"])
|
|
}
|
|
}
|
|
}
|
|
|
|
private func startLibraryItemDownload(_ item: LibraryItem) throws {
|
|
try startLibraryItemDownload(item, episode: nil)
|
|
}
|
|
|
|
private func startLibraryItemDownload(_ item: LibraryItem, episode: PodcastEpisode?) throws {
|
|
var tracks = List<AudioTrack>()
|
|
var episodeId: String?
|
|
|
|
// Handle the different media type downloads
|
|
switch item.mediaType {
|
|
case "book":
|
|
guard item.media?.tracks.count ?? 0 > 0 else { throw LibraryItemDownloadError.noTracks }
|
|
tracks = item.media?.tracks ?? tracks
|
|
case "podcast":
|
|
guard let episode = episode else { throw LibraryItemDownloadError.podcastEpisodeNotFound }
|
|
guard let podcastTrack = episode.audioTrack else { throw LibraryItemDownloadError.noTracks }
|
|
episodeId = episode.id
|
|
tracks.append(podcastTrack)
|
|
default:
|
|
throw LibraryItemDownloadError.unknownMediaType
|
|
}
|
|
|
|
// Queue up everything for downloading
|
|
let downloadItem = DownloadItem(libraryItem: item, episodeId: episodeId, server: Store.serverConfig!)
|
|
for (i, track) in tracks.enumerated() {
|
|
downloadItem.downloadItemParts.append(try startLibraryItemTrackDownload(item: item, position: i, track: track))
|
|
}
|
|
|
|
// Also download the cover
|
|
if item.media?.coverPath != nil && !(item.media?.coverPath!.isEmpty ?? true) {
|
|
if let coverDownload = try? startLibraryItemCoverDownload(item: item) {
|
|
downloadItem.downloadItemParts.append(coverDownload)
|
|
}
|
|
}
|
|
|
|
// Persist in the database before status start coming in
|
|
Database.shared.saveDownloadItem(downloadItem)
|
|
|
|
// Start all the downloads
|
|
for downloadItemPart in downloadItem.downloadItemParts {
|
|
downloadItemPart.task?.resume()
|
|
}
|
|
}
|
|
|
|
private func startLibraryItemTrackDownload(item: LibraryItem, position: Int, track: AudioTrack) throws -> DownloadItemPart {
|
|
NSLog("TRACK \(track.contentUrl!)")
|
|
|
|
// If we don't name metadata, then we can't proceed
|
|
guard let filename = track.metadata?.filename else {
|
|
throw LibraryItemDownloadError.noMetadata
|
|
}
|
|
|
|
let serverUrl = urlForTrack(item: item, track: track)
|
|
let itemDirectory = try createLibraryItemFileDirectory(item: item)
|
|
let localUrl = "\(itemDirectory)/\(filename)"
|
|
|
|
let task = session.downloadTask(with: serverUrl)
|
|
let downloadItemPart = DownloadItemPart(filename: filename, destination: localUrl, itemTitle: track.title ?? "Unknown", serverPath: Store.serverConfig!.address, audioTrack: track, episode: nil)
|
|
|
|
// Store the id on the task so the download item can be pulled from the database later
|
|
task.taskDescription = downloadItemPart.id
|
|
downloadItemPart.task = task
|
|
|
|
return downloadItemPart
|
|
}
|
|
|
|
private func startLibraryItemCoverDownload(item: LibraryItem) throws -> DownloadItemPart {
|
|
let filename = "cover.jpg"
|
|
let serverPath = "/api/items/\(item.id)/cover"
|
|
let itemDirectory = try createLibraryItemFileDirectory(item: item)
|
|
let localUrl = "\(itemDirectory)/\(filename)"
|
|
|
|
let downloadItemPart = DownloadItemPart(filename: filename, destination: localUrl, itemTitle: "cover", serverPath: serverPath, audioTrack: nil, episode: nil)
|
|
let task = session.downloadTask(with: downloadItemPart.downloadURL!)
|
|
|
|
// Store the id on the task so the download item can be pulled from the database later
|
|
task.taskDescription = downloadItemPart.id
|
|
downloadItemPart.task = task
|
|
|
|
return downloadItemPart
|
|
}
|
|
|
|
private func urlForTrack(item: LibraryItem, track: AudioTrack) -> URL {
|
|
// filename needs to be encoded otherwise would just use contentUrl
|
|
let filenameEncoded = track.metadata?.filename.addingPercentEncoding(withAllowedCharacters: NSCharacterSet.urlQueryAllowed)
|
|
let urlstr = "\(Store.serverConfig!.address)/s/item/\(item.id)/\(filenameEncoded ?? "")?token=\(Store.serverConfig!.token)"
|
|
return URL(string: urlstr)!
|
|
}
|
|
|
|
private func createLibraryItemFileDirectory(item: LibraryItem) throws -> String {
|
|
let itemDirectory = item.id
|
|
NSLog("ITEM DIR \(itemDirectory)")
|
|
|
|
do {
|
|
try FileManager.default.createDirectory(at: AbsDownloader.downloadsDirectory.appendingPathComponent(itemDirectory), withIntermediateDirectories: true)
|
|
} catch {
|
|
NSLog("Failed to CREATE LI DIRECTORY \(error)")
|
|
throw LibraryItemDownloadError.failedDirectory
|
|
}
|
|
|
|
return itemDirectory
|
|
}
|
|
|
|
}
|
|
|
|
enum LibraryItemDownloadError: String, Error {
|
|
case noTracks = "No tracks on library item"
|
|
case noMetadata = "No metadata for track, unable to download"
|
|
case libraryItemNotPodcast = "Library item is not a podcast but episode was requested"
|
|
case podcastEpisodeNotFound = "Invalid podcast episode not found"
|
|
case unknownMediaType = "Unknown media type"
|
|
case failedDirectory = "Failed to create directory"
|
|
case failedDownload = "Failed to download item"
|
|
case noTaskDescription = "No task description"
|
|
case downloadItemNotFound = "DownloadItem not found"
|
|
case downloadItemPartNotFound = "DownloadItemPart not found"
|
|
case downloadItemPartDestinationUrlNotDefined = "DownloadItemPart destination URL not defined"
|
|
case libraryItemNotFound = "LibraryItem not found for id"
|
|
}
|