style: reformat code
This commit is contained in:
parent
d40743d5ff
commit
6b29a810ba
@ -133,142 +133,149 @@ class FeverRssService @Inject constructor(
|
|||||||
* 3. Fetch the Fever articles
|
* 3. Fetch the Fever articles
|
||||||
* 4. Synchronize read/unread and starred/un-starred items
|
* 4. Synchronize read/unread and starred/un-starred items
|
||||||
*/
|
*/
|
||||||
override suspend fun sync(coroutineWorker: CoroutineWorker): ListenableWorker.Result = supervisorScope {
|
override suspend fun sync(coroutineWorker: CoroutineWorker): ListenableWorker.Result =
|
||||||
coroutineWorker.setProgress(SyncWorker.setIsSyncing(true))
|
supervisorScope {
|
||||||
|
coroutineWorker.setProgress(SyncWorker.setIsSyncing(true))
|
||||||
|
|
||||||
try {
|
try {
|
||||||
val preTime = System.currentTimeMillis()
|
val preTime = System.currentTimeMillis()
|
||||||
val preDate = Date(preTime)
|
val preDate = Date(preTime)
|
||||||
val accountId = context.currentAccountId
|
val accountId = context.currentAccountId
|
||||||
val account = accountDao.queryById(accountId)!!
|
val account = accountDao.queryById(accountId)!!
|
||||||
val feverAPI = getFeverAPI()
|
val feverAPI = getFeverAPI()
|
||||||
|
|
||||||
// 1. Fetch the Fever groups
|
// 1. Fetch the Fever groups
|
||||||
val groups = feverAPI.getGroups().groups?.map {
|
val groups = feverAPI.getGroups().groups?.map {
|
||||||
Group(
|
Group(
|
||||||
id = accountId.spacerDollar(it.id!!),
|
|
||||||
name = it.title ?: context.getString(R.string.empty),
|
|
||||||
accountId = accountId,
|
|
||||||
)
|
|
||||||
} ?: emptyList()
|
|
||||||
groupDao.insertOrUpdate(groups)
|
|
||||||
|
|
||||||
// 2. Fetch the Fever feeds
|
|
||||||
val feedsBody = feverAPI.getFeeds()
|
|
||||||
val feedsGroupsMap = mutableMapOf<String, String>()
|
|
||||||
feedsBody.feeds_groups?.forEach { feedsGroups ->
|
|
||||||
feedsGroups.group_id?.toString()?.let { groupId ->
|
|
||||||
feedsGroups.feed_ids?.split(",")?.forEach { feedId ->
|
|
||||||
feedsGroupsMap[feedId] = groupId
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fetch the Fever favicons
|
|
||||||
val faviconsById = feverAPI.getFavicons().favicons?.associateBy { it.id } ?: emptyMap()
|
|
||||||
feedDao.insertOrUpdate(
|
|
||||||
feedsBody.feeds?.map {
|
|
||||||
Feed(
|
|
||||||
id = accountId.spacerDollar(it.id!!),
|
id = accountId.spacerDollar(it.id!!),
|
||||||
name = it.title.decodeHTML() ?: context.getString(R.string.empty),
|
name = it.title ?: context.getString(R.string.empty),
|
||||||
url = it.url!!,
|
|
||||||
groupId = accountId.spacerDollar(feedsGroupsMap[it.id.toString()]!!),
|
|
||||||
accountId = accountId,
|
accountId = accountId,
|
||||||
icon = faviconsById[it.favicon_id]?.data
|
|
||||||
)
|
)
|
||||||
} ?: emptyList()
|
} ?: emptyList()
|
||||||
)
|
groupDao.insertOrUpdate(groups)
|
||||||
|
|
||||||
// Handle empty icon for feeds
|
// 2. Fetch the Fever feeds
|
||||||
val noIconFeeds = feedDao.queryNoIcon(accountId)
|
val feedsBody = feverAPI.getFeeds()
|
||||||
noIconFeeds.forEach {
|
val feedsGroupsMap = mutableMapOf<String, String>()
|
||||||
it.icon = rssHelper.queryRssIconLink(it.url)
|
feedsBody.feeds_groups?.forEach { feedsGroups ->
|
||||||
}
|
feedsGroups.group_id?.toString()?.let { groupId ->
|
||||||
feedDao.update(*noIconFeeds.toTypedArray())
|
feedsGroups.feed_ids?.split(",")?.forEach { feedId ->
|
||||||
|
feedsGroupsMap[feedId] = groupId
|
||||||
// 3. Fetch the Fever articles (up to unlimited counts)
|
|
||||||
var sinceId = account.lastArticleId?.dollarLast() ?: ""
|
|
||||||
var itemsBody = feverAPI.getItemsSince(sinceId)
|
|
||||||
while (itemsBody.items?.isNotEmpty() == true) {
|
|
||||||
articleDao.insert(
|
|
||||||
*itemsBody.items?.map {
|
|
||||||
Article(
|
|
||||||
id = accountId.spacerDollar(it.id!!),
|
|
||||||
date = it.created_on_time
|
|
||||||
?.run { Date(this * 1000) }
|
|
||||||
?.takeIf { !it.isFuture(preDate) }
|
|
||||||
?: preDate,
|
|
||||||
title = it.title.decodeHTML() ?: context.getString(R.string.empty),
|
|
||||||
author = it.author,
|
|
||||||
rawDescription = it.html ?: "",
|
|
||||||
shortDescription = Readability.parseToText(it.html, it.url).take(110),
|
|
||||||
fullContent = it.html,
|
|
||||||
img = rssHelper.findImg(it.html ?: ""),
|
|
||||||
link = it.url ?: "",
|
|
||||||
feedId = accountId.spacerDollar(it.feed_id!!),
|
|
||||||
accountId = accountId,
|
|
||||||
isUnread = (it.is_read ?: 0) <= 0,
|
|
||||||
isStarred = (it.is_saved ?: 0) > 0,
|
|
||||||
updateAt = preDate,
|
|
||||||
).also {
|
|
||||||
sinceId = it.id.dollarLast()
|
|
||||||
}
|
}
|
||||||
}?.toTypedArray() ?: emptyArray()
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch the Fever favicons
|
||||||
|
val faviconsById =
|
||||||
|
feverAPI.getFavicons().favicons?.associateBy { it.id } ?: emptyMap()
|
||||||
|
feedDao.insertOrUpdate(
|
||||||
|
feedsBody.feeds?.map {
|
||||||
|
Feed(
|
||||||
|
id = accountId.spacerDollar(it.id!!),
|
||||||
|
name = it.title.decodeHTML() ?: context.getString(R.string.empty),
|
||||||
|
url = it.url!!,
|
||||||
|
groupId = accountId.spacerDollar(feedsGroupsMap[it.id.toString()]!!),
|
||||||
|
accountId = accountId,
|
||||||
|
icon = faviconsById[it.favicon_id]?.data
|
||||||
|
)
|
||||||
|
} ?: emptyList()
|
||||||
)
|
)
|
||||||
if (itemsBody.items?.size!! >= 50) {
|
|
||||||
itemsBody = feverAPI.getItemsSince(sinceId)
|
|
||||||
} else {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 4. Synchronize read/unread and starred/un-starred
|
// Handle empty icon for feeds
|
||||||
val unreadArticleIds = feverAPI.getUnreadItems().unread_item_ids?.split(",")
|
val noIconFeeds = feedDao.queryNoIcon(accountId)
|
||||||
val starredArticleIds = feverAPI.getSavedItems().saved_item_ids?.split(",")
|
noIconFeeds.forEach {
|
||||||
val articleMeta = articleDao.queryMetadataAll(accountId)
|
it.icon = rssHelper.queryRssIconLink(it.url)
|
||||||
for (meta: ArticleMeta in articleMeta) {
|
|
||||||
val articleId = meta.id.dollarLast()
|
|
||||||
val shouldBeUnread = unreadArticleIds?.contains(articleId)
|
|
||||||
val shouldBeStarred = starredArticleIds?.contains(articleId)
|
|
||||||
if (meta.isUnread != shouldBeUnread) {
|
|
||||||
articleDao.markAsReadByArticleId(accountId, meta.id, shouldBeUnread ?: true)
|
|
||||||
}
|
}
|
||||||
if (meta.isStarred != shouldBeStarred) {
|
feedDao.update(*noIconFeeds.toTypedArray())
|
||||||
articleDao.markAsStarredByArticleId(accountId, meta.id, shouldBeStarred ?: false)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove orphaned groups and feeds, after synchronizing the starred/un-starred
|
// 3. Fetch the Fever articles (up to unlimited counts)
|
||||||
val groupIds = groups.map { it.id }
|
var sinceId = account.lastArticleId?.dollarLast() ?: ""
|
||||||
groupDao.queryAll(accountId).forEach {
|
var itemsBody = feverAPI.getItemsSince(sinceId)
|
||||||
if (!groupIds.contains(it.id)) {
|
while (itemsBody.items?.isNotEmpty() == true) {
|
||||||
super.deleteGroup(it, true)
|
articleDao.insert(
|
||||||
|
*itemsBody.items?.map {
|
||||||
|
Article(
|
||||||
|
id = accountId.spacerDollar(it.id!!),
|
||||||
|
date = it.created_on_time
|
||||||
|
?.run { Date(this * 1000) }
|
||||||
|
?.takeIf { !it.isFuture(preDate) }
|
||||||
|
?: preDate,
|
||||||
|
title = it.title.decodeHTML() ?: context.getString(R.string.empty),
|
||||||
|
author = it.author,
|
||||||
|
rawDescription = it.html ?: "",
|
||||||
|
shortDescription = Readability.parseToText(it.html, it.url)
|
||||||
|
.take(110),
|
||||||
|
fullContent = it.html,
|
||||||
|
img = rssHelper.findImg(it.html ?: ""),
|
||||||
|
link = it.url ?: "",
|
||||||
|
feedId = accountId.spacerDollar(it.feed_id!!),
|
||||||
|
accountId = accountId,
|
||||||
|
isUnread = (it.is_read ?: 0) <= 0,
|
||||||
|
isStarred = (it.is_saved ?: 0) > 0,
|
||||||
|
updateAt = preDate,
|
||||||
|
).also {
|
||||||
|
sinceId = it.id.dollarLast()
|
||||||
|
}
|
||||||
|
}?.toTypedArray() ?: emptyArray()
|
||||||
|
)
|
||||||
|
if (itemsBody.items?.size!! >= 50) {
|
||||||
|
itemsBody = feverAPI.getItemsSince(sinceId)
|
||||||
|
} else {
|
||||||
|
break
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
feedDao.queryAll(accountId).forEach {
|
// 4. Synchronize read/unread and starred/un-starred
|
||||||
if (!feedsGroupsMap.contains(it.id.dollarLast())) {
|
val unreadArticleIds = feverAPI.getUnreadItems().unread_item_ids?.split(",")
|
||||||
super.deleteFeed(it, true)
|
val starredArticleIds = feverAPI.getSavedItems().saved_item_ids?.split(",")
|
||||||
|
val articleMeta = articleDao.queryMetadataAll(accountId)
|
||||||
|
for (meta: ArticleMeta in articleMeta) {
|
||||||
|
val articleId = meta.id.dollarLast()
|
||||||
|
val shouldBeUnread = unreadArticleIds?.contains(articleId)
|
||||||
|
val shouldBeStarred = starredArticleIds?.contains(articleId)
|
||||||
|
if (meta.isUnread != shouldBeUnread) {
|
||||||
|
articleDao.markAsReadByArticleId(accountId, meta.id, shouldBeUnread ?: true)
|
||||||
|
}
|
||||||
|
if (meta.isStarred != shouldBeStarred) {
|
||||||
|
articleDao.markAsStarredByArticleId(
|
||||||
|
accountId,
|
||||||
|
meta.id,
|
||||||
|
shouldBeStarred ?: false
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove orphaned groups and feeds, after synchronizing the starred/un-starred
|
||||||
|
val groupIds = groups.map { it.id }
|
||||||
|
groupDao.queryAll(accountId).forEach {
|
||||||
|
if (!groupIds.contains(it.id)) {
|
||||||
|
super.deleteGroup(it, true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
feedDao.queryAll(accountId).forEach {
|
||||||
|
if (!feedsGroupsMap.contains(it.id.dollarLast())) {
|
||||||
|
super.deleteFeed(it, true)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
Log.i("RLog", "onCompletion: ${System.currentTimeMillis() - preTime}")
|
Log.i("RLog", "onCompletion: ${System.currentTimeMillis() - preTime}")
|
||||||
accountDao.update(account.apply {
|
accountDao.update(account.apply {
|
||||||
updateAt = Date()
|
updateAt = Date()
|
||||||
if (sinceId.isNotEmpty()) {
|
if (sinceId.isNotEmpty()) {
|
||||||
lastArticleId = accountId.spacerDollar(sinceId)
|
lastArticleId = accountId.spacerDollar(sinceId)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
ListenableWorker.Result.success(SyncWorker.setIsSyncing(false))
|
||||||
|
} catch (e: Exception) {
|
||||||
|
Log.e("RLog", "On sync exception: ${e.message}", e)
|
||||||
|
withContext(mainDispatcher) {
|
||||||
|
context.showToast(e.message)
|
||||||
}
|
}
|
||||||
})
|
ListenableWorker.Result.failure(SyncWorker.setIsSyncing(false))
|
||||||
ListenableWorker.Result.success(SyncWorker.setIsSyncing(false))
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e("RLog", "On sync exception: ${e.message}", e)
|
|
||||||
withContext(mainDispatcher) {
|
|
||||||
context.showToast(e.message)
|
|
||||||
}
|
}
|
||||||
ListenableWorker.Result.failure(SyncWorker.setIsSyncing(false))
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
override suspend fun markAsRead(
|
override suspend fun markAsRead(
|
||||||
groupId: String?,
|
groupId: String?,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user