fix: reduce concurrency of playlist import api calls

This commit is contained in:
Bnyro 2024-02-11 13:03:02 +01:00
parent f87e22dd3f
commit 26025cf393

View File

@ -27,6 +27,7 @@ import kotlinx.coroutines.withContext
object PlaylistsHelper { object PlaylistsHelper {
private val pipedPlaylistRegex = private val pipedPlaylistRegex =
"[\\da-fA-F]{8}-[\\da-fA-F]{4}-[\\da-fA-F]{4}-[\\da-fA-F]{4}-[\\da-fA-F]{12}".toRegex() "[\\da-fA-F]{8}-[\\da-fA-F]{4}-[\\da-fA-F]{4}-[\\da-fA-F]{4}-[\\da-fA-F]{12}".toRegex()
private const val MAX_CONCURRENT_IMPORT_CALLS = 5
private val token get() = PreferenceHelper.getToken() private val token get() = PreferenceHelper.getToken()
val loggedIn: Boolean get() = token.isNotEmpty() val loggedIn: Boolean get() = token.isNotEmpty()
@ -59,6 +60,7 @@ object PlaylistsHelper {
"alphabetic" -> playlists.sortedBy { it.name?.lowercase() } "alphabetic" -> playlists.sortedBy { it.name?.lowercase() }
"alphabetic_reversed" -> playlists.sortedBy { it.name?.lowercase() } "alphabetic_reversed" -> playlists.sortedBy { it.name?.lowercase() }
.reversed() .reversed()
else -> playlists else -> playlists
} }
} }
@ -176,34 +178,25 @@ object PlaylistsHelper {
suspend fun importPlaylists(playlists: List<PipedImportPlaylist>) = suspend fun importPlaylists(playlists: List<PipedImportPlaylist>) =
withContext(Dispatchers.IO) { withContext(Dispatchers.IO) {
playlists.map { playlist -> for (playlist in playlists) {
val playlistId = createPlaylist(playlist.name!!) val playlistId = createPlaylist(playlist.name!!) ?: return@withContext
async {
playlistId ?: return@async
// if logged in, add the playlists by their ID via an api call // if logged in, add the playlists by their ID via an api call
if (loggedIn) { if (loggedIn) {
addToPlaylist( val streams = playlist.videos.map { StreamItem(url = it) }
playlistId, addToPlaylist(playlistId, *streams.toTypedArray())
*playlist.videos.map {
StreamItem(url = it)
}.toTypedArray()
)
} else { } else {
// if not logged in, all video information needs to become fetched manually // if not logged in, all video information needs to become fetched manually
// Only do so with 20 videos at once to prevent performance issues // Only do so with `MAX_CONCURRENT_IMPORT_CALLS` videos at once to prevent performance issues
val streams = playlist.videos.mapIndexed { index, id -> id to index } val streams = playlist.videos.chunked(MAX_CONCURRENT_IMPORT_CALLS).map { videos ->
.groupBy { it.second % 20 }.map { (_, videos) ->
videos.parallelMap { videos.parallelMap {
runCatching { runCatching { RetrofitInstance.api.getStreams(it) }
RetrofitInstance.api.getStreams(it.first) .getOrNull()
.toStreamItem(it.first) ?.toStreamItem(it)
} }.filterNotNull()
}.mapNotNull { it.getOrNull() } }.flatten()
} addToPlaylist(playlistId, *streams.toTypedArray())
addToPlaylist(playlistId, *streams.flatten().toTypedArray())
} }
} }
}.awaitAll()
} }
suspend fun exportPipedPlaylists(): List<PipedImportPlaylist> = withContext(Dispatchers.IO) { suspend fun exportPipedPlaylists(): List<PipedImportPlaylist> = withContext(Dispatchers.IO) {