LibraryUpdateService.kt 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593
  1. package eu.kanade.tachiyomi.data.library
  2. import android.app.Notification
  3. import android.app.PendingIntent
  4. import android.app.Service
  5. import android.content.Context
  6. import android.content.Intent
  7. import android.graphics.Bitmap
  8. import android.graphics.BitmapFactory
  9. import android.os.Build
  10. import android.os.IBinder
  11. import android.os.PowerManager
  12. import androidx.core.app.NotificationCompat
  13. import androidx.core.app.NotificationCompat.GROUP_ALERT_SUMMARY
  14. import androidx.core.app.NotificationManagerCompat
  15. import com.bumptech.glide.Glide
  16. import eu.kanade.tachiyomi.R
  17. import eu.kanade.tachiyomi.data.database.DatabaseHelper
  18. import eu.kanade.tachiyomi.data.database.models.Category
  19. import eu.kanade.tachiyomi.data.database.models.Chapter
  20. import eu.kanade.tachiyomi.data.database.models.LibraryManga
  21. import eu.kanade.tachiyomi.data.database.models.Manga
  22. import eu.kanade.tachiyomi.data.download.DownloadManager
  23. import eu.kanade.tachiyomi.data.download.DownloadService
  24. import eu.kanade.tachiyomi.data.library.LibraryUpdateRanker.rankingScheme
  25. import eu.kanade.tachiyomi.data.library.LibraryUpdateService.Companion.start
  26. import eu.kanade.tachiyomi.data.notification.NotificationReceiver
  27. import eu.kanade.tachiyomi.data.notification.Notifications
  28. import eu.kanade.tachiyomi.data.preference.PreferencesHelper
  29. import eu.kanade.tachiyomi.data.preference.getOrDefault
  30. import eu.kanade.tachiyomi.data.track.TrackManager
  31. import eu.kanade.tachiyomi.source.SourceManager
  32. import eu.kanade.tachiyomi.source.model.SManga
  33. import eu.kanade.tachiyomi.source.online.HttpSource
  34. import eu.kanade.tachiyomi.ui.main.MainActivity
  35. import eu.kanade.tachiyomi.util.chapter.syncChaptersWithSource
  36. import eu.kanade.tachiyomi.util.lang.chop
  37. import eu.kanade.tachiyomi.util.system.isServiceRunning
  38. import eu.kanade.tachiyomi.util.system.notification
  39. import eu.kanade.tachiyomi.util.system.notificationBuilder
  40. import eu.kanade.tachiyomi.util.system.notificationManager
  41. import rx.Observable
  42. import rx.Subscription
  43. import rx.schedulers.Schedulers
  44. import timber.log.Timber
  45. import uy.kohesive.injekt.Injekt
  46. import uy.kohesive.injekt.api.get
  47. import java.text.DecimalFormat
  48. import java.text.DecimalFormatSymbols
  49. import java.util.ArrayList
  50. import java.util.concurrent.atomic.AtomicInteger
  51. /**
  52. * This class will take care of updating the chapters of the manga from the library. It can be
  53. * started calling the [start] method. If it's already running, it won't do anything.
  54. * While the library is updating, a [PowerManager.WakeLock] will be held until the update is
  55. * completed, preventing the device from going to sleep mode. A notification will display the
  56. * progress of the update, and if case of an unexpected error, this service will be silently
  57. * destroyed.
  58. */
  59. class LibraryUpdateService(
  60. val db: DatabaseHelper = Injekt.get(),
  61. val sourceManager: SourceManager = Injekt.get(),
  62. val preferences: PreferencesHelper = Injekt.get(),
  63. val downloadManager: DownloadManager = Injekt.get(),
  64. val trackManager: TrackManager = Injekt.get()
  65. ) : Service() {
  66. /**
  67. * Wake lock that will be held until the service is destroyed.
  68. */
  69. private lateinit var wakeLock: PowerManager.WakeLock
  70. /**
  71. * Subscription where the update is done.
  72. */
  73. private var subscription: Subscription? = null
  74. /**
  75. * Pending intent of action that cancels the library update
  76. */
  77. private val cancelIntent by lazy {
  78. NotificationReceiver.cancelLibraryUpdatePendingBroadcast(this)
  79. }
  80. /**
  81. * Bitmap of the app for notifications.
  82. */
  83. private val notificationBitmap by lazy {
  84. BitmapFactory.decodeResource(resources, R.mipmap.ic_launcher)
  85. }
  86. /**
  87. * Cached progress notification to avoid creating a lot.
  88. */
  89. private val progressNotificationBuilder by lazy {
  90. notificationBuilder(Notifications.CHANNEL_LIBRARY) {
  91. setContentTitle(getString(R.string.app_name))
  92. setSmallIcon(R.drawable.ic_refresh_white_24dp)
  93. setLargeIcon(notificationBitmap)
  94. setOngoing(true)
  95. setOnlyAlertOnce(true)
  96. addAction(R.drawable.ic_close_white_24dp, getString(android.R.string.cancel), cancelIntent)
  97. }
  98. }
  99. /**
  100. * Defines what should be updated within a service execution.
  101. */
  102. enum class Target {
  103. CHAPTERS, // Manga chapters
  104. DETAILS, // Manga metadata
  105. TRACKING // Tracking metadata
  106. }
  107. companion object {
  108. /**
  109. * Key for category to update.
  110. */
  111. const val KEY_CATEGORY = "category"
  112. /**
  113. * Key that defines what should be updated.
  114. */
  115. const val KEY_TARGET = "target"
  116. private const val NOTIF_MAX_CHAPTERS = 5
  117. private const val NOTIF_TITLE_MAX_LEN = 45
  118. private const val NOTIF_ICON_SIZE = 192
  119. /**
  120. * Returns the status of the service.
  121. *
  122. * @param context the application context.
  123. * @return true if the service is running, false otherwise.
  124. */
  125. fun isRunning(context: Context): Boolean {
  126. return context.isServiceRunning(LibraryUpdateService::class.java)
  127. }
  128. /**
  129. * Starts the service. It will be started only if there isn't another instance already
  130. * running.
  131. *
  132. * @param context the application context.
  133. * @param category a specific category to update, or null for global update.
  134. * @param target defines what should be updated.
  135. */
  136. fun start(context: Context, category: Category? = null, target: Target = Target.CHAPTERS) {
  137. if (!isRunning(context)) {
  138. val intent = Intent(context, LibraryUpdateService::class.java).apply {
  139. putExtra(KEY_TARGET, target)
  140. category?.let { putExtra(KEY_CATEGORY, it.id) }
  141. }
  142. if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O) {
  143. context.startService(intent)
  144. } else {
  145. context.startForegroundService(intent)
  146. }
  147. }
  148. }
  149. /**
  150. * Stops the service.
  151. *
  152. * @param context the application context.
  153. */
  154. fun stop(context: Context) {
  155. context.stopService(Intent(context, LibraryUpdateService::class.java))
  156. }
  157. }
  158. /**
  159. * Method called when the service is created. It injects dagger dependencies and acquire
  160. * the wake lock.
  161. */
  162. override fun onCreate() {
  163. super.onCreate()
  164. startForeground(Notifications.ID_LIBRARY_PROGRESS, progressNotificationBuilder.build())
  165. wakeLock = (getSystemService(Context.POWER_SERVICE) as PowerManager).newWakeLock(
  166. PowerManager.PARTIAL_WAKE_LOCK, "LibraryUpdateService:WakeLock")
  167. wakeLock.acquire()
  168. }
  169. /**
  170. * Method called when the service is destroyed. It destroys subscriptions and releases the wake
  171. * lock.
  172. */
  173. override fun onDestroy() {
  174. subscription?.unsubscribe()
  175. if (wakeLock.isHeld) {
  176. wakeLock.release()
  177. }
  178. super.onDestroy()
  179. }
  180. /**
  181. * This method needs to be implemented, but it's not used/needed.
  182. */
  183. override fun onBind(intent: Intent): IBinder? {
  184. return null
  185. }
  186. /**
  187. * Method called when the service receives an intent.
  188. *
  189. * @param intent the start intent from.
  190. * @param flags the flags of the command.
  191. * @param startId the start id of this command.
  192. * @return the start value of the command.
  193. */
  194. override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
  195. if (intent == null) return START_NOT_STICKY
  196. val target = intent.getSerializableExtra(KEY_TARGET) as? Target
  197. ?: return START_NOT_STICKY
  198. // Unsubscribe from any previous subscription if needed.
  199. subscription?.unsubscribe()
  200. // Update favorite manga. Destroy service when completed or in case of an error.
  201. subscription = Observable
  202. .defer {
  203. val selectedScheme = preferences.libraryUpdatePrioritization().getOrDefault()
  204. val mangaList = getMangaToUpdate(intent, target)
  205. .sortedWith(rankingScheme[selectedScheme])
  206. // Update either chapter list or manga details.
  207. when (target) {
  208. Target.CHAPTERS -> updateChapterList(mangaList)
  209. Target.DETAILS -> updateDetails(mangaList)
  210. Target.TRACKING -> updateTrackings(mangaList)
  211. }
  212. }
  213. .subscribeOn(Schedulers.io())
  214. .subscribe({
  215. }, {
  216. Timber.e(it)
  217. stopSelf(startId)
  218. }, {
  219. stopSelf(startId)
  220. })
  221. return Service.START_REDELIVER_INTENT
  222. }
  223. /**
  224. * Returns the list of manga to be updated.
  225. *
  226. * @param intent the update intent.
  227. * @param target the target to update.
  228. * @return a list of manga to update
  229. */
  230. fun getMangaToUpdate(intent: Intent, target: Target): List<LibraryManga> {
  231. val categoryId = intent.getIntExtra(KEY_CATEGORY, -1)
  232. var listToUpdate = if (categoryId != -1)
  233. db.getLibraryMangas().executeAsBlocking().filter { it.category == categoryId }
  234. else {
  235. val categoriesToUpdate = preferences.libraryUpdateCategories().getOrDefault().map(String::toInt)
  236. if (categoriesToUpdate.isNotEmpty())
  237. db.getLibraryMangas().executeAsBlocking()
  238. .filter { it.category in categoriesToUpdate }
  239. .distinctBy { it.id }
  240. else
  241. db.getLibraryMangas().executeAsBlocking().distinctBy { it.id }
  242. }
  243. if (target == Target.CHAPTERS && preferences.updateOnlyNonCompleted()) {
  244. listToUpdate = listToUpdate.filter { it.status != SManga.COMPLETED }
  245. }
  246. return listToUpdate
  247. }
  248. /**
  249. * Method that updates the given list of manga. It's called in a background thread, so it's safe
  250. * to do heavy operations or network calls here.
  251. * For each manga it calls [updateManga] and updates the notification showing the current
  252. * progress.
  253. *
  254. * @param mangaToUpdate the list to update
  255. * @return an observable delivering the progress of each update.
  256. */
  257. fun updateChapterList(mangaToUpdate: List<LibraryManga>): Observable<LibraryManga> {
  258. // Initialize the variables holding the progress of the updates.
  259. val count = AtomicInteger(0)
  260. // List containing new updates
  261. val newUpdates = ArrayList<Pair<LibraryManga, Array<Chapter>>>()
  262. // List containing failed updates
  263. val failedUpdates = ArrayList<Manga>()
  264. // List containing categories that get included in downloads.
  265. val categoriesToDownload = preferences.downloadNewCategories().getOrDefault().map(String::toInt)
  266. // Boolean to determine if user wants to automatically download new chapters.
  267. val downloadNew = preferences.downloadNew().getOrDefault()
  268. // Boolean to determine if DownloadManager has downloads
  269. var hasDownloads = false
  270. // Emit each manga and update it sequentially.
  271. return Observable.from(mangaToUpdate)
  272. // Notify manga that will update.
  273. .doOnNext { showProgressNotification(it, count.andIncrement, mangaToUpdate.size) }
  274. // Update the chapters of the manga.
  275. .concatMap { manga ->
  276. updateManga(manga)
  277. // If there's any error, return empty update and continue.
  278. .onErrorReturn {
  279. failedUpdates.add(manga)
  280. Pair(emptyList(), emptyList())
  281. }
  282. // Filter out mangas without new chapters (or failed).
  283. .filter { pair -> pair.first.isNotEmpty() }
  284. .doOnNext {
  285. if (downloadNew && (categoriesToDownload.isEmpty() ||
  286. manga.category in categoriesToDownload)) {
  287. downloadChapters(manga, it.first)
  288. hasDownloads = true
  289. }
  290. }
  291. // Convert to the manga that contains new chapters.
  292. .map {
  293. Pair(
  294. manga,
  295. (it.first.sortedByDescending { ch -> ch.source_order }.toTypedArray())
  296. )
  297. }
  298. }
  299. // Add manga with new chapters to the list.
  300. .doOnNext { manga ->
  301. // Add to the list
  302. newUpdates.add(manga)
  303. }
  304. // Notify result of the overall update.
  305. .doOnCompleted {
  306. if (newUpdates.isNotEmpty()) {
  307. showUpdateNotifications(newUpdates)
  308. if (downloadNew && hasDownloads) {
  309. DownloadService.start(this)
  310. }
  311. }
  312. if (failedUpdates.isNotEmpty()) {
  313. Timber.e("Failed updating: ${failedUpdates.map { it.title }}")
  314. }
  315. cancelProgressNotification()
  316. }
  317. .map { manga -> manga.first }
  318. }
  319. fun downloadChapters(manga: Manga, chapters: List<Chapter>) {
  320. // we need to get the chapters from the db so we have chapter ids
  321. val mangaChapters = db.getChapters(manga).executeAsBlocking()
  322. val dbChapters = chapters.map {
  323. mangaChapters.find { mangaChapter -> mangaChapter.url == it.url }!!
  324. }
  325. // We don't want to start downloading while the library is updating, because websites
  326. // may don't like it and they could ban the user.
  327. downloadManager.downloadChapters(manga, dbChapters, false)
  328. }
  329. /**
  330. * Updates the chapters for the given manga and adds them to the database.
  331. *
  332. * @param manga the manga to update.
  333. * @return a pair of the inserted and removed chapters.
  334. */
  335. fun updateManga(manga: Manga): Observable<Pair<List<Chapter>, List<Chapter>>> {
  336. val source = sourceManager.get(manga.source) as? HttpSource ?: return Observable.empty()
  337. return source.fetchChapterList(manga)
  338. .map { syncChaptersWithSource(db, it, manga, source) }
  339. }
  340. /**
  341. * Method that updates the details of the given list of manga. It's called in a background
  342. * thread, so it's safe to do heavy operations or network calls here.
  343. *
  344. * @param mangaToUpdate the list to update
  345. * @return an observable delivering the progress of each update.
  346. */
  347. fun updateDetails(mangaToUpdate: List<LibraryManga>): Observable<LibraryManga> {
  348. // Initialize the variables holding the progress of the updates.
  349. val count = AtomicInteger(0)
  350. // Emit each manga and update it sequentially.
  351. return Observable.from(mangaToUpdate)
  352. // Notify manga that will update.
  353. .doOnNext { showProgressNotification(it, count.andIncrement, mangaToUpdate.size) }
  354. // Update the details of the manga.
  355. .concatMap { manga ->
  356. val source = sourceManager.get(manga.source) as? HttpSource
  357. ?: return@concatMap Observable.empty<LibraryManga>()
  358. source.fetchMangaDetails(manga)
  359. .map { networkManga ->
  360. manga.copyFrom(networkManga)
  361. db.insertManga(manga).executeAsBlocking()
  362. manga
  363. }
  364. .onErrorReturn { manga }
  365. }
  366. .doOnCompleted {
  367. cancelProgressNotification()
  368. }
  369. }
  370. /**
  371. * Method that updates the metadata of the connected tracking services. It's called in a
  372. * background thread, so it's safe to do heavy operations or network calls here.
  373. */
  374. private fun updateTrackings(mangaToUpdate: List<LibraryManga>): Observable<LibraryManga> {
  375. // Initialize the variables holding the progress of the updates.
  376. var count = 0
  377. val loggedServices = trackManager.services.filter { it.isLogged }
  378. // Emit each manga and update it sequentially.
  379. return Observable.from(mangaToUpdate)
  380. // Notify manga that will update.
  381. .doOnNext { showProgressNotification(it, count++, mangaToUpdate.size) }
  382. // Update the tracking details.
  383. .concatMap { manga ->
  384. val tracks = db.getTracks(manga).executeAsBlocking()
  385. Observable.from(tracks)
  386. .concatMap { track ->
  387. val service = trackManager.getService(track.sync_id)
  388. if (service != null && service in loggedServices) {
  389. service.refresh(track)
  390. .doOnNext { db.insertTrack(it).executeAsBlocking() }
  391. .onErrorReturn { track }
  392. } else {
  393. Observable.empty()
  394. }
  395. }
  396. .map { manga }
  397. }
  398. .doOnCompleted {
  399. cancelProgressNotification()
  400. }
  401. }
  402. /**
  403. * Shows the notification containing the currently updating manga and the progress.
  404. *
  405. * @param manga the manga that's being updated.
  406. * @param current the current progress.
  407. * @param total the total progress.
  408. */
  409. private fun showProgressNotification(manga: Manga, current: Int, total: Int) {
  410. notificationManager.notify(Notifications.ID_LIBRARY_PROGRESS, progressNotificationBuilder
  411. .setContentTitle(manga.title)
  412. .setProgress(total, current, false)
  413. .build())
  414. }
  415. /**
  416. * Shows the notification containing the result of the update done by the service.
  417. *
  418. * @param updates a list of manga with new updates.
  419. */
  420. private fun showUpdateNotifications(updates: List<Pair<Manga, Array<Chapter>>>) {
  421. if (updates.isEmpty()) {
  422. return
  423. }
  424. NotificationManagerCompat.from(this).apply {
  425. // Parent group notification
  426. notify(Notifications.ID_NEW_CHAPTERS, notification(Notifications.CHANNEL_NEW_CHAPTERS) {
  427. setContentTitle(getString(R.string.notification_new_chapters))
  428. if (updates.size == 1) {
  429. setContentText(updates.first().first.title.chop(NOTIF_TITLE_MAX_LEN))
  430. } else {
  431. setContentText(resources.getQuantityString(R.plurals.notification_new_chapters_text, updates.size, updates.size))
  432. setStyle(NotificationCompat.BigTextStyle().bigText(updates.joinToString("\n") {
  433. it.first.title.chop(NOTIF_TITLE_MAX_LEN)
  434. }))
  435. }
  436. setSmallIcon(R.drawable.ic_tachi)
  437. setLargeIcon(notificationBitmap)
  438. setGroup(Notifications.GROUP_NEW_CHAPTERS)
  439. setGroupAlertBehavior(GROUP_ALERT_SUMMARY)
  440. setGroupSummary(true)
  441. priority = NotificationCompat.PRIORITY_HIGH
  442. setContentIntent(getNotificationIntent())
  443. setAutoCancel(true)
  444. })
  445. // Per-manga notification
  446. updates.forEach {
  447. val (manga, chapters) = it
  448. notify(manga.id.hashCode(), createNewChaptersNotification(manga, chapters))
  449. }
  450. }
  451. }
  452. private fun createNewChaptersNotification(manga: Manga, chapters: Array<Chapter>): Notification {
  453. return notification(Notifications.CHANNEL_NEW_CHAPTERS) {
  454. setContentTitle(manga.title)
  455. val description = getChaptersDescriptionString(chapters)
  456. setContentText(description)
  457. setStyle(NotificationCompat.BigTextStyle().bigText(description))
  458. setSmallIcon(R.drawable.ic_tachi)
  459. val icon = getMangaIcon(manga)
  460. if (icon != null) {
  461. setLargeIcon(icon)
  462. }
  463. setGroup(Notifications.GROUP_NEW_CHAPTERS)
  464. setGroupAlertBehavior(GROUP_ALERT_SUMMARY)
  465. priority = NotificationCompat.PRIORITY_HIGH
  466. // Open first chapter on tap
  467. setContentIntent(NotificationReceiver.openChapterPendingActivity(this@LibraryUpdateService, manga, chapters.first()))
  468. setAutoCancel(true)
  469. // Mark chapters as read action
  470. addAction(R.drawable.ic_glasses_black_24dp, getString(R.string.action_mark_as_read),
  471. NotificationReceiver.markAsReadPendingBroadcast(this@LibraryUpdateService,
  472. manga, chapters, Notifications.ID_NEW_CHAPTERS))
  473. // View chapters action
  474. addAction(R.drawable.ic_book_white_24dp, getString(R.string.action_view_chapters),
  475. NotificationReceiver.openChapterPendingActivity(this@LibraryUpdateService,
  476. manga, Notifications.ID_NEW_CHAPTERS))
  477. }
  478. }
  479. /**
  480. * Cancels the progress notification.
  481. */
  482. private fun cancelProgressNotification() {
  483. notificationManager.cancel(Notifications.ID_LIBRARY_PROGRESS)
  484. }
  485. private fun getMangaIcon(manga: Manga): Bitmap? {
  486. return try {
  487. Glide.with(this)
  488. .asBitmap()
  489. .load(manga)
  490. .dontTransform()
  491. .centerCrop()
  492. .circleCrop()
  493. .override(NOTIF_ICON_SIZE, NOTIF_ICON_SIZE)
  494. .submit()
  495. .get()
  496. } catch (e: Exception) {
  497. null
  498. }
  499. }
  500. private fun getChaptersDescriptionString(chapters: Array<Chapter>): String {
  501. val formatter = DecimalFormat("#.###", DecimalFormatSymbols()
  502. .apply { decimalSeparator = '.' })
  503. val chapterNumbers = chapters
  504. .sortedBy { it.chapter_number }
  505. .map { formatter.format(it.chapter_number) }
  506. .toSet()
  507. val shouldTruncate = chapterNumbers.size > NOTIF_MAX_CHAPTERS
  508. val chaptersDescription = if (shouldTruncate) {
  509. chapterNumbers.take(NOTIF_MAX_CHAPTERS - 1).joinToString(", ")
  510. } else {
  511. chapterNumbers.joinToString(", ")
  512. }
  513. var description = resources.getQuantityString(R.plurals.notification_chapters, chapters.size, chaptersDescription)
  514. if (shouldTruncate) {
  515. description += " ${resources.getString(R.string.notification_and_n_more, (chapterNumbers.size - (NOTIF_MAX_CHAPTERS - 1)))}"
  516. }
  517. return description
  518. }
  519. /**
  520. * Returns an intent to open the main activity.
  521. */
  522. private fun getNotificationIntent(): PendingIntent {
  523. val intent = Intent(this, MainActivity::class.java).apply {
  524. flags = Intent.FLAG_ACTIVITY_CLEAR_TOP or Intent.FLAG_ACTIVITY_SINGLE_TOP
  525. action = MainActivity.SHORTCUT_RECENTLY_UPDATED
  526. }
  527. return PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT)
  528. }
  529. }