Convert refrshing manga metadata to coroutines + 5 sources concurrently
Sorry and good luck Carlos 😔
This commit is contained in:
parent
f3d4e87542
commit
2bbef55737
@ -31,6 +31,7 @@ import eu.kanade.tachiyomi.data.preference.PreferencesHelper
|
|||||||
import eu.kanade.tachiyomi.data.preference.getOrDefault
|
import eu.kanade.tachiyomi.data.preference.getOrDefault
|
||||||
import eu.kanade.tachiyomi.data.track.TrackManager
|
import eu.kanade.tachiyomi.data.track.TrackManager
|
||||||
import eu.kanade.tachiyomi.source.SourceManager
|
import eu.kanade.tachiyomi.source.SourceManager
|
||||||
|
import eu.kanade.tachiyomi.source.fetchMangaDetailsAsync
|
||||||
import eu.kanade.tachiyomi.source.model.SManga
|
import eu.kanade.tachiyomi.source.model.SManga
|
||||||
import eu.kanade.tachiyomi.source.online.HttpSource
|
import eu.kanade.tachiyomi.source.online.HttpSource
|
||||||
import eu.kanade.tachiyomi.ui.main.MainActivity
|
import eu.kanade.tachiyomi.ui.main.MainActivity
|
||||||
@ -50,9 +51,6 @@ import kotlinx.coroutines.launch
|
|||||||
import kotlinx.coroutines.sync.Semaphore
|
import kotlinx.coroutines.sync.Semaphore
|
||||||
import kotlinx.coroutines.sync.withPermit
|
import kotlinx.coroutines.sync.withPermit
|
||||||
import kotlinx.coroutines.withContext
|
import kotlinx.coroutines.withContext
|
||||||
import rx.Observable
|
|
||||||
import rx.Subscription
|
|
||||||
import rx.schedulers.Schedulers
|
|
||||||
import timber.log.Timber
|
import timber.log.Timber
|
||||||
import uy.kohesive.injekt.Injekt
|
import uy.kohesive.injekt.Injekt
|
||||||
import uy.kohesive.injekt.api.get
|
import uy.kohesive.injekt.api.get
|
||||||
@ -82,11 +80,6 @@ class LibraryUpdateService(
|
|||||||
*/
|
*/
|
||||||
private lateinit var wakeLock: PowerManager.WakeLock
|
private lateinit var wakeLock: PowerManager.WakeLock
|
||||||
|
|
||||||
/**
|
|
||||||
* Subscription where the update is done.
|
|
||||||
*/
|
|
||||||
private var subscription: Subscription? = null
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Pending intent of action that cancels the library update
|
* Pending intent of action that cancels the library update
|
||||||
*/
|
*/
|
||||||
@ -318,14 +311,12 @@ class LibraryUpdateService(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Method called when the service is destroyed. It destroys subscriptions and releases the wake
|
* Method called when the service is destroyed. It cancels jobs and releases the wake lock.
|
||||||
* lock.
|
|
||||||
*/
|
*/
|
||||||
override fun onDestroy() {
|
override fun onDestroy() {
|
||||||
job?.cancel()
|
job?.cancel()
|
||||||
if (instance == this)
|
if (instance == this)
|
||||||
instance = null
|
instance = null
|
||||||
subscription?.unsubscribe()
|
|
||||||
if (wakeLock.isHeld) {
|
if (wakeLock.isHeld) {
|
||||||
wakeLock.release()
|
wakeLock.release()
|
||||||
}
|
}
|
||||||
@ -350,30 +341,14 @@ class LibraryUpdateService(
|
|||||||
*/
|
*/
|
||||||
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
|
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
|
||||||
if (intent == null) return START_NOT_STICKY
|
if (intent == null) return START_NOT_STICKY
|
||||||
val target = intent.getSerializableExtra(KEY_TARGET) as? Target
|
val target = intent.getSerializableExtra(KEY_TARGET) as? Target ?: return START_NOT_STICKY
|
||||||
?: return START_NOT_STICKY
|
|
||||||
|
|
||||||
// Unsubscribe from any previous subscription if needed.
|
|
||||||
subscription?.unsubscribe()
|
|
||||||
instance = this
|
instance = this
|
||||||
|
|
||||||
val selectedScheme = preferences.libraryUpdatePrioritization().getOrDefault()
|
val selectedScheme = preferences.libraryUpdatePrioritization().getOrDefault()
|
||||||
val mangaList =
|
val mangaList = getMangaToUpdate(intent, target).sortedWith(rankingScheme[selectedScheme])
|
||||||
getMangaToUpdate(intent, target).sortedWith(rankingScheme[selectedScheme])
|
|
||||||
// Update favorite manga. Destroy service when completed or in case of an error.
|
// Update favorite manga. Destroy service when completed or in case of an error.
|
||||||
if (target == Target.DETAILS) {
|
launchTarget(target, mangaList, startId)
|
||||||
// Update either chapter list or manga details.
|
|
||||||
subscription = Observable.defer {
|
|
||||||
updateDetails(mangaList)
|
|
||||||
}.subscribeOn(Schedulers.io()).subscribe({}, {
|
|
||||||
Timber.e(it)
|
|
||||||
stopSelf(startId)
|
|
||||||
}, {
|
|
||||||
stopSelf(startId)
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
launchTarget(target, mangaList, startId)
|
|
||||||
}
|
|
||||||
return START_REDELIVER_INTENT
|
return START_REDELIVER_INTENT
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -382,14 +357,18 @@ class LibraryUpdateService(
|
|||||||
Timber.e(exception)
|
Timber.e(exception)
|
||||||
stopSelf(startId)
|
stopSelf(startId)
|
||||||
}
|
}
|
||||||
job = if (target == Target.CHAPTERS) {
|
job = GlobalScope.launch(handler) {
|
||||||
listener?.onUpdateManga(LibraryManga())
|
when (target) {
|
||||||
GlobalScope.launch(handler) {
|
Target.CHAPTERS -> {
|
||||||
updateChaptersJob(mangaToAdd)
|
listener?.onUpdateManga(LibraryManga())
|
||||||
}
|
updateChaptersJob(mangaToAdd)
|
||||||
} else {
|
}
|
||||||
GlobalScope.launch(handler) {
|
Target.DETAILS -> {
|
||||||
updateTrackings(mangaToAdd)
|
updateDetails(mangaToAdd)
|
||||||
|
}
|
||||||
|
else -> {
|
||||||
|
updateTrackings(mangaToAdd)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -417,18 +396,17 @@ class LibraryUpdateService(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun finishUpdates() {
|
private suspend fun finishUpdates() {
|
||||||
if (jobCount.get() != 0) return
|
if (jobCount.get() != 0) return
|
||||||
if (newUpdates.isNotEmpty()) {
|
if (newUpdates.isNotEmpty()) {
|
||||||
showResultNotification(newUpdates)
|
showResultNotification(newUpdates)
|
||||||
|
|
||||||
if (preferences.refreshCoversToo().getOrDefault() && job?.isCancelled == false) {
|
if (preferences.refreshCoversToo().getOrDefault() && job?.isCancelled == false) {
|
||||||
updateDetails(newUpdates.map { it.key }).observeOn(Schedulers.io()).doOnCompleted {
|
updateDetails(newUpdates.keys.toList())
|
||||||
cancelProgressNotification()
|
cancelProgressNotification()
|
||||||
if (downloadNew && hasDownloads) {
|
if (downloadNew && hasDownloads) {
|
||||||
DownloadService.start(this)
|
DownloadService.start(this)
|
||||||
}
|
}
|
||||||
}.subscribeOn(Schedulers.io()).subscribe {}
|
|
||||||
} else if (downloadNew && hasDownloads) {
|
} else if (downloadNew && hasDownloads) {
|
||||||
DownloadService.start(this)
|
DownloadService.start(this)
|
||||||
}
|
}
|
||||||
@ -517,51 +495,46 @@ class LibraryUpdateService(
|
|||||||
downloadManager.downloadChapters(manga, dbChapters, false)
|
downloadManager.downloadChapters(manga, dbChapters, 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.
|
|
||||||
*/
|
|
||||||
fun updateManga(manga: Manga): Observable<Pair<List<Chapter>, List<Chapter>>> {
|
|
||||||
val source = sourceManager.get(manga.source) as? HttpSource ?: return Observable.empty()
|
|
||||||
return source.fetchChapterList(manga)
|
|
||||||
.map { syncChaptersWithSource(db, it, manga, source) }
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Method that updates the details of the given list of manga. It's called in a background
|
* Method that updates the details of the given list of manga. It's called in a background
|
||||||
* thread, so it's safe to do heavy operations or network calls here.
|
* thread, so it's safe to do heavy operations or network calls here.
|
||||||
*
|
*
|
||||||
* @param mangaToUpdate the list to update
|
* @param mangaToUpdate the list to update
|
||||||
* @return an observable delivering the progress of each update.
|
|
||||||
*/
|
*/
|
||||||
fun updateDetails(mangaToUpdate: List<LibraryManga>): Observable<LibraryManga> {
|
suspend fun updateDetails(mangaToUpdate: List<LibraryManga>) = coroutineScope {
|
||||||
// Initialize the variables holding the progress of the updates.
|
// Initialize the variables holding the progress of the updates.
|
||||||
val count = AtomicInteger(0)
|
val count = AtomicInteger(0)
|
||||||
|
val asyncList = mangaToUpdate.groupBy { it.source }.values.map { list ->
|
||||||
|
async {
|
||||||
|
requestSemaphore.withPermit {
|
||||||
|
list.forEach { manga ->
|
||||||
|
if (job?.isCancelled == true) {
|
||||||
|
return@async
|
||||||
|
}
|
||||||
|
val source = sourceManager.get(manga.source) as? HttpSource ?: return@async
|
||||||
|
showProgressNotification(manga, count.andIncrement, mangaToUpdate.size)
|
||||||
|
|
||||||
// Emit each manga and update it sequentially.
|
val networkManga = try {
|
||||||
return Observable.from(mangaToUpdate)
|
source.fetchMangaDetailsAsync(manga)
|
||||||
// Notify manga that will update.
|
} catch (e: java.lang.Exception) {
|
||||||
.doOnNext { showProgressNotification(it, count.andIncrement, mangaToUpdate.size) }
|
Timber.e(e)
|
||||||
// Update the details of the manga.
|
null
|
||||||
.concatMap { manga ->
|
}
|
||||||
val source = sourceManager.get(manga.source) as? HttpSource
|
if (networkManga != null) {
|
||||||
?: return@concatMap Observable.empty<LibraryManga>()
|
val thumbnailUrl = manga.thumbnail_url
|
||||||
source.fetchMangaDetails(manga)
|
manga.copyFrom(networkManga)
|
||||||
.map { networkManga ->
|
manga.initialized = true
|
||||||
val thumbnailUrl = manga.thumbnail_url
|
db.insertManga(manga).executeAsBlocking()
|
||||||
manga.copyFrom(networkManga)
|
if (thumbnailUrl != networkManga.thumbnail_url && !manga.hasCustomCover()) {
|
||||||
db.insertManga(manga).executeAsBlocking()
|
MangaImpl.setLastCoverFetch(manga.id!!, Date().time)
|
||||||
if (thumbnailUrl != networkManga.thumbnail_url)
|
}
|
||||||
MangaImpl.setLastCoverFetch(manga.id!!, Date().time)
|
}
|
||||||
manga
|
|
||||||
}
|
}
|
||||||
.onErrorReturn { manga }
|
}
|
||||||
}
|
|
||||||
.doOnCompleted {
|
|
||||||
cancelProgressNotification()
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
asyncList.awaitAll()
|
||||||
|
cancelProgressNotification()
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -78,13 +78,11 @@ class LibraryUpdateServiceTest {
|
|||||||
|
|
||||||
`when`(source.fetchChapterList(manga)).thenReturn(Observable.just(sourceChapters))
|
`when`(source.fetchChapterList(manga)).thenReturn(Observable.just(sourceChapters))
|
||||||
|
|
||||||
service.updateManga(manga).subscribe()
|
|
||||||
|
|
||||||
assertThat(service.db.getChapters(manga).executeAsBlocking()).hasSize(2)
|
assertThat(service.db.getChapters(manga).executeAsBlocking()).hasSize(2)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun testContinuesUpdatingWhenAMangaFails() {
|
suspend fun testContinuesUpdatingWhenAMangaFails() {
|
||||||
var favManga = createManga("/manga1", "/manga2", "/manga3")
|
var favManga = createManga("/manga1", "/manga2", "/manga3")
|
||||||
service.db.insertMangas(favManga).executeAsBlocking()
|
service.db.insertMangas(favManga).executeAsBlocking()
|
||||||
favManga = service.db.getLibraryMangas().executeAsBlocking()
|
favManga = service.db.getLibraryMangas().executeAsBlocking()
|
||||||
@ -99,7 +97,7 @@ class LibraryUpdateServiceTest {
|
|||||||
|
|
||||||
val intent = Intent()
|
val intent = Intent()
|
||||||
val target = LibraryUpdateService.Target.CHAPTERS
|
val target = LibraryUpdateService.Target.CHAPTERS
|
||||||
service.updateDetails(favManga).subscribe()
|
service.updateDetails(favManga)
|
||||||
|
|
||||||
// There are 3 network attempts and 2 insertions (1 request failed)
|
// There are 3 network attempts and 2 insertions (1 request failed)
|
||||||
assertThat(service.db.getChapters(favManga[0]).executeAsBlocking()).hasSize(2)
|
assertThat(service.db.getChapters(favManga[0]).executeAsBlocking()).hasSize(2)
|
||||||
|
Loading…
Reference in New Issue
Block a user