From cd8b2adf2512d118cb26c45ce8b364838143709f Mon Sep 17 00:00:00 2001 From: Sebastian Schuberth Date: Fri, 15 Mar 2024 19:03:20 +0100 Subject: [PATCH] WIP --- gradle/libs.versions.toml | 1 + plugins/commands/downloader/build.gradle.kts | 2 + .../src/main/kotlin/DownloaderCommand.kt | 53 +++++++++++++++---- 3 files changed, 47 insertions(+), 9 deletions(-) diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index 501df3e3738c1..823af32ced4aa 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -150,6 +150,7 @@ maven-resolver-transport-http = { module = "org.apache.maven.resolver:maven-reso maven-resolver-transport-wagon = { module = "org.apache.maven.resolver:maven-resolver-transport-wagon", version.ref = "mavenResolver" } mockk = { module = "io.mockk:mockk", version.ref = "mockk" } mordant = { module = "com.github.ajalt.mordant:mordant", version.ref = "mordant" } +mordantCoroutines = { module = "com.github.ajalt.mordant:mordant-coroutines", version.ref = "mordant" } okhttp = { module = "com.squareup.okhttp3:okhttp", version.ref = "okhttp" } postgres = { module = "org.postgresql:postgresql", version.ref = "postgres" } postgresEmbedded = { module = "com.opentable.components:otj-pg-embedded", version.ref = "postgresEmbedded" } diff --git a/plugins/commands/downloader/build.gradle.kts b/plugins/commands/downloader/build.gradle.kts index a48ba1caa9809..3dfe75b15baa1 100644 --- a/plugins/commands/downloader/build.gradle.kts +++ b/plugins/commands/downloader/build.gradle.kts @@ -33,4 +33,6 @@ dependencies { implementation(libs.clikt) implementation(libs.kotlinx.coroutines) + implementation(libs.mordant) + implementation(libs.mordantCoroutines) } diff --git a/plugins/commands/downloader/src/main/kotlin/DownloaderCommand.kt b/plugins/commands/downloader/src/main/kotlin/DownloaderCommand.kt index 18107a505b929..67a24f1dba09d 100644 --- a/plugins/commands/downloader/src/main/kotlin/DownloaderCommand.kt +++ b/plugins/commands/downloader/src/main/kotlin/DownloaderCommand.kt @@ -20,6 +20,7 @@ package org.ossreviewtoolkit.plugins.commands.downloader import com.github.ajalt.clikt.core.ProgramResult +import com.github.ajalt.clikt.core.terminal import com.github.ajalt.clikt.parameters.groups.default import com.github.ajalt.clikt.parameters.groups.mutuallyExclusiveOptions import com.github.ajalt.clikt.parameters.groups.required @@ -34,14 +35,27 @@ import com.github.ajalt.clikt.parameters.options.split import com.github.ajalt.clikt.parameters.options.switch import com.github.ajalt.clikt.parameters.types.enum import com.github.ajalt.clikt.parameters.types.file +import com.github.ajalt.mordant.animation.coroutines.animateInCoroutine +import com.github.ajalt.mordant.animation.progress.MultiProgressBarAnimation +import com.github.ajalt.mordant.animation.progress.addTask +import com.github.ajalt.mordant.animation.progress.advance +import com.github.ajalt.mordant.rendering.TextAlign +import com.github.ajalt.mordant.widgets.progress.percentage +import com.github.ajalt.mordant.widgets.progress.progressBar +import com.github.ajalt.mordant.widgets.progress.progressBarContextLayout +import com.github.ajalt.mordant.widgets.progress.progressBarLayout +import com.github.ajalt.mordant.widgets.progress.text +import com.github.ajalt.mordant.widgets.progress.timeRemaining import java.io.File import kotlin.time.measureTime import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.ExperimentalCoroutinesApi import kotlinx.coroutines.async import kotlinx.coroutines.awaitAll +import kotlinx.coroutines.launch import kotlinx.coroutines.runBlocking import kotlinx.coroutines.withContext @@ -316,21 +330,42 @@ class DownloaderCommand : OrtCommand( } } + @OptIn(ExperimentalCoroutinesApi::class) private suspend fun downloadAllPackages( packageDownloadDirs: Map, - failureMessages: MutableList + failureMessages: MutableList, + maxParallelDownloads: Int = 8 ) { - withContext(Dispatchers.IO) { - packageDownloadDirs.entries.mapIndexed { index, (pkg, dir) -> - async { - val progress = "${index + 1} of ${packageDownloadDirs.size}" + val parallelDownloads = packageDownloadDirs.size.coerceAtMost(maxParallelDownloads) + + val overallLayout = progressBarLayout { + text(if (dryRun) "Verifying" else "Downloading", align = TextAlign.LEFT) + progressBar() + percentage() + timeRemaining() + } + + val taskLayout = progressBarContextLayout { + text(fps = animationFps, align = TextAlign.LEFT) { "> Package '${context.id.toCoordinates()}'..." } + percentage() + } + + val progress = MultiProgressBarAnimation(terminal).animateInCoroutine() + val overall = progress.addTask(overallLayout, total = packageDownloadDirs.size.toLong()) + val tasks = List(parallelDownloads) { progress.addTask(taskLayout, context = Package.EMPTY, total = 1) } - val verb = if (dryRun) "Verifying" else "Starting" - echo("$verb download for '${pkg.id.toCoordinates()}' ($progress).") + withContext(Dispatchers.IO.limitedParallelism(parallelDownloads)) { + launch { progress.execute() } - downloadPackage(pkg, dir, failureMessages).also { - if (!dryRun) echo("Finished download for ${pkg.id.toCoordinates()} ($progress).") + packageDownloadDirs.entries.mapIndexed { index, (pkg, dir) -> + async { + with(tasks[index % parallelDownloads]) { + update { context = pkg } + downloadPackage(pkg, dir, failureMessages) + advance() } + + overall.advance() } }.awaitAll() }