LibraryUpdateService.kt 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579
  1. package eu.kanade.tachiyomi.data.library
  2. import android.app.Service
  3. import android.content.Context
  4. import android.content.Intent
  5. import android.os.IBinder
  6. import android.os.PowerManager
  7. import android.widget.Toast
  8. import androidx.core.content.ContextCompat
  9. import eu.kanade.tachiyomi.R
  10. import eu.kanade.tachiyomi.data.cache.CoverCache
  11. import eu.kanade.tachiyomi.data.database.DatabaseHelper
  12. import eu.kanade.tachiyomi.data.database.models.Category
  13. import eu.kanade.tachiyomi.data.database.models.Chapter
  14. import eu.kanade.tachiyomi.data.database.models.LibraryManga
  15. import eu.kanade.tachiyomi.data.database.models.Manga
  16. import eu.kanade.tachiyomi.data.database.models.toMangaInfo
  17. import eu.kanade.tachiyomi.data.download.DownloadManager
  18. import eu.kanade.tachiyomi.data.download.DownloadService
  19. import eu.kanade.tachiyomi.data.library.LibraryUpdateService.Companion.start
  20. import eu.kanade.tachiyomi.data.notification.Notifications
  21. import eu.kanade.tachiyomi.data.preference.MANGA_FULLY_READ
  22. import eu.kanade.tachiyomi.data.preference.MANGA_ONGOING
  23. import eu.kanade.tachiyomi.data.preference.PreferencesHelper
  24. import eu.kanade.tachiyomi.data.track.EnhancedTrackService
  25. import eu.kanade.tachiyomi.data.track.TrackManager
  26. import eu.kanade.tachiyomi.data.track.TrackService
  27. import eu.kanade.tachiyomi.source.SourceManager
  28. import eu.kanade.tachiyomi.source.UnmeteredSource
  29. import eu.kanade.tachiyomi.source.model.SManga
  30. import eu.kanade.tachiyomi.source.model.toSChapter
  31. import eu.kanade.tachiyomi.source.model.toSManga
  32. import eu.kanade.tachiyomi.util.chapter.NoChaptersException
  33. import eu.kanade.tachiyomi.util.chapter.syncChaptersWithSource
  34. import eu.kanade.tachiyomi.util.chapter.syncChaptersWithTrackServiceTwoWay
  35. import eu.kanade.tachiyomi.util.lang.withIOContext
  36. import eu.kanade.tachiyomi.util.prepUpdateCover
  37. import eu.kanade.tachiyomi.util.shouldDownloadNewChapters
  38. import eu.kanade.tachiyomi.util.storage.getUriCompat
  39. import eu.kanade.tachiyomi.util.system.acquireWakeLock
  40. import eu.kanade.tachiyomi.util.system.createFileInCacheDir
  41. import eu.kanade.tachiyomi.util.system.isServiceRunning
  42. import eu.kanade.tachiyomi.util.system.logcat
  43. import eu.kanade.tachiyomi.util.system.toast
  44. import kotlinx.coroutines.CoroutineExceptionHandler
  45. import kotlinx.coroutines.CoroutineScope
  46. import kotlinx.coroutines.Dispatchers
  47. import kotlinx.coroutines.Job
  48. import kotlinx.coroutines.SupervisorJob
  49. import kotlinx.coroutines.async
  50. import kotlinx.coroutines.awaitAll
  51. import kotlinx.coroutines.cancel
  52. import kotlinx.coroutines.launch
  53. import kotlinx.coroutines.supervisorScope
  54. import kotlinx.coroutines.sync.Semaphore
  55. import kotlinx.coroutines.sync.withPermit
  56. import logcat.LogPriority
  57. import uy.kohesive.injekt.Injekt
  58. import uy.kohesive.injekt.api.get
  59. import java.io.File
  60. import java.util.concurrent.CopyOnWriteArrayList
  61. import java.util.concurrent.atomic.AtomicBoolean
  62. import java.util.concurrent.atomic.AtomicInteger
  63. /**
  64. * This class will take care of updating the chapters of the manga from the library. It can be
  65. * started calling the [start] method. If it's already running, it won't do anything.
  66. * While the library is updating, a [PowerManager.WakeLock] will be held until the update is
  67. * completed, preventing the device from going to sleep mode. A notification will display the
  68. * progress of the update, and if case of an unexpected error, this service will be silently
  69. * destroyed.
  70. */
  71. class LibraryUpdateService(
  72. val db: DatabaseHelper = Injekt.get(),
  73. val sourceManager: SourceManager = Injekt.get(),
  74. val preferences: PreferencesHelper = Injekt.get(),
  75. val downloadManager: DownloadManager = Injekt.get(),
  76. val trackManager: TrackManager = Injekt.get(),
  77. val coverCache: CoverCache = Injekt.get()
  78. ) : Service() {
  79. private lateinit var wakeLock: PowerManager.WakeLock
  80. private lateinit var notifier: LibraryUpdateNotifier
  81. private lateinit var ioScope: CoroutineScope
  82. private var mangaToUpdate: List<LibraryManga> = mutableListOf()
  83. private var updateJob: Job? = null
  84. /**
  85. * Defines what should be updated within a service execution.
  86. */
  87. enum class Target {
  88. CHAPTERS, // Manga chapters
  89. COVERS, // Manga covers
  90. TRACKING // Tracking metadata
  91. }
  92. companion object {
  93. private var instance: LibraryUpdateService? = null
  94. /**
  95. * Key for category to update.
  96. */
  97. const val KEY_CATEGORY = "category"
  98. /**
  99. * Key that defines what should be updated.
  100. */
  101. const val KEY_TARGET = "target"
  102. /**
  103. * Returns the status of the service.
  104. *
  105. * @param context the application context.
  106. * @return true if the service is running, false otherwise.
  107. */
  108. fun isRunning(context: Context): Boolean {
  109. return context.isServiceRunning(LibraryUpdateService::class.java)
  110. }
  111. /**
  112. * Starts the service. It will be started only if there isn't another instance already
  113. * running.
  114. *
  115. * @param context the application context.
  116. * @param category a specific category to update, or null for global update.
  117. * @param target defines what should be updated.
  118. * @return true if service newly started, false otherwise
  119. */
  120. fun start(context: Context, category: Category? = null, target: Target = Target.CHAPTERS): Boolean {
  121. return if (!isRunning(context)) {
  122. val intent = Intent(context, LibraryUpdateService::class.java).apply {
  123. putExtra(KEY_TARGET, target)
  124. category?.let { putExtra(KEY_CATEGORY, it.id) }
  125. }
  126. ContextCompat.startForegroundService(context, intent)
  127. true
  128. } else {
  129. instance?.addMangaToQueue(category?.id ?: -1, target)
  130. false
  131. }
  132. }
  133. /**
  134. * Stops the service.
  135. *
  136. * @param context the application context.
  137. */
  138. fun stop(context: Context) {
  139. context.stopService(Intent(context, LibraryUpdateService::class.java))
  140. }
  141. }
  142. /**
  143. * Method called when the service is created. It injects dagger dependencies and acquire
  144. * the wake lock.
  145. */
  146. override fun onCreate() {
  147. super.onCreate()
  148. ioScope = CoroutineScope(SupervisorJob() + Dispatchers.IO)
  149. notifier = LibraryUpdateNotifier(this)
  150. wakeLock = acquireWakeLock(javaClass.name)
  151. startForeground(Notifications.ID_LIBRARY_PROGRESS, notifier.progressNotificationBuilder.build())
  152. }
  153. /**
  154. * Method called when the service is destroyed. It destroys subscriptions and releases the wake
  155. * lock.
  156. */
  157. override fun onDestroy() {
  158. updateJob?.cancel()
  159. ioScope?.cancel()
  160. if (wakeLock.isHeld) {
  161. wakeLock.release()
  162. }
  163. if (instance == this) {
  164. instance = null
  165. }
  166. super.onDestroy()
  167. }
  168. /**
  169. * This method needs to be implemented, but it's not used/needed.
  170. */
  171. override fun onBind(intent: Intent): IBinder? {
  172. return null
  173. }
  174. /**
  175. * Method called when the service receives an intent.
  176. *
  177. * @param intent the start intent from.
  178. * @param flags the flags of the command.
  179. * @param startId the start id of this command.
  180. * @return the start value of the command.
  181. */
  182. override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
  183. if (intent == null) return START_NOT_STICKY
  184. val target = intent.getSerializableExtra(KEY_TARGET) as? Target
  185. ?: return START_NOT_STICKY
  186. instance = this
  187. // Unsubscribe from any previous subscription if needed
  188. updateJob?.cancel()
  189. // Update favorite manga
  190. val categoryId = intent.getIntExtra(KEY_CATEGORY, -1)
  191. addMangaToQueue(categoryId, target)
  192. // Destroy service when completed or in case of an error.
  193. val handler = CoroutineExceptionHandler { _, exception ->
  194. logcat(LogPriority.ERROR, exception)
  195. stopSelf(startId)
  196. }
  197. updateJob = ioScope.launch(handler) {
  198. when (target) {
  199. Target.CHAPTERS -> updateChapterList()
  200. Target.COVERS -> updateCovers()
  201. Target.TRACKING -> updateTrackings()
  202. }
  203. }
  204. updateJob?.invokeOnCompletion { stopSelf(startId) }
  205. return START_REDELIVER_INTENT
  206. }
  207. /**
  208. * Adds list of manga to be updated.
  209. *
  210. * @param category the ID of the category to update, or -1 if no category specified.
  211. * @param target the target to update.
  212. */
  213. fun addMangaToQueue(categoryId: Int, target: Target) {
  214. val libraryManga = db.getLibraryMangas().executeAsBlocking()
  215. var listToUpdate = if (categoryId != -1) {
  216. libraryManga.filter { it.category == categoryId }
  217. } else {
  218. val categoriesToUpdate = preferences.libraryUpdateCategories().get().map(String::toInt)
  219. val listToInclude = if (categoriesToUpdate.isNotEmpty()) {
  220. libraryManga.filter { it.category in categoriesToUpdate }
  221. } else {
  222. libraryManga
  223. }
  224. val categoriesToExclude = preferences.libraryUpdateCategoriesExclude().get().map(String::toInt)
  225. val listToExclude = if (categoriesToExclude.isNotEmpty()) {
  226. libraryManga.filter { it.category in categoriesToExclude }
  227. } else {
  228. emptyList()
  229. }
  230. listToInclude.minus(listToExclude)
  231. }
  232. if (target == Target.CHAPTERS) {
  233. val restrictions = preferences.libraryUpdateMangaRestriction().get()
  234. if (MANGA_ONGOING in restrictions) {
  235. listToUpdate = listToUpdate.filterNot { it.status == SManga.COMPLETED }
  236. }
  237. if (MANGA_FULLY_READ in restrictions) {
  238. listToUpdate = listToUpdate.filter { it.unread == 0 }
  239. }
  240. }
  241. mangaToUpdate = listToUpdate
  242. .distinctBy { it.id }
  243. .sortedBy { it.title }
  244. // Warn when excessively checking a single source
  245. val maxUpdatesFromSource = mangaToUpdate
  246. .groupBy { it.source }
  247. .filterKeys { sourceManager.get(it) !is UnmeteredSource }
  248. .maxOfOrNull { it.value.size } ?: 0
  249. if (maxUpdatesFromSource > MANGA_PER_SOURCE_QUEUE_WARNING_THRESHOLD) {
  250. toast(R.string.notification_size_warning, Toast.LENGTH_LONG)
  251. }
  252. }
  253. /**
  254. * Method that updates the given list of manga. It's called in a background thread, so it's safe
  255. * to do heavy operations or network calls here.
  256. * For each manga it calls [updateManga] and updates the notification showing the current
  257. * progress.
  258. *
  259. * @param mangaToUpdate the list to update
  260. * @return an observable delivering the progress of each update.
  261. */
  262. suspend fun updateChapterList() {
  263. val semaphore = Semaphore(5)
  264. val progressCount = AtomicInteger(0)
  265. val currentlyUpdatingManga = CopyOnWriteArrayList<LibraryManga>()
  266. val newUpdates = CopyOnWriteArrayList<Pair<LibraryManga, Array<Chapter>>>()
  267. val failedUpdates = CopyOnWriteArrayList<Pair<Manga, String?>>()
  268. val hasDownloads = AtomicBoolean(false)
  269. val loggedServices by lazy { trackManager.services.filter { it.isLogged } }
  270. val currentUnreadUpdatesCount = preferences.unreadUpdatesCount().get()
  271. withIOContext {
  272. mangaToUpdate.groupBy { it.source }
  273. .values
  274. .map { mangaInSource ->
  275. async {
  276. semaphore.withPermit {
  277. mangaInSource.forEach { manga ->
  278. if (updateJob?.isActive != true) {
  279. return@async
  280. }
  281. withUpdateNotification(
  282. currentlyUpdatingManga,
  283. progressCount,
  284. manga,
  285. ) { manga ->
  286. try {
  287. val (newChapters, _) = updateManga(manga)
  288. if (newChapters.isNotEmpty()) {
  289. if (manga.shouldDownloadNewChapters(db, preferences)) {
  290. downloadChapters(manga, newChapters)
  291. hasDownloads.set(true)
  292. }
  293. // Convert to the manga that contains new chapters
  294. newUpdates.add(
  295. manga to newChapters.sortedByDescending { ch -> ch.source_order }
  296. .toTypedArray()
  297. )
  298. }
  299. } catch (e: Throwable) {
  300. val errorMessage = when (e) {
  301. is NoChaptersException -> {
  302. getString(R.string.no_chapters_error)
  303. }
  304. is SourceManager.SourceNotInstalledException -> {
  305. // failedUpdates will already have the source, don't need to copy it into the message
  306. getString(R.string.loader_not_implemented_error)
  307. }
  308. else -> {
  309. e.message
  310. }
  311. }
  312. failedUpdates.add(manga to errorMessage)
  313. }
  314. if (preferences.autoUpdateTrackers()) {
  315. updateTrackings(manga, loggedServices)
  316. }
  317. }
  318. }
  319. }
  320. }
  321. }
  322. .awaitAll()
  323. }
  324. notifier.cancelProgressNotification()
  325. if (newUpdates.isNotEmpty()) {
  326. notifier.showUpdateNotifications(newUpdates)
  327. val newChapterCount = newUpdates.sumOf { it.second.size }
  328. preferences.unreadUpdatesCount().set(currentUnreadUpdatesCount + newChapterCount)
  329. if (hasDownloads.get()) {
  330. DownloadService.start(this)
  331. }
  332. }
  333. if (failedUpdates.isNotEmpty()) {
  334. val errorFile = writeErrorFile(failedUpdates)
  335. notifier.showUpdateErrorNotification(
  336. failedUpdates.map { it.first.title },
  337. errorFile.getUriCompat(this)
  338. )
  339. }
  340. }
  341. private fun downloadChapters(manga: Manga, chapters: List<Chapter>) {
  342. // We don't want to start downloading while the library is updating, because websites
  343. // may don't like it and they could ban the user.
  344. downloadManager.downloadChapters(manga, chapters, false)
  345. }
  346. /**
  347. * Updates the chapters for the given manga and adds them to the database.
  348. *
  349. * @param manga the manga to update.
  350. * @return a pair of the inserted and removed chapters.
  351. */
  352. suspend fun updateManga(manga: Manga): Pair<List<Chapter>, List<Chapter>> {
  353. val source = sourceManager.getOrStub(manga.source)
  354. // Update manga details metadata
  355. if (preferences.autoUpdateMetadata()) {
  356. val updatedManga = source.getMangaDetails(manga.toMangaInfo())
  357. val sManga = updatedManga.toSManga()
  358. // Avoid "losing" existing cover
  359. if (!sManga.thumbnail_url.isNullOrEmpty()) {
  360. manga.prepUpdateCover(coverCache, sManga, false)
  361. } else {
  362. sManga.thumbnail_url = manga.thumbnail_url
  363. }
  364. manga.copyFrom(sManga)
  365. db.insertManga(manga).executeAsBlocking()
  366. }
  367. val chapters = source.getChapterList(manga.toMangaInfo())
  368. .map { it.toSChapter() }
  369. return syncChaptersWithSource(db, chapters, manga, source)
  370. }
  371. private suspend fun updateCovers() {
  372. val semaphore = Semaphore(5)
  373. val progressCount = AtomicInteger(0)
  374. val currentlyUpdatingManga = CopyOnWriteArrayList<LibraryManga>()
  375. withIOContext {
  376. mangaToUpdate.groupBy { it.source }
  377. .values
  378. .map { mangaInSource ->
  379. async {
  380. semaphore.withPermit {
  381. mangaInSource.forEach { manga ->
  382. if (updateJob?.isActive != true) {
  383. return@async
  384. }
  385. withUpdateNotification(
  386. currentlyUpdatingManga,
  387. progressCount,
  388. manga,
  389. ) { manga ->
  390. sourceManager.get(manga.source)?.let { source ->
  391. try {
  392. val networkManga =
  393. source.getMangaDetails(manga.toMangaInfo())
  394. val sManga = networkManga.toSManga()
  395. manga.prepUpdateCover(coverCache, sManga, true)
  396. sManga.thumbnail_url?.let {
  397. manga.thumbnail_url = it
  398. db.insertManga(manga).executeAsBlocking()
  399. }
  400. } catch (e: Throwable) {
  401. // Ignore errors and continue
  402. logcat(LogPriority.ERROR, e)
  403. }
  404. }
  405. }
  406. }
  407. }
  408. }
  409. }
  410. .awaitAll()
  411. }
  412. coverCache.clearMemoryCache()
  413. notifier.cancelProgressNotification()
  414. }
  415. /**
  416. * Method that updates the metadata of the connected tracking services. It's called in a
  417. * background thread, so it's safe to do heavy operations or network calls here.
  418. */
  419. private suspend fun updateTrackings() {
  420. var progressCount = 0
  421. val loggedServices = trackManager.services.filter { it.isLogged }
  422. mangaToUpdate.forEach { manga ->
  423. if (updateJob?.isActive != true) {
  424. return
  425. }
  426. notifier.showProgressNotification(listOf(manga), progressCount++, mangaToUpdate.size)
  427. // Update the tracking details.
  428. updateTrackings(manga, loggedServices)
  429. }
  430. notifier.cancelProgressNotification()
  431. }
  432. private suspend fun updateTrackings(manga: LibraryManga, loggedServices: List<TrackService>) {
  433. db.getTracks(manga).executeAsBlocking()
  434. .map { track ->
  435. supervisorScope {
  436. async {
  437. val service = trackManager.getService(track.sync_id)
  438. if (service != null && service in loggedServices) {
  439. try {
  440. val updatedTrack = service.refresh(track)
  441. db.insertTrack(updatedTrack).executeAsBlocking()
  442. if (service is EnhancedTrackService) {
  443. syncChaptersWithTrackServiceTwoWay(db, db.getChapters(manga).executeAsBlocking(), track, service)
  444. }
  445. } catch (e: Throwable) {
  446. // Ignore errors and continue
  447. logcat(LogPriority.ERROR, e)
  448. }
  449. }
  450. }
  451. }
  452. }
  453. .awaitAll()
  454. }
  455. private suspend fun withUpdateNotification(
  456. updatingManga: CopyOnWriteArrayList<LibraryManga>,
  457. completed: AtomicInteger,
  458. manga: LibraryManga,
  459. block: suspend (LibraryManga) -> Unit,
  460. ) {
  461. if (updateJob?.isActive != true) {
  462. return
  463. }
  464. updatingManga.add(manga)
  465. notifier.showProgressNotification(
  466. updatingManga,
  467. completed.get(),
  468. mangaToUpdate.size
  469. )
  470. block(manga)
  471. if (updateJob?.isActive != true) {
  472. return
  473. }
  474. updatingManga.remove(manga)
  475. completed.andIncrement
  476. notifier.showProgressNotification(
  477. updatingManga,
  478. completed.get(),
  479. mangaToUpdate.size
  480. )
  481. }
  482. /**
  483. * Writes basic file of update errors to cache dir.
  484. */
  485. private fun writeErrorFile(errors: List<Pair<Manga, String?>>): File {
  486. try {
  487. if (errors.isNotEmpty()) {
  488. val file = createFileInCacheDir("tachiyomi_update_errors.txt")
  489. file.bufferedWriter().use { out ->
  490. // Error file format:
  491. // ! Error
  492. // # Source
  493. // - Manga
  494. errors.groupBy({ it.second }, { it.first }).forEach { (error, mangas) ->
  495. out.write("! ${error}\n")
  496. mangas.groupBy { it.source }.forEach { (srcId, mangas) ->
  497. val source = sourceManager.getOrStub(srcId)
  498. out.write(" # $source\n")
  499. mangas.forEach {
  500. out.write(" - ${it.title}\n")
  501. }
  502. }
  503. }
  504. }
  505. return file
  506. }
  507. } catch (e: Exception) {
  508. // Empty
  509. }
  510. return File("")
  511. }
  512. }
  513. private const val MANGA_PER_SOURCE_QUEUE_WARNING_THRESHOLD = 60