Minor Downloader cleanup (#9511)

* Inline completeDownload

* Consolidate queueState updates in removeFromQueue

* Inline post-download steps into downloadChapter
This commit is contained in:
Two-Ai 2023-05-19 17:16:32 -04:00 committed by GitHub
parent f50f5c4b54
commit c27bf4e866
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 66 additions and 52 deletions

View File

@ -59,6 +59,17 @@ class DownloadStore(
} }
} }
/**
* Removes a list of downloads from the store.
*
* @param downloads the download to remove.
*/
fun removeAll(downloads: List<Download>) {
preferences.edit {
downloads.forEach { remove(getKey(it)) }
}
}
/** /**
* Removes all the downloads from the store. * Removes all the downloads from the store.
*/ */

View File

@ -217,7 +217,13 @@ class Downloader(
.observeOn(AndroidSchedulers.mainThread()) .observeOn(AndroidSchedulers.mainThread())
.subscribe( .subscribe(
{ {
completeDownload(it) // Remove successful download from queue
if (it.status == Download.State.DOWNLOADED) {
removeFromQueue(it)
}
if (areAllDownloadsFinished()) {
stop()
}
}, },
{ error -> { error ->
logcat(LogPriority.ERROR, error) logcat(LogPriority.ERROR, error)
@ -363,7 +369,30 @@ class Downloader(
} }
// Do after download completes // Do after download completes
ensureSuccessfulDownload(download, mangaDir, tmpDir, chapterDirname)
if (!isDownloadSuccessful(download, tmpDir)) {
download.status = Download.State.ERROR
return
}
createComicInfoFile(
tmpDir,
download.manga,
download.chapter,
download.source,
)
// Only rename the directory if it's downloaded
if (downloadPreferences.saveChaptersAsCBZ().get()) {
archiveChapter(mangaDir, chapterDirname, tmpDir)
} else {
tmpDir.renameTo(chapterDirname)
}
cache.addChapter(chapterDirname, mangaDir, download.manga)
DiskUtil.createNoMediaFile(tmpDir, context)
download.status = Download.State.DOWNLOADED
} catch (error: Throwable) { } catch (error: Throwable) {
if (error is CancellationException) throw error if (error is CancellationException) throw error
// If the page list threw, it will resume here // If the page list threw, it will resume here
@ -515,26 +544,20 @@ class Downloader(
* Checks if the download was successful. * Checks if the download was successful.
* *
* @param download the download to check. * @param download the download to check.
* @param mangaDir the manga directory of the download.
* @param tmpDir the directory where the download is currently stored. * @param tmpDir the directory where the download is currently stored.
* @param dirname the real (non temporary) directory name of the download.
*/ */
private suspend fun ensureSuccessfulDownload( private fun isDownloadSuccessful(
download: Download, download: Download,
mangaDir: UniFile,
tmpDir: UniFile, tmpDir: UniFile,
dirname: String, ): Boolean {
) {
// Page list hasn't been initialized // Page list hasn't been initialized
val downloadPageCount = download.pages?.size ?: run { val downloadPageCount = download.pages?.size ?: return false
download.status = Download.State.ERROR
return
}
// Ensure that all pages have been downloaded // Ensure that all pages have been downloaded
if (download.downloadedImages != downloadPageCount) { if (download.downloadedImages != downloadPageCount) {
download.status = Download.State.ERROR return false
return
} }
// Ensure that the chapter folder has all the pages // Ensure that the chapter folder has all the pages
val downloadedImagesCount = tmpDir.listFiles().orEmpty().count { val downloadedImagesCount = tmpDir.listFiles().orEmpty().count {
val fileName = it.name.orEmpty() val fileName = it.name.orEmpty()
@ -547,28 +570,10 @@ class Downloader(
} }
} }
if (downloadedImagesCount != downloadPageCount) { if (downloadedImagesCount != downloadPageCount) {
download.status = Download.State.ERROR return false
return
} }
createComicInfoFile( return true
tmpDir,
download.manga,
download.chapter,
download.source,
)
// Only rename the directory if it's downloaded
if (downloadPreferences.saveChaptersAsCBZ().get()) {
archiveChapter(mangaDir, dirname, tmpDir)
} else {
tmpDir.renameTo(dirname)
}
cache.addChapter(dirname, mangaDir, download.manga)
DiskUtil.createNoMediaFile(tmpDir, context)
download.status = Download.State.DOWNLOADED
} }
/** /**
@ -624,20 +629,6 @@ class Downloader(
} }
} }
/**
* Completes a download. This method is called in the main thread.
*/
private fun completeDownload(download: Download) {
// Delete successful downloads from queue
if (download.status == Download.State.DOWNLOADED) {
// Remove downloaded chapter from queue
removeFromQueue(download)
}
if (areAllDownloadsFinished()) {
stop()
}
}
/** /**
* Returns true if all the queued downloads are in DOWNLOADED or ERROR state. * Returns true if all the queued downloads are in DOWNLOADED or ERROR state.
*/ */
@ -665,14 +656,26 @@ class Downloader(
} }
} }
fun removeFromQueue(chapters: List<Chapter>) { private inline fun removeFromQueueByPredicate(predicate: (Download) -> Boolean) {
chapters.forEach { chapter -> _queueState.update { queue ->
queueState.value.find { it.chapter.id == chapter.id }?.let { removeFromQueue(it) } val downloads = queue.filter { predicate(it) }
store.removeAll(downloads)
downloads.forEach { download ->
if (download.status == Download.State.DOWNLOADING || download.status == Download.State.QUEUE) {
download.status = Download.State.NOT_DOWNLOADED
}
}
queue - downloads
} }
} }
fun removeFromQueue(chapters: List<Chapter>) {
val chapterIds = chapters.map { it.id }
removeFromQueueByPredicate { it.chapter.id in chapterIds }
}
fun removeFromQueue(manga: Manga) { fun removeFromQueue(manga: Manga) {
queueState.value.filter { it.manga.id == manga.id }.forEach { removeFromQueue(it) } removeFromQueueByPredicate { it.manga.id == manga.id }
} }
private fun _clearQueue() { private fun _clearQueue() {