refactor: use OkHttpClient for downloading videos

This commit is contained in:
Bnyro 2024-12-18 16:40:15 +01:00
parent ac8736c41d
commit a3a2de5452
2 changed files with 188 additions and 130 deletions

View File

@ -37,7 +37,6 @@ object DownloadHelper {
const val THUMBNAIL_DIR = "thumbnail" const val THUMBNAIL_DIR = "thumbnail"
const val DOWNLOAD_CHUNK_SIZE = 8L * 1024 const val DOWNLOAD_CHUNK_SIZE = 8L * 1024
const val DEFAULT_TIMEOUT = 15 * 1000 const val DEFAULT_TIMEOUT = 15 * 1000
const val DEFAULT_RETRY = 3
private const val VIDEO_MIMETYPE = "video/*" private const val VIDEO_MIMETYPE = "video/*"
fun getDownloadDir(context: Context, path: String): Path { fun getDownloadDir(context: Context, path: String): Path {

View File

@ -6,8 +6,10 @@ import android.app.PendingIntent.FLAG_UPDATE_CURRENT
import android.content.Intent import android.content.Intent
import android.os.Binder import android.os.Binder
import android.os.IBinder import android.os.IBinder
import android.util.Log
import android.util.SparseBooleanArray import android.util.SparseBooleanArray
import androidx.core.app.NotificationCompat import androidx.core.app.NotificationCompat
import androidx.core.app.NotificationCompat.Builder
import androidx.core.app.PendingIntentCompat import androidx.core.app.PendingIntentCompat
import androidx.core.app.ServiceCompat import androidx.core.app.ServiceCompat
import androidx.core.content.getSystemService import androidx.core.content.getSystemService
@ -19,8 +21,8 @@ import androidx.lifecycle.lifecycleScope
import com.github.libretube.LibreTubeApp.Companion.DOWNLOAD_CHANNEL_NAME import com.github.libretube.LibreTubeApp.Companion.DOWNLOAD_CHANNEL_NAME
import com.github.libretube.R import com.github.libretube.R
import com.github.libretube.api.CronetHelper import com.github.libretube.api.CronetHelper
import com.github.libretube.api.RetrofitInstance
import com.github.libretube.api.StreamsExtractor import com.github.libretube.api.StreamsExtractor
import com.github.libretube.api.obj.Streams
import com.github.libretube.constants.IntentData import com.github.libretube.constants.IntentData
import com.github.libretube.db.DatabaseHolder.Database import com.github.libretube.db.DatabaseHolder.Database
import com.github.libretube.db.obj.Download import com.github.libretube.db.obj.Download
@ -31,6 +33,7 @@ import com.github.libretube.enums.NotificationId
import com.github.libretube.extensions.formatAsFileSize import com.github.libretube.extensions.formatAsFileSize
import com.github.libretube.extensions.getContentLength import com.github.libretube.extensions.getContentLength
import com.github.libretube.extensions.parcelableExtra import com.github.libretube.extensions.parcelableExtra
import com.github.libretube.extensions.toastFromMainDispatcher
import com.github.libretube.extensions.toastFromMainThread import com.github.libretube.extensions.toastFromMainThread
import com.github.libretube.helpers.DownloadHelper import com.github.libretube.helpers.DownloadHelper
import com.github.libretube.helpers.DownloadHelper.getNotificationId import com.github.libretube.helpers.DownloadHelper.getNotificationId
@ -43,7 +46,7 @@ import com.github.libretube.receivers.NotificationReceiver.Companion.ACTION_DOWN
import com.github.libretube.receivers.NotificationReceiver.Companion.ACTION_DOWNLOAD_RESUME import com.github.libretube.receivers.NotificationReceiver.Companion.ACTION_DOWNLOAD_RESUME
import com.github.libretube.receivers.NotificationReceiver.Companion.ACTION_DOWNLOAD_STOP import com.github.libretube.receivers.NotificationReceiver.Companion.ACTION_DOWNLOAD_STOP
import com.github.libretube.ui.activities.MainActivity import com.github.libretube.ui.activities.MainActivity
import kotlinx.coroutines.CancellationException import com.google.net.cronet.okhttptransport.CronetInterceptor
import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.Job import kotlinx.coroutines.Job
import kotlinx.coroutines.SupervisorJob import kotlinx.coroutines.SupervisorJob
@ -55,16 +58,22 @@ import kotlinx.coroutines.launch
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
import kotlinx.datetime.TimeZone import kotlinx.datetime.TimeZone
import kotlinx.datetime.toLocalDateTime import kotlinx.datetime.toLocalDateTime
import okhttp3.OkHttpClient
import okhttp3.Request
import okhttp3.Response
import okhttp3.ResponseBody
import okio.buffer import okio.buffer
import okio.sink import okio.sink
import okio.source import okio.source
import java.io.File import java.io.File
import java.io.IOException
import java.net.HttpURLConnection import java.net.HttpURLConnection
import java.net.SocketTimeoutException
import java.net.URL import java.net.URL
import java.nio.file.Path import java.nio.file.Path
import java.nio.file.StandardOpenOption import java.nio.file.StandardOpenOption
import java.time.Duration
import java.util.concurrent.Executors import java.util.concurrent.Executors
import kotlin.coroutines.cancellation.CancellationException
import kotlin.io.path.createFile import kotlin.io.path.createFile
import kotlin.io.path.deleteIfExists import kotlin.io.path.deleteIfExists
import kotlin.io.path.div import kotlin.io.path.div
@ -80,12 +89,23 @@ class DownloadService : LifecycleService() {
private val coroutineContext = dispatcher + SupervisorJob() private val coroutineContext = dispatcher + SupervisorJob()
private lateinit var notificationManager: NotificationManager private lateinit var notificationManager: NotificationManager
private lateinit var summaryNotificationBuilder: NotificationCompat.Builder private lateinit var summaryNotificationBuilder: Builder
private val downloadQueue = SparseBooleanArray() private val downloadQueue = SparseBooleanArray()
private val _downloadFlow = MutableSharedFlow<Pair<Int, DownloadStatus>>() private val _downloadFlow = MutableSharedFlow<Pair<Int, DownloadStatus>>()
val downloadFlow: SharedFlow<Pair<Int, DownloadStatus>> = _downloadFlow val downloadFlow: SharedFlow<Pair<Int, DownloadStatus>> = _downloadFlow
private val httpClient: OkHttpClient by lazy {
val cronetInterceptor = CronetInterceptor.newBuilder(CronetHelper.cronetEngine).build()
OkHttpClient.Builder()
.connectTimeout(Duration.ofMillis(DownloadHelper.DEFAULT_TIMEOUT.toLong()))
.readTimeout(Duration.ofMillis(DownloadHelper.DEFAULT_TIMEOUT.toLong()))
.addInterceptor(cronetInterceptor)
.retryOnConnectionFailure(true)
.build()
}
override fun onCreate() { override fun onCreate() {
super.onCreate() super.onCreate()
IS_DOWNLOAD_RUNNING = true IS_DOWNLOAD_RUNNING = true
@ -108,11 +128,29 @@ class DownloadService : LifecycleService() {
val fileName = name.ifEmpty { videoId } val fileName = name.ifEmpty { videoId }
lifecycleScope.launch(coroutineContext) { lifecycleScope.launch(coroutineContext) {
try { val streams = try {
val streams = withContext(Dispatchers.IO) { withContext(Dispatchers.IO) {
StreamsExtractor.extractStreams(videoId) StreamsExtractor.extractStreams(videoId)
} }
} catch (e: Exception) {
toastFromMainDispatcher(
StreamsExtractor.getExtractorErrorMessageString(this@DownloadService, e)
)
return@launch
}
storeVideoMetadata(videoId, streams, fileName)
val downloadItems = streams.toDownloadItems(downloadData.copy(fileName = fileName))
for (downloadItem in downloadItems) {
start(downloadItem)
}
}
return START_NOT_STICKY
}
private suspend fun storeVideoMetadata(videoId: String, streams: Streams, fileName: String) {
val thumbnailTargetPath = getDownloadPath(DownloadHelper.THUMBNAIL_DIR, fileName) val thumbnailTargetPath = getDownloadPath(DownloadHelper.THUMBNAIL_DIR, fileName)
val download = Download( val download = Download(
@ -125,6 +163,7 @@ class DownloadService : LifecycleService() {
thumbnailTargetPath thumbnailTargetPath
) )
Database.downloadDao().insertDownload(download) Database.downloadDao().insertDownload(download)
for (chapter in streams.chapters) { for (chapter in streams.chapters) {
val downloadChapter = DownloadChapter( val downloadChapter = DownloadChapter(
videoId = videoId, videoId = videoId,
@ -134,22 +173,21 @@ class DownloadService : LifecycleService() {
) )
Database.downloadDao().insertDownloadChapter(downloadChapter) Database.downloadDao().insertDownloadChapter(downloadChapter)
} }
try {
ImageHelper.downloadImage( ImageHelper.downloadImage(
this@DownloadService, this@DownloadService,
streams.thumbnailUrl, streams.thumbnailUrl,
thumbnailTargetPath thumbnailTargetPath
) )
val downloadItems = streams.toDownloadItems(downloadData.copy(fileName = fileName))
downloadItems.forEach { start(it) }
} catch (e: Exception) { } catch (e: Exception) {
return@launch Log.e(
this@DownloadService::class.java.name,
"failed to download image ${streams.thumbnailUrl}"
)
} }
} }
return START_NOT_STICKY
}
/** /**
* Initiate download [Job] using [DownloadItem] by creating file according to [FileType] * Initiate download [Job] using [DownloadItem] by creating file according to [FileType]
* for the requested file. * for the requested file.
@ -189,67 +227,15 @@ class DownloadService : LifecycleService() {
while (totalRead < item.downloadSize) { while (totalRead < item.downloadSize) {
try { try {
val con = startConnection(item, url, totalRead, item.downloadSize) ?: return totalRead = progressDownload(item, url, totalRead, notificationBuilder)
@Suppress("NewApi") // The StandardOpenOption enum is desugared.
val sink = item.path.sink(StandardOpenOption.APPEND).buffer()
val sourceByte = con.inputStream.source()
var lastTime = System.currentTimeMillis() / 1000
var lastRead = 0L
try {
// Check if downloading is still active and read next bytes.
while (downloadQueue[item.id] && sourceByte
.read(sink.buffer, DownloadHelper.DOWNLOAD_CHUNK_SIZE)
.also { lastRead = it } != -1L
) {
sink.emit()
totalRead += lastRead
_downloadFlow.emit(
item.id to DownloadStatus.Progress(
lastRead,
totalRead,
item.downloadSize
)
)
if (item.downloadSize != -1L &&
System.currentTimeMillis() / 1000 > lastTime
) {
notificationBuilder
.setContentText(
totalRead.formatAsFileSize() + " / " +
item.downloadSize.formatAsFileSize()
)
.setProgress(
item.downloadSize.toInt(),
totalRead.toInt(),
false
)
notificationManager.notify(
item.getNotificationId(),
notificationBuilder.build()
)
lastTime = System.currentTimeMillis() / 1000
}
}
} catch (_: CancellationException) { } catch (_: CancellationException) {
break break
} catch (e: Exception) { } catch (e: Exception) {
toastFromMainThread("${getString(R.string.download)}: ${e.message}") toastFromMainThread("${getString(R.string.download)}: ${e.message}")
Log.e(this@DownloadService::class.java.name, e.stackTraceToString())
_downloadFlow.emit(item.id to DownloadStatus.Error(e.message.toString(), e)) _downloadFlow.emit(item.id to DownloadStatus.Error(e.message.toString(), e))
break break
} }
withContext(Dispatchers.IO) {
sink.flush()
sink.close()
sourceByte.close()
con.disconnect()
}
} catch (_: Exception) {
break
}
} }
val completed = totalRead >= item.downloadSize val completed = totalRead >= item.downloadSize
@ -270,56 +256,131 @@ class DownloadService : LifecycleService() {
stopServiceIfDone() stopServiceIfDone()
} }
private suspend fun progressDownload(
item: DownloadItem,
url: URL,
totalReadBefore: Long,
notificationBuilder: Builder
): Long {
val source =
startConnection(item, url, totalReadBefore, item.downloadSize) ?: return totalReadBefore
var totalRead = totalReadBefore
val sink = item.path.sink(StandardOpenOption.APPEND).buffer()
val sourceByte = source.byteStream().source()
var lastTime = System.currentTimeMillis() / 1000
var lastRead = 0L
// Check if downloading is still active and read next bytes.
while (downloadQueue[item.id] && sourceByte
.read(sink.buffer, DownloadHelper.DOWNLOAD_CHUNK_SIZE)
.also { lastRead = it } != -1L
) {
sink.emit()
totalRead += lastRead
_downloadFlow.emit(
item.id to DownloadStatus.Progress(
lastRead,
totalRead,
item.downloadSize
)
)
if (item.downloadSize != -1L &&
System.currentTimeMillis() / 1000 > lastTime
) {
updateNotification(notificationBuilder, item, totalRead.toInt())
lastTime = System.currentTimeMillis() / 1000
}
}
withContext(Dispatchers.IO) {
sink.flush()
sink.close()
sourceByte.close()
source.close()
}
return totalRead
}
private fun updateNotification(
notificationBuilder: Builder,
item: DownloadItem,
totalRead: Int
) {
notificationBuilder
.setContentText(
totalRead.formatAsFileSize() + " / " +
item.downloadSize.formatAsFileSize()
)
.setProgress(
item.downloadSize.toInt(),
totalRead,
false
)
notificationManager.notify(
item.getNotificationId(),
notificationBuilder.build()
)
}
private suspend fun startConnection( private suspend fun startConnection(
item: DownloadItem, item: DownloadItem,
url: URL, url: URL,
alreadyRead: Long, alreadyRead: Long,
readLimit: Long? readLimit: Long?
): HttpURLConnection? { ): ResponseBody? {
// Set start range where last downloading was held. val limit = readLimit?.let {
val con = CronetHelper.cronetEngine.openConnection(url) as HttpURLConnection
con.requestMethod = "GET"
val limit = if (readLimit == null) {
""
} else {
// generate a random byte distance to make it more difficult to fingerprint // generate a random byte distance to make it more difficult to fingerprint
val nextBytesToReadSize = (BYTES_PER_REQUEST_MIN..BYTES_PER_REQUEST_MAX).random() val nextBytesToReadSize = (BYTES_PER_REQUEST_MIN..BYTES_PER_REQUEST_MAX).random()
min(readLimit, alreadyRead + nextBytesToReadSize) min(readLimit, alreadyRead + nextBytesToReadSize)
} }?.toString().orEmpty()
con.setRequestProperty("Range", "bytes=$alreadyRead-$limit")
con.connectTimeout = DownloadHelper.DEFAULT_TIMEOUT
con.readTimeout = DownloadHelper.DEFAULT_TIMEOUT
withContext(Dispatchers.IO) { val request = Request.Builder()
.url(url)
.method("GET", null)
.header("Range", "bytes=$alreadyRead-$limit")
.build()
return withContext(Dispatchers.IO) {
// Retry connecting to server for n times. // Retry connecting to server for n times.
for (i in 1..DownloadHelper.DEFAULT_RETRY) {
try { try {
con.connect() val call = httpClient.newCall(request)
break val response = call.execute()
} catch (_: SocketTimeoutException) {
val message = getString(R.string.downloadfailed) + " " + i return@withContext handleResponse(item, response)
} catch (e: IOException) {
Log.e(this::javaClass.name, e.printStackTrace().toString())
val message = getString(R.string.downloadfailed)
_downloadFlow.emit(item.id to DownloadStatus.Error(message)) _downloadFlow.emit(item.id to DownloadStatus.Error(message))
toastFromMainThread(message) toastFromMainThread(message)
return@withContext null
} }
} }
} }
private suspend fun handleResponse(item: DownloadItem, response: Response): ResponseBody? {
// If link is expired try to regenerate using available info. // If link is expired try to regenerate using available info.
if (con.responseCode == 403) { if (response.code == 403) {
regenerateLink(item) regenerateLink(item)
con.disconnect() response.close()
downloadFile(item) downloadFile(item)
return null return null
} else if (con.responseCode !in 200..299) { } else if (response.code !in 200..299) {
val message = getString(R.string.downloadfailed) + ": " + con.responseMessage val message = getString(R.string.downloadfailed) + ": " + response.message
_downloadFlow.emit(item.id to DownloadStatus.Error(message)) _downloadFlow.emit(item.id to DownloadStatus.Error(message))
toastFromMainThread(message) toastFromMainThread(message)
con.disconnect() response.close()
pause(item.id) pause(item.id)
return null return null
} }
return con return response.body
} }
/** /**
@ -413,8 +474,7 @@ class DownloadService : LifecycleService() {
private fun notifyForeground() { private fun notifyForeground() {
notificationManager = getSystemService()!! notificationManager = getSystemService()!!
summaryNotificationBuilder = NotificationCompat summaryNotificationBuilder = Builder(this, DOWNLOAD_CHANNEL_NAME)
.Builder(this, DOWNLOAD_CHANNEL_NAME)
.setSmallIcon(R.drawable.ic_launcher_lockscreen) .setSmallIcon(R.drawable.ic_launcher_lockscreen)
.setContentTitle(getString(R.string.downloading)) .setContentTitle(getString(R.string.downloading))
.setForegroundServiceBehavior(NotificationCompat.FOREGROUND_SERVICE_IMMEDIATE) .setForegroundServiceBehavior(NotificationCompat.FOREGROUND_SERVICE_IMMEDIATE)
@ -426,14 +486,13 @@ class DownloadService : LifecycleService() {
startForeground(NotificationId.DOWNLOAD_IN_PROGRESS.id, summaryNotificationBuilder.build()) startForeground(NotificationId.DOWNLOAD_IN_PROGRESS.id, summaryNotificationBuilder.build())
} }
private fun getNotificationBuilder(item: DownloadItem): NotificationCompat.Builder { private fun getNotificationBuilder(item: DownloadItem): Builder {
val intent = Intent(this@DownloadService, MainActivity::class.java) val intent = Intent(this@DownloadService, MainActivity::class.java)
.putExtra("fragmentToOpen", "downloads") .putExtra("fragmentToOpen", "downloads")
val activityIntent = PendingIntentCompat val activityIntent = PendingIntentCompat
.getActivity(this@DownloadService, 0, intent, FLAG_CANCEL_CURRENT, false) .getActivity(this@DownloadService, 0, intent, FLAG_CANCEL_CURRENT, false)
return NotificationCompat return Builder(this, DOWNLOAD_CHANNEL_NAME)
.Builder(this, DOWNLOAD_CHANNEL_NAME)
.setContentTitle("[${item.type}] ${item.fileName}") .setContentTitle("[${item.type}] ${item.fileName}")
.setProgress(0, 0, true) .setProgress(0, 0, true)
.setOngoing(true) .setOngoing(true)
@ -444,7 +503,7 @@ class DownloadService : LifecycleService() {
} }
private fun setResumeNotification( private fun setResumeNotification(
notificationBuilder: NotificationCompat.Builder, notificationBuilder: Builder,
item: DownloadItem item: DownloadItem
) { ) {
notificationBuilder notificationBuilder
@ -459,7 +518,7 @@ class DownloadService : LifecycleService() {
} }
private fun setPauseNotification( private fun setPauseNotification(
notificationBuilder: NotificationCompat.Builder, notificationBuilder: Builder,
item: DownloadItem, item: DownloadItem,
isCompleted: Boolean = false isCompleted: Boolean = false
) { ) {