| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579 | package eu.kanade.tachiyomi.data.libraryimport android.app.Serviceimport android.content.Contextimport android.content.Intentimport android.os.IBinderimport android.os.PowerManagerimport android.widget.Toastimport androidx.core.content.ContextCompatimport eu.kanade.tachiyomi.Rimport eu.kanade.tachiyomi.data.cache.CoverCacheimport eu.kanade.tachiyomi.data.database.DatabaseHelperimport eu.kanade.tachiyomi.data.database.models.Categoryimport eu.kanade.tachiyomi.data.database.models.Chapterimport eu.kanade.tachiyomi.data.database.models.LibraryMangaimport eu.kanade.tachiyomi.data.database.models.Mangaimport eu.kanade.tachiyomi.data.database.models.toMangaInfoimport eu.kanade.tachiyomi.data.download.DownloadManagerimport eu.kanade.tachiyomi.data.download.DownloadServiceimport eu.kanade.tachiyomi.data.library.LibraryUpdateService.Companion.startimport eu.kanade.tachiyomi.data.notification.Notificationsimport eu.kanade.tachiyomi.data.preference.MANGA_FULLY_READimport eu.kanade.tachiyomi.data.preference.MANGA_ONGOINGimport eu.kanade.tachiyomi.data.preference.PreferencesHelperimport eu.kanade.tachiyomi.data.track.EnhancedTrackServiceimport eu.kanade.tachiyomi.data.track.TrackManagerimport eu.kanade.tachiyomi.data.track.TrackServiceimport eu.kanade.tachiyomi.source.SourceManagerimport eu.kanade.tachiyomi.source.UnmeteredSourceimport eu.kanade.tachiyomi.source.model.SMangaimport eu.kanade.tachiyomi.source.model.toSChapterimport eu.kanade.tachiyomi.source.model.toSMangaimport eu.kanade.tachiyomi.util.chapter.NoChaptersExceptionimport eu.kanade.tachiyomi.util.chapter.syncChaptersWithSourceimport eu.kanade.tachiyomi.util.chapter.syncChaptersWithTrackServiceTwoWayimport eu.kanade.tachiyomi.util.lang.withIOContextimport eu.kanade.tachiyomi.util.prepUpdateCoverimport eu.kanade.tachiyomi.util.shouldDownloadNewChaptersimport eu.kanade.tachiyomi.util.storage.getUriCompatimport eu.kanade.tachiyomi.util.system.acquireWakeLockimport eu.kanade.tachiyomi.util.system.createFileInCacheDirimport eu.kanade.tachiyomi.util.system.isServiceRunningimport eu.kanade.tachiyomi.util.system.logcatimport eu.kanade.tachiyomi.util.system.toastimport kotlinx.coroutines.CoroutineExceptionHandlerimport kotlinx.coroutines.CoroutineScopeimport kotlinx.coroutines.Dispatchersimport kotlinx.coroutines.Jobimport kotlinx.coroutines.SupervisorJobimport kotlinx.coroutines.asyncimport kotlinx.coroutines.awaitAllimport kotlinx.coroutines.cancelimport kotlinx.coroutines.launchimport kotlinx.coroutines.supervisorScopeimport kotlinx.coroutines.sync.Semaphoreimport kotlinx.coroutines.sync.withPermitimport logcat.LogPriorityimport uy.kohesive.injekt.Injektimport uy.kohesive.injekt.api.getimport java.io.Fileimport java.util.concurrent.CopyOnWriteArrayListimport java.util.concurrent.atomic.AtomicBooleanimport 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
 |