123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579 |
- package eu.kanade.tachiyomi.data.library
- import android.app.Service
- import android.content.Context
- import android.content.Intent
- import android.os.IBinder
- import android.os.PowerManager
- import android.widget.Toast
- import androidx.core.content.ContextCompat
- import eu.kanade.tachiyomi.R
- import eu.kanade.tachiyomi.data.cache.CoverCache
- import eu.kanade.tachiyomi.data.database.DatabaseHelper
- import eu.kanade.tachiyomi.data.database.models.Category
- import eu.kanade.tachiyomi.data.database.models.Chapter
- import eu.kanade.tachiyomi.data.database.models.LibraryManga
- import eu.kanade.tachiyomi.data.database.models.Manga
- import eu.kanade.tachiyomi.data.database.models.toMangaInfo
- import eu.kanade.tachiyomi.data.download.DownloadManager
- import eu.kanade.tachiyomi.data.download.DownloadService
- import eu.kanade.tachiyomi.data.library.LibraryUpdateService.Companion.start
- import eu.kanade.tachiyomi.data.notification.Notifications
- import eu.kanade.tachiyomi.data.preference.MANGA_FULLY_READ
- import eu.kanade.tachiyomi.data.preference.MANGA_ONGOING
- import eu.kanade.tachiyomi.data.preference.PreferencesHelper
- import eu.kanade.tachiyomi.data.track.EnhancedTrackService
- import eu.kanade.tachiyomi.data.track.TrackManager
- import eu.kanade.tachiyomi.data.track.TrackService
- import eu.kanade.tachiyomi.source.SourceManager
- import eu.kanade.tachiyomi.source.UnmeteredSource
- import eu.kanade.tachiyomi.source.model.SManga
- import eu.kanade.tachiyomi.source.model.toSChapter
- import eu.kanade.tachiyomi.source.model.toSManga
- import eu.kanade.tachiyomi.util.chapter.NoChaptersException
- import eu.kanade.tachiyomi.util.chapter.syncChaptersWithSource
- import eu.kanade.tachiyomi.util.chapter.syncChaptersWithTrackServiceTwoWay
- import eu.kanade.tachiyomi.util.lang.withIOContext
- import eu.kanade.tachiyomi.util.prepUpdateCover
- import eu.kanade.tachiyomi.util.shouldDownloadNewChapters
- import eu.kanade.tachiyomi.util.storage.getUriCompat
- import eu.kanade.tachiyomi.util.system.acquireWakeLock
- import eu.kanade.tachiyomi.util.system.createFileInCacheDir
- import eu.kanade.tachiyomi.util.system.isServiceRunning
- import eu.kanade.tachiyomi.util.system.logcat
- import eu.kanade.tachiyomi.util.system.toast
- import kotlinx.coroutines.CoroutineExceptionHandler
- import kotlinx.coroutines.CoroutineScope
- import kotlinx.coroutines.Dispatchers
- import kotlinx.coroutines.Job
- import kotlinx.coroutines.SupervisorJob
- import kotlinx.coroutines.async
- import kotlinx.coroutines.awaitAll
- import kotlinx.coroutines.cancel
- import kotlinx.coroutines.launch
- import kotlinx.coroutines.supervisorScope
- import kotlinx.coroutines.sync.Semaphore
- import kotlinx.coroutines.sync.withPermit
- import logcat.LogPriority
- import uy.kohesive.injekt.Injekt
- import uy.kohesive.injekt.api.get
- import java.io.File
- import java.util.concurrent.CopyOnWriteArrayList
- import java.util.concurrent.atomic.AtomicBoolean
- import java.util.concurrent.atomic.AtomicInteger
- /**
- * This class will take care of updating the chapters of the manga from the library. It can be
- * started calling the [start] method. If it's already running, it won't do anything.
- * While the library is updating, a [PowerManager.WakeLock] will be held until the update is
- * completed, preventing the device from going to sleep mode. A notification will display the
- * progress of the update, and if case of an unexpected error, this service will be silently
- * destroyed.
- */
- class LibraryUpdateService(
- val db: DatabaseHelper = Injekt.get(),
- val sourceManager: SourceManager = Injekt.get(),
- val preferences: PreferencesHelper = Injekt.get(),
- val downloadManager: DownloadManager = Injekt.get(),
- val trackManager: TrackManager = Injekt.get(),
- val coverCache: CoverCache = Injekt.get()
- ) : Service() {
- private lateinit var wakeLock: PowerManager.WakeLock
- private lateinit var notifier: LibraryUpdateNotifier
- private lateinit var ioScope: CoroutineScope
- private var mangaToUpdate: List<LibraryManga> = mutableListOf()
- private var updateJob: Job? = null
- /**
- * Defines what should be updated within a service execution.
- */
- enum class Target {
- CHAPTERS, // Manga chapters
- COVERS, // Manga covers
- TRACKING // Tracking metadata
- }
- companion object {
- private var instance: LibraryUpdateService? = null
- /**
- * Key for category to update.
- */
- const val KEY_CATEGORY = "category"
- /**
- * Key that defines what should be updated.
- */
- const val KEY_TARGET = "target"
- /**
- * Returns the status of the service.
- *
- * @param context the application context.
- * @return true if the service is running, false otherwise.
- */
- fun isRunning(context: Context): Boolean {
- return context.isServiceRunning(LibraryUpdateService::class.java)
- }
- /**
- * Starts the service. It will be started only if there isn't another instance already
- * running.
- *
- * @param context the application context.
- * @param category a specific category to update, or null for global update.
- * @param target defines what should be updated.
- * @return true if service newly started, false otherwise
- */
- fun start(context: Context, category: Category? = null, target: Target = Target.CHAPTERS): Boolean {
- return if (!isRunning(context)) {
- val intent = Intent(context, LibraryUpdateService::class.java).apply {
- putExtra(KEY_TARGET, target)
- category?.let { putExtra(KEY_CATEGORY, it.id) }
- }
- ContextCompat.startForegroundService(context, intent)
- true
- } else {
- instance?.addMangaToQueue(category?.id ?: -1, target)
- false
- }
- }
- /**
- * Stops the service.
- *
- * @param context the application context.
- */
- fun stop(context: Context) {
- context.stopService(Intent(context, LibraryUpdateService::class.java))
- }
- }
- /**
- * Method called when the service is created. It injects dagger dependencies and acquire
- * the wake lock.
- */
- override fun onCreate() {
- super.onCreate()
- ioScope = CoroutineScope(SupervisorJob() + Dispatchers.IO)
- notifier = LibraryUpdateNotifier(this)
- wakeLock = acquireWakeLock(javaClass.name)
- startForeground(Notifications.ID_LIBRARY_PROGRESS, notifier.progressNotificationBuilder.build())
- }
- /**
- * Method called when the service is destroyed. It destroys subscriptions and releases the wake
- * lock.
- */
- override fun onDestroy() {
- updateJob?.cancel()
- ioScope?.cancel()
- if (wakeLock.isHeld) {
- wakeLock.release()
- }
- if (instance == this) {
- instance = null
- }
- super.onDestroy()
- }
- /**
- * This method needs to be implemented, but it's not used/needed.
- */
- override fun onBind(intent: Intent): IBinder? {
- return null
- }
- /**
- * Method called when the service receives an intent.
- *
- * @param intent the start intent from.
- * @param flags the flags of the command.
- * @param startId the start id of this command.
- * @return the start value of the command.
- */
- override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
- if (intent == null) return START_NOT_STICKY
- val target = intent.getSerializableExtra(KEY_TARGET) as? Target
- ?: return START_NOT_STICKY
- instance = this
- // Unsubscribe from any previous subscription if needed
- updateJob?.cancel()
- // Update favorite manga
- val categoryId = intent.getIntExtra(KEY_CATEGORY, -1)
- addMangaToQueue(categoryId, target)
- // Destroy service when completed or in case of an error.
- val handler = CoroutineExceptionHandler { _, exception ->
- logcat(LogPriority.ERROR, exception)
- stopSelf(startId)
- }
- updateJob = ioScope.launch(handler) {
- when (target) {
- Target.CHAPTERS -> updateChapterList()
- Target.COVERS -> updateCovers()
- Target.TRACKING -> updateTrackings()
- }
- }
- updateJob?.invokeOnCompletion { stopSelf(startId) }
- return START_REDELIVER_INTENT
- }
- /**
- * Adds list of manga to be updated.
- *
- * @param category the ID of the category to update, or -1 if no category specified.
- * @param target the target to update.
- */
- fun addMangaToQueue(categoryId: Int, target: Target) {
- val libraryManga = db.getLibraryMangas().executeAsBlocking()
- var listToUpdate = if (categoryId != -1) {
- libraryManga.filter { it.category == categoryId }
- } else {
- val categoriesToUpdate = preferences.libraryUpdateCategories().get().map(String::toInt)
- val listToInclude = if (categoriesToUpdate.isNotEmpty()) {
- libraryManga.filter { it.category in categoriesToUpdate }
- } else {
- libraryManga
- }
- val categoriesToExclude = preferences.libraryUpdateCategoriesExclude().get().map(String::toInt)
- val listToExclude = if (categoriesToExclude.isNotEmpty()) {
- libraryManga.filter { it.category in categoriesToExclude }
- } else {
- emptyList()
- }
- listToInclude.minus(listToExclude)
- }
- if (target == Target.CHAPTERS) {
- val restrictions = preferences.libraryUpdateMangaRestriction().get()
- if (MANGA_ONGOING in restrictions) {
- listToUpdate = listToUpdate.filterNot { it.status == SManga.COMPLETED }
- }
- if (MANGA_FULLY_READ in restrictions) {
- listToUpdate = listToUpdate.filter { it.unread == 0 }
- }
- }
- mangaToUpdate = listToUpdate
- .distinctBy { it.id }
- .sortedBy { it.title }
- // Warn when excessively checking a single source
- val maxUpdatesFromSource = mangaToUpdate
- .groupBy { it.source }
- .filterKeys { sourceManager.get(it) !is UnmeteredSource }
- .maxOfOrNull { it.value.size } ?: 0
- if (maxUpdatesFromSource > MANGA_PER_SOURCE_QUEUE_WARNING_THRESHOLD) {
- toast(R.string.notification_size_warning, Toast.LENGTH_LONG)
- }
- }
- /**
- * Method that updates the given list of manga. It's called in a background thread, so it's safe
- * to do heavy operations or network calls here.
- * For each manga it calls [updateManga] and updates the notification showing the current
- * progress.
- *
- * @param mangaToUpdate the list to update
- * @return an observable delivering the progress of each update.
- */
- suspend fun updateChapterList() {
- val semaphore = Semaphore(5)
- val progressCount = AtomicInteger(0)
- val currentlyUpdatingManga = CopyOnWriteArrayList<LibraryManga>()
- val newUpdates = CopyOnWriteArrayList<Pair<LibraryManga, Array<Chapter>>>()
- val failedUpdates = CopyOnWriteArrayList<Pair<Manga, String?>>()
- val hasDownloads = AtomicBoolean(false)
- val loggedServices by lazy { trackManager.services.filter { it.isLogged } }
- val currentUnreadUpdatesCount = preferences.unreadUpdatesCount().get()
- withIOContext {
- mangaToUpdate.groupBy { it.source }
- .values
- .map { mangaInSource ->
- async {
- semaphore.withPermit {
- mangaInSource.forEach { manga ->
- if (updateJob?.isActive != true) {
- return@async
- }
- withUpdateNotification(
- currentlyUpdatingManga,
- progressCount,
- manga,
- ) { manga ->
- try {
- val (newChapters, _) = updateManga(manga)
- if (newChapters.isNotEmpty()) {
- if (manga.shouldDownloadNewChapters(db, preferences)) {
- downloadChapters(manga, newChapters)
- hasDownloads.set(true)
- }
- // Convert to the manga that contains new chapters
- newUpdates.add(
- manga to newChapters.sortedByDescending { ch -> ch.source_order }
- .toTypedArray()
- )
- }
- } catch (e: Throwable) {
- val errorMessage = when (e) {
- is NoChaptersException -> {
- getString(R.string.no_chapters_error)
- }
- is SourceManager.SourceNotInstalledException -> {
- // failedUpdates will already have the source, don't need to copy it into the message
- getString(R.string.loader_not_implemented_error)
- }
- else -> {
- e.message
- }
- }
- failedUpdates.add(manga to errorMessage)
- }
- if (preferences.autoUpdateTrackers()) {
- updateTrackings(manga, loggedServices)
- }
- }
- }
- }
- }
- }
- .awaitAll()
- }
- notifier.cancelProgressNotification()
- if (newUpdates.isNotEmpty()) {
- notifier.showUpdateNotifications(newUpdates)
- val newChapterCount = newUpdates.sumOf { it.second.size }
- preferences.unreadUpdatesCount().set(currentUnreadUpdatesCount + newChapterCount)
- if (hasDownloads.get()) {
- DownloadService.start(this)
- }
- }
- if (failedUpdates.isNotEmpty()) {
- val errorFile = writeErrorFile(failedUpdates)
- notifier.showUpdateErrorNotification(
- failedUpdates.map { it.first.title },
- errorFile.getUriCompat(this)
- )
- }
- }
- private fun downloadChapters(manga: Manga, chapters: List<Chapter>) {
- // We don't want to start downloading while the library is updating, because websites
- // may don't like it and they could ban the user.
- downloadManager.downloadChapters(manga, chapters, false)
- }
- /**
- * Updates the chapters for the given manga and adds them to the database.
- *
- * @param manga the manga to update.
- * @return a pair of the inserted and removed chapters.
- */
- suspend fun updateManga(manga: Manga): Pair<List<Chapter>, List<Chapter>> {
- val source = sourceManager.getOrStub(manga.source)
- // Update manga details metadata
- if (preferences.autoUpdateMetadata()) {
- val updatedManga = source.getMangaDetails(manga.toMangaInfo())
- val sManga = updatedManga.toSManga()
- // Avoid "losing" existing cover
- if (!sManga.thumbnail_url.isNullOrEmpty()) {
- manga.prepUpdateCover(coverCache, sManga, false)
- } else {
- sManga.thumbnail_url = manga.thumbnail_url
- }
- manga.copyFrom(sManga)
- db.insertManga(manga).executeAsBlocking()
- }
- val chapters = source.getChapterList(manga.toMangaInfo())
- .map { it.toSChapter() }
- return syncChaptersWithSource(db, chapters, manga, source)
- }
- private suspend fun updateCovers() {
- val semaphore = Semaphore(5)
- val progressCount = AtomicInteger(0)
- val currentlyUpdatingManga = CopyOnWriteArrayList<LibraryManga>()
- withIOContext {
- mangaToUpdate.groupBy { it.source }
- .values
- .map { mangaInSource ->
- async {
- semaphore.withPermit {
- mangaInSource.forEach { manga ->
- if (updateJob?.isActive != true) {
- return@async
- }
- withUpdateNotification(
- currentlyUpdatingManga,
- progressCount,
- manga,
- ) { manga ->
- sourceManager.get(manga.source)?.let { source ->
- try {
- val networkManga =
- source.getMangaDetails(manga.toMangaInfo())
- val sManga = networkManga.toSManga()
- manga.prepUpdateCover(coverCache, sManga, true)
- sManga.thumbnail_url?.let {
- manga.thumbnail_url = it
- db.insertManga(manga).executeAsBlocking()
- }
- } catch (e: Throwable) {
- // Ignore errors and continue
- logcat(LogPriority.ERROR, e)
- }
- }
- }
- }
- }
- }
- }
- .awaitAll()
- }
- coverCache.clearMemoryCache()
- notifier.cancelProgressNotification()
- }
- /**
- * Method that updates the metadata of the connected tracking services. It's called in a
- * background thread, so it's safe to do heavy operations or network calls here.
- */
- private suspend fun updateTrackings() {
- var progressCount = 0
- val loggedServices = trackManager.services.filter { it.isLogged }
- mangaToUpdate.forEach { manga ->
- if (updateJob?.isActive != true) {
- return
- }
- notifier.showProgressNotification(listOf(manga), progressCount++, mangaToUpdate.size)
- // Update the tracking details.
- updateTrackings(manga, loggedServices)
- }
- notifier.cancelProgressNotification()
- }
- private suspend fun updateTrackings(manga: LibraryManga, loggedServices: List<TrackService>) {
- db.getTracks(manga).executeAsBlocking()
- .map { track ->
- supervisorScope {
- async {
- val service = trackManager.getService(track.sync_id)
- if (service != null && service in loggedServices) {
- try {
- val updatedTrack = service.refresh(track)
- db.insertTrack(updatedTrack).executeAsBlocking()
- if (service is EnhancedTrackService) {
- syncChaptersWithTrackServiceTwoWay(db, db.getChapters(manga).executeAsBlocking(), track, service)
- }
- } catch (e: Throwable) {
- // Ignore errors and continue
- logcat(LogPriority.ERROR, e)
- }
- }
- }
- }
- }
- .awaitAll()
- }
- private suspend fun withUpdateNotification(
- updatingManga: CopyOnWriteArrayList<LibraryManga>,
- completed: AtomicInteger,
- manga: LibraryManga,
- block: suspend (LibraryManga) -> Unit,
- ) {
- if (updateJob?.isActive != true) {
- return
- }
- updatingManga.add(manga)
- notifier.showProgressNotification(
- updatingManga,
- completed.get(),
- mangaToUpdate.size
- )
- block(manga)
- if (updateJob?.isActive != true) {
- return
- }
- updatingManga.remove(manga)
- completed.andIncrement
- notifier.showProgressNotification(
- updatingManga,
- completed.get(),
- mangaToUpdate.size
- )
- }
- /**
- * Writes basic file of update errors to cache dir.
- */
- private fun writeErrorFile(errors: List<Pair<Manga, String?>>): File {
- try {
- if (errors.isNotEmpty()) {
- val file = createFileInCacheDir("tachiyomi_update_errors.txt")
- file.bufferedWriter().use { out ->
- // Error file format:
- // ! Error
- // # Source
- // - Manga
- errors.groupBy({ it.second }, { it.first }).forEach { (error, mangas) ->
- out.write("! ${error}\n")
- mangas.groupBy { it.source }.forEach { (srcId, mangas) ->
- val source = sourceManager.getOrStub(srcId)
- out.write(" # $source\n")
- mangas.forEach {
- out.write(" - ${it.title}\n")
- }
- }
- }
- }
- return file
- }
- } catch (e: Exception) {
- // Empty
- }
- return File("")
- }
- }
- private const val MANGA_PER_SOURCE_QUEUE_WARNING_THRESHOLD = 60
|