Downloader.kt 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641
  1. package eu.kanade.tachiyomi.data.download
  2. import android.content.Context
  3. import android.graphics.Bitmap
  4. import android.graphics.BitmapFactory
  5. import android.webkit.MimeTypeMap
  6. import androidx.core.graphics.BitmapCompat
  7. import com.hippo.unifile.UniFile
  8. import com.jakewharton.rxrelay.BehaviorRelay
  9. import com.jakewharton.rxrelay.PublishRelay
  10. import eu.kanade.tachiyomi.R
  11. import eu.kanade.tachiyomi.data.cache.ChapterCache
  12. import eu.kanade.tachiyomi.data.database.models.Chapter
  13. import eu.kanade.tachiyomi.data.database.models.Manga
  14. import eu.kanade.tachiyomi.data.download.model.Download
  15. import eu.kanade.tachiyomi.data.download.model.DownloadQueue
  16. import eu.kanade.tachiyomi.data.library.LibraryUpdateNotifier
  17. import eu.kanade.tachiyomi.data.notification.NotificationHandler
  18. import eu.kanade.tachiyomi.data.preference.PreferencesHelper
  19. import eu.kanade.tachiyomi.source.SourceManager
  20. import eu.kanade.tachiyomi.source.UnmeteredSource
  21. import eu.kanade.tachiyomi.source.model.Page
  22. import eu.kanade.tachiyomi.source.online.HttpSource
  23. import eu.kanade.tachiyomi.source.online.fetchAllImageUrlsFromPageList
  24. import eu.kanade.tachiyomi.util.lang.RetryWithDelay
  25. import eu.kanade.tachiyomi.util.lang.launchIO
  26. import eu.kanade.tachiyomi.util.lang.launchNow
  27. import eu.kanade.tachiyomi.util.lang.plusAssign
  28. import eu.kanade.tachiyomi.util.lang.withUIContext
  29. import eu.kanade.tachiyomi.util.storage.DiskUtil
  30. import eu.kanade.tachiyomi.util.storage.saveTo
  31. import eu.kanade.tachiyomi.util.system.ImageUtil
  32. import eu.kanade.tachiyomi.util.system.ImageUtil.isAnimatedAndSupported
  33. import eu.kanade.tachiyomi.util.system.ImageUtil.isTallImage
  34. import eu.kanade.tachiyomi.util.system.logcat
  35. import kotlinx.coroutines.async
  36. import logcat.LogPriority
  37. import okhttp3.Response
  38. import rx.Observable
  39. import rx.android.schedulers.AndroidSchedulers
  40. import rx.schedulers.Schedulers
  41. import rx.subscriptions.CompositeSubscription
  42. import uy.kohesive.injekt.injectLazy
  43. import java.io.BufferedOutputStream
  44. import java.io.File
  45. import java.io.FileOutputStream
  46. import java.io.OutputStream
  47. import java.util.zip.CRC32
  48. import java.util.zip.ZipEntry
  49. import java.util.zip.ZipOutputStream
  50. /**
  51. * This class is the one in charge of downloading chapters.
  52. *
  53. * Its [queue] contains the list of chapters to download. In order to download them, the downloader
  54. * subscriptions must be running and the list of chapters must be sent to them by [downloadsRelay].
  55. *
  56. * The queue manipulation must be done in one thread (currently the main thread) to avoid unexpected
  57. * behavior, but it's safe to read it from multiple threads.
  58. *
  59. * @param context the application context.
  60. * @param provider the downloads directory provider.
  61. * @param cache the downloads cache, used to add the downloads to the cache after their completion.
  62. * @param sourceManager the source manager.
  63. */
  64. class Downloader(
  65. private val context: Context,
  66. private val provider: DownloadProvider,
  67. private val cache: DownloadCache,
  68. private val sourceManager: SourceManager,
  69. ) {
  70. private val chapterCache: ChapterCache by injectLazy()
  71. private val preferences: PreferencesHelper by injectLazy()
  72. /**
  73. * Store for persisting downloads across restarts.
  74. */
  75. private val store = DownloadStore(context, sourceManager)
  76. /**
  77. * Queue where active downloads are kept.
  78. */
  79. val queue = DownloadQueue(store)
  80. /**
  81. * Notifier for the downloader state and progress.
  82. */
  83. private val notifier by lazy { DownloadNotifier(context) }
  84. /**
  85. * Downloader subscriptions.
  86. */
  87. private val subscriptions = CompositeSubscription()
  88. /**
  89. * Relay to send a list of downloads to the downloader.
  90. */
  91. private val downloadsRelay = PublishRelay.create<List<Download>>()
  92. /**
  93. * Relay to subscribe to the downloader status.
  94. */
  95. val runningRelay: BehaviorRelay<Boolean> = BehaviorRelay.create(false)
  96. /**
  97. * Whether the downloader is running.
  98. */
  99. @Volatile
  100. var isRunning: Boolean = false
  101. private set
  102. init {
  103. launchNow {
  104. val chapters = async { store.restore() }
  105. queue.addAll(chapters.await())
  106. }
  107. }
  108. /**
  109. * Starts the downloader. It doesn't do anything if it's already running or there isn't anything
  110. * to download.
  111. *
  112. * @return true if the downloader is started, false otherwise.
  113. */
  114. fun start(): Boolean {
  115. if (isRunning || queue.isEmpty()) {
  116. return false
  117. }
  118. if (!subscriptions.hasSubscriptions()) {
  119. initializeSubscriptions()
  120. }
  121. val pending = queue.filter { it.status != Download.State.DOWNLOADED }
  122. pending.forEach { if (it.status != Download.State.QUEUE) it.status = Download.State.QUEUE }
  123. notifier.paused = false
  124. downloadsRelay.call(pending)
  125. return pending.isNotEmpty()
  126. }
  127. /**
  128. * Stops the downloader.
  129. */
  130. fun stop(reason: String? = null) {
  131. destroySubscriptions()
  132. queue
  133. .filter { it.status == Download.State.DOWNLOADING }
  134. .forEach { it.status = Download.State.ERROR }
  135. if (reason != null) {
  136. notifier.onWarning(reason)
  137. return
  138. }
  139. if (notifier.paused && !queue.isEmpty()) {
  140. notifier.onPaused()
  141. } else {
  142. notifier.onComplete()
  143. }
  144. notifier.paused = false
  145. }
  146. /**
  147. * Pauses the downloader
  148. */
  149. fun pause() {
  150. destroySubscriptions()
  151. queue
  152. .filter { it.status == Download.State.DOWNLOADING }
  153. .forEach { it.status = Download.State.QUEUE }
  154. notifier.paused = true
  155. }
  156. /**
  157. * Check if downloader is paused
  158. */
  159. fun isPaused() = !isRunning
  160. /**
  161. * Removes everything from the queue.
  162. *
  163. * @param isNotification value that determines if status is set (needed for view updates)
  164. */
  165. fun clearQueue(isNotification: Boolean = false) {
  166. destroySubscriptions()
  167. // Needed to update the chapter view
  168. if (isNotification) {
  169. queue
  170. .filter { it.status == Download.State.QUEUE }
  171. .forEach { it.status = Download.State.NOT_DOWNLOADED }
  172. }
  173. queue.clear()
  174. notifier.dismissProgress()
  175. }
  176. /**
  177. * Prepares the subscriptions to start downloading.
  178. */
  179. private fun initializeSubscriptions() {
  180. if (isRunning) return
  181. isRunning = true
  182. runningRelay.call(true)
  183. subscriptions.clear()
  184. subscriptions += downloadsRelay.concatMapIterable { it }
  185. // Concurrently download from 5 different sources
  186. .groupBy { it.source }
  187. .flatMap(
  188. { bySource ->
  189. bySource.concatMap { download ->
  190. downloadChapter(download).subscribeOn(Schedulers.io())
  191. }
  192. },
  193. 5,
  194. )
  195. .onBackpressureLatest()
  196. .observeOn(AndroidSchedulers.mainThread())
  197. .subscribe(
  198. {
  199. completeDownload(it)
  200. },
  201. { error ->
  202. DownloadService.stop(context)
  203. logcat(LogPriority.ERROR, error)
  204. notifier.onError(error.message)
  205. },
  206. )
  207. }
  208. /**
  209. * Destroys the downloader subscriptions.
  210. */
  211. private fun destroySubscriptions() {
  212. if (!isRunning) return
  213. isRunning = false
  214. runningRelay.call(false)
  215. subscriptions.clear()
  216. }
  217. /**
  218. * Creates a download object for every chapter and adds them to the downloads queue.
  219. *
  220. * @param manga the manga of the chapters to download.
  221. * @param chapters the list of chapters to download.
  222. * @param autoStart whether to start the downloader after enqueing the chapters.
  223. */
  224. fun queueChapters(manga: Manga, chapters: List<Chapter>, autoStart: Boolean) = launchIO {
  225. val source = sourceManager.get(manga.source) as? HttpSource ?: return@launchIO
  226. val wasEmpty = queue.isEmpty()
  227. // Called in background thread, the operation can be slow with SAF.
  228. val chaptersWithoutDir = async {
  229. chapters
  230. // Filter out those already downloaded.
  231. .filter { provider.findChapterDir(it, manga, source) == null }
  232. // Add chapters to queue from the start.
  233. .sortedByDescending { it.source_order }
  234. }
  235. // Runs in main thread (synchronization needed).
  236. val chaptersToQueue = chaptersWithoutDir.await()
  237. // Filter out those already enqueued.
  238. .filter { chapter -> queue.none { it.chapter.id == chapter.id } }
  239. // Create a download for each one.
  240. .map { Download(source, manga, it) }
  241. if (chaptersToQueue.isNotEmpty()) {
  242. queue.addAll(chaptersToQueue)
  243. if (isRunning) {
  244. // Send the list of downloads to the downloader.
  245. downloadsRelay.call(chaptersToQueue)
  246. }
  247. // Start downloader if needed
  248. if (autoStart && wasEmpty) {
  249. val queuedDownloads = queue.filter { it.source !is UnmeteredSource }.count()
  250. val maxDownloadsFromSource = queue
  251. .groupBy { it.source }
  252. .filterKeys { it !is UnmeteredSource }
  253. .maxOf { it.value.size }
  254. if (
  255. queuedDownloads > DOWNLOADS_QUEUED_WARNING_THRESHOLD ||
  256. maxDownloadsFromSource > CHAPTERS_PER_SOURCE_QUEUE_WARNING_THRESHOLD
  257. ) {
  258. withUIContext {
  259. notifier.onWarning(
  260. context.getString(R.string.download_queue_size_warning),
  261. WARNING_NOTIF_TIMEOUT_MS,
  262. NotificationHandler.openUrl(context, LibraryUpdateNotifier.HELP_WARNING_URL),
  263. )
  264. }
  265. }
  266. DownloadService.start(context)
  267. }
  268. }
  269. }
  270. /**
  271. * Returns the observable which downloads a chapter.
  272. *
  273. * @param download the chapter to be downloaded.
  274. */
  275. private fun downloadChapter(download: Download): Observable<Download> = Observable.defer {
  276. val mangaDir = provider.getMangaDir(download.manga, download.source)
  277. val availSpace = DiskUtil.getAvailableStorageSpace(mangaDir)
  278. if (availSpace != -1L && availSpace < MIN_DISK_SPACE) {
  279. download.status = Download.State.ERROR
  280. notifier.onError(context.getString(R.string.download_insufficient_space), download.chapter.name, download.manga.title)
  281. return@defer Observable.just(download)
  282. }
  283. val chapterDirname = provider.getChapterDirName(download.chapter)
  284. val tmpDir = mangaDir.createDirectory(chapterDirname + TMP_DIR_SUFFIX)
  285. val pageListObservable = if (download.pages == null) {
  286. // Pull page list from network and add them to download object
  287. download.source.fetchPageList(download.chapter)
  288. .doOnNext { pages ->
  289. if (pages.isEmpty()) {
  290. throw Exception(context.getString(R.string.page_list_empty_error))
  291. }
  292. download.pages = pages
  293. }
  294. } else {
  295. // Or if the page list already exists, start from the file
  296. Observable.just(download.pages!!)
  297. }
  298. pageListObservable
  299. .doOnNext { _ ->
  300. // Delete all temporary (unfinished) files
  301. tmpDir.listFiles()
  302. ?.filter { it.name!!.endsWith(".tmp") }
  303. ?.forEach { it.delete() }
  304. download.downloadedImages = 0
  305. download.status = Download.State.DOWNLOADING
  306. }
  307. // Get all the URLs to the source images, fetch pages if necessary
  308. .flatMap { download.source.fetchAllImageUrlsFromPageList(it) }
  309. // Start downloading images, consider we can have downloaded images already
  310. // Concurrently do 5 pages at a time
  311. .flatMap({ page -> getOrDownloadImage(page, download, tmpDir) }, 5)
  312. .onBackpressureLatest()
  313. // Do when page is downloaded.
  314. .doOnNext { page ->
  315. if (preferences.splitTallImages().get()) {
  316. splitTallImage(page, download, tmpDir)
  317. }
  318. notifier.onProgressChange(download)
  319. }
  320. .toList()
  321. .map { download }
  322. // Do after download completes
  323. .doOnNext { ensureSuccessfulDownload(download, mangaDir, tmpDir, chapterDirname) }
  324. // If the page list threw, it will resume here
  325. .onErrorReturn { error ->
  326. download.status = Download.State.ERROR
  327. notifier.onError(error.message, download.chapter.name, download.manga.title)
  328. download
  329. }
  330. }
  331. /**
  332. * Returns the observable which gets the image from the filesystem if it exists or downloads it
  333. * otherwise.
  334. *
  335. * @param page the page to download.
  336. * @param download the download of the page.
  337. * @param tmpDir the temporary directory of the download.
  338. */
  339. private fun getOrDownloadImage(page: Page, download: Download, tmpDir: UniFile): Observable<Page> {
  340. // If the image URL is empty, do nothing
  341. if (page.imageUrl == null) {
  342. return Observable.just(page)
  343. }
  344. val filename = String.format("%03d", page.number)
  345. val tmpFile = tmpDir.findFile("$filename.tmp")
  346. // Delete temp file if it exists.
  347. tmpFile?.delete()
  348. // Try to find the image file.
  349. val imageFile = tmpDir.listFiles()!!.find { it.name!!.startsWith("$filename.") || it.name!!.contains("${filename}__001") }
  350. // If the image is already downloaded, do nothing. Otherwise download from network
  351. val pageObservable = when {
  352. imageFile != null -> Observable.just(imageFile)
  353. chapterCache.isImageInCache(page.imageUrl!!) -> copyImageFromCache(chapterCache.getImageFile(page.imageUrl!!), tmpDir, filename)
  354. else -> downloadImage(page, download.source, tmpDir, filename)
  355. }
  356. return pageObservable
  357. // When the image is ready, set image path, progress (just in case) and status
  358. .doOnNext { file ->
  359. page.uri = file.uri
  360. page.progress = 100
  361. download.downloadedImages++
  362. page.status = Page.READY
  363. }
  364. .map { page }
  365. // Mark this page as error and allow to download the remaining
  366. .onErrorReturn {
  367. page.progress = 0
  368. page.status = Page.ERROR
  369. notifier.onError(it.message, download.chapter.name, download.manga.title)
  370. page
  371. }
  372. }
  373. /**
  374. * Returns the observable which downloads the image from network.
  375. *
  376. * @param page the page to download.
  377. * @param source the source of the page.
  378. * @param tmpDir the temporary directory of the download.
  379. * @param filename the filename of the image.
  380. */
  381. private fun downloadImage(page: Page, source: HttpSource, tmpDir: UniFile, filename: String): Observable<UniFile> {
  382. page.status = Page.DOWNLOAD_IMAGE
  383. page.progress = 0
  384. return source.fetchImage(page)
  385. .map { response ->
  386. val file = tmpDir.createFile("$filename.tmp")
  387. try {
  388. response.body!!.source().saveTo(file.openOutputStream())
  389. val extension = getImageExtension(response, file)
  390. file.renameTo("$filename.$extension")
  391. } catch (e: Exception) {
  392. response.close()
  393. file.delete()
  394. throw e
  395. }
  396. file
  397. }
  398. // Retry 3 times, waiting 2, 4 and 8 seconds between attempts.
  399. .retryWhen(RetryWithDelay(3, { (2 shl it - 1) * 1000 }, Schedulers.trampoline()))
  400. }
  401. /**
  402. * Return the observable which copies the image from cache.
  403. *
  404. * @param cacheFile the file from cache.
  405. * @param tmpDir the temporary directory of the download.
  406. * @param filename the filename of the image.
  407. */
  408. private fun copyImageFromCache(cacheFile: File, tmpDir: UniFile, filename: String): Observable<UniFile> {
  409. return Observable.just(cacheFile).map {
  410. val tmpFile = tmpDir.createFile("$filename.tmp")
  411. cacheFile.inputStream().use { input ->
  412. tmpFile.openOutputStream().use { output ->
  413. input.copyTo(output)
  414. }
  415. }
  416. val extension = ImageUtil.findImageType(cacheFile.inputStream()) ?: return@map tmpFile
  417. tmpFile.renameTo("$filename.${extension.extension}")
  418. cacheFile.delete()
  419. tmpFile
  420. }
  421. }
  422. /**
  423. * Returns the extension of the downloaded image from the network response, or if it's null,
  424. * analyze the file. If everything fails, assume it's a jpg.
  425. *
  426. * @param response the network response of the image.
  427. * @param file the file where the image is already downloaded.
  428. */
  429. private fun getImageExtension(response: Response, file: UniFile): String {
  430. // Read content type if available.
  431. val mime = response.body?.contentType()?.let { ct -> "${ct.type}/${ct.subtype}" }
  432. // Else guess from the uri.
  433. ?: context.contentResolver.getType(file.uri)
  434. // Else read magic numbers.
  435. ?: ImageUtil.findImageType { file.openInputStream() }?.mime
  436. return MimeTypeMap.getSingleton().getExtensionFromMimeType(mime) ?: "jpg"
  437. }
  438. /**
  439. * Checks if the download was successful.
  440. *
  441. * @param download the download to check.
  442. * @param mangaDir the manga directory of the download.
  443. * @param tmpDir the directory where the download is currently stored.
  444. * @param dirname the real (non temporary) directory name of the download.
  445. */
  446. private fun ensureSuccessfulDownload(
  447. download: Download,
  448. mangaDir: UniFile,
  449. tmpDir: UniFile,
  450. dirname: String,
  451. ) {
  452. // Ensure that the chapter folder has all the images.
  453. val downloadedImages = tmpDir.listFiles().orEmpty().filterNot { it.name!!.endsWith(".tmp") || (it.name!!.contains("__") && !it.name!!.contains("__001.jpg")) }
  454. download.status = if (downloadedImages.size == download.pages!!.size) {
  455. Download.State.DOWNLOADED
  456. } else {
  457. Download.State.ERROR
  458. }
  459. // Only rename the directory if it's downloaded.
  460. if (download.status == Download.State.DOWNLOADED) {
  461. if (preferences.saveChaptersAsCBZ().get()) {
  462. archiveChapter(mangaDir, dirname, tmpDir)
  463. } else {
  464. tmpDir.renameTo(dirname)
  465. }
  466. cache.addChapter(dirname, mangaDir, download.manga)
  467. DiskUtil.createNoMediaFile(tmpDir, context)
  468. }
  469. }
  470. /**
  471. * Archive the chapter pages as a CBZ.
  472. */
  473. private fun archiveChapter(
  474. mangaDir: UniFile,
  475. dirname: String,
  476. tmpDir: UniFile,
  477. ) {
  478. val zip = mangaDir.createFile("$dirname.cbz.tmp")
  479. ZipOutputStream(BufferedOutputStream(zip.openOutputStream())).use { zipOut ->
  480. zipOut.setMethod(ZipEntry.STORED)
  481. tmpDir.listFiles()?.forEach { img ->
  482. img.openInputStream().use { input ->
  483. val data = input.readBytes()
  484. val size = img.length()
  485. val entry = ZipEntry(img.name).apply {
  486. val crc = CRC32().apply {
  487. update(data)
  488. }
  489. setCrc(crc.value)
  490. compressedSize = size
  491. setSize(size)
  492. }
  493. zipOut.putNextEntry(entry)
  494. zipOut.write(data)
  495. }
  496. }
  497. }
  498. zip.renameTo("$dirname.cbz")
  499. tmpDir.delete()
  500. }
  501. /**
  502. * Splits tall images to improve performance of reader
  503. */
  504. private fun splitTallImage(page: Page, download: Download, tmpDir: UniFile) {
  505. val filename = String.format("%03d", page.number)
  506. val imageFile = tmpDir.listFiles()!!.find { it.name!!.startsWith("$filename.") }
  507. if (imageFile == null) {
  508. notifier.onError("Error: imageFile was not found", download.chapter.name, download.manga.title)
  509. return
  510. }
  511. if (!isAnimatedAndSupported(imageFile.openInputStream()) && isTallImage(imageFile.openInputStream())) {
  512. // Getting the scaled bitmap of the source image
  513. val bitmap = BitmapFactory.decodeFile(imageFile.filePath)
  514. val scaledBitmap: Bitmap =
  515. BitmapCompat.createScaledBitmap(bitmap, bitmap.width, bitmap.height, null, true)
  516. val splitsCount: Int = bitmap.height / context.resources.displayMetrics.heightPixels + 1
  517. val splitHeight = bitmap.height / splitsCount
  518. // xCoord and yCoord are the pixel positions of the image splits
  519. val xCoord = 0
  520. var yCoord = 0
  521. try {
  522. for (i in 0 until splitsCount) {
  523. val splitPath = imageFile.filePath!!.substringBeforeLast(".") + "__${"%03d".format(i + 1)}.jpg"
  524. // Compress the bitmap and save in jpg format
  525. val stream: OutputStream = FileOutputStream(splitPath)
  526. stream.use {
  527. Bitmap.createBitmap(
  528. scaledBitmap,
  529. xCoord,
  530. yCoord,
  531. bitmap.width,
  532. splitHeight,
  533. ).compress(Bitmap.CompressFormat.JPEG, 100, stream)
  534. }
  535. yCoord += splitHeight
  536. }
  537. imageFile.delete()
  538. } catch (e: Exception) {
  539. // Image splits were not successfully saved so delete them and keep the original image
  540. for (i in 0 until splitsCount) {
  541. val splitPath = imageFile.filePath!!.substringBeforeLast(".") + "__${"%03d".format(i + 1)}.jpg"
  542. File(splitPath).delete()
  543. }
  544. throw e
  545. }
  546. }
  547. }
  548. /**
  549. * Completes a download. This method is called in the main thread.
  550. */
  551. private fun completeDownload(download: Download) {
  552. // Delete successful downloads from queue
  553. if (download.status == Download.State.DOWNLOADED) {
  554. // remove downloaded chapter from queue
  555. queue.remove(download)
  556. }
  557. if (areAllDownloadsFinished()) {
  558. DownloadService.stop(context)
  559. }
  560. }
  561. /**
  562. * Returns true if all the queued downloads are in DOWNLOADED or ERROR state.
  563. */
  564. private fun areAllDownloadsFinished(): Boolean {
  565. return queue.none { it.status.value <= Download.State.DOWNLOADING.value }
  566. }
  567. companion object {
  568. const val TMP_DIR_SUFFIX = "_tmp"
  569. const val WARNING_NOTIF_TIMEOUT_MS = 30_000L
  570. const val CHAPTERS_PER_SOURCE_QUEUE_WARNING_THRESHOLD = 15
  571. private const val DOWNLOADS_QUEUED_WARNING_THRESHOLD = 30
  572. }
  573. }
  574. // Arbitrary minimum required space to start a download: 200 MB
  575. private const val MIN_DISK_SPACE = 200L * 1024 * 1024