Skip to content

Commit

Permalink
During package download setup first add all downloads then handle local
Browse files Browse the repository at this point in the history
This caused atleast two problems:
1. When the first package was local and its `end` callback was called
   immediately it updated and printed the MultiProgressBar which thought
   that its finished (because it contained only one finished bar).
   This lead to an extra new line being printed which broke the output,
   reported here: #1957
2. The numbering of bars was wrong, most noticeable with bigger
   transactions with some already downloaded packages. It could look
   like:
   ```
   Transaction Summary:
    Installing:       296 packages

   Total size of inbound packages is 414 MiB. Need to download 412 MiB.
   After this operation, 1 GiB extra will be used (install 1 GiB, remove 0 B).
   Is this ok [y/N]: y
   [1/1] libreoffice-1:24.8.4.2-2.fc41.x86_64                                                       100% |   0.0   B/s |   0.0   B |  00m00s
   >>> Already downloaded
   -----------------------------------------------------------------------------------------------------------------------------------------
   [1/2] libreoffice-base-1:24.8.4.2-2.fc41.x86_64                                                  100% |   0.0   B/s |   0.0   B |  00m00s
   >>> Already downloaded
   -----------------------------------------------------------------------------------------------------------------------------------------
   [1/4] libreoffice-draw-1:24.8.4.2-2.fc41.x86_64                                                  100% |   0.0   B/s |   0.0   B |  00m00s
   >>> Already downloaded
   -----------------------------------------------------------------------------------------------------------------------------------------
   [2/5] libreoffice-emailmerge-1:24.8.4.2-2.fc41.x86_64                                            100% |   0.0   B/s |   0.0   B |  00m00s
   >>> Already downloaded
   -----------------------------------------------------------------------------------------------------------------------------------------
   [3/6] libreoffice-impress-1:24.8.4.2-2.fc41.x86_64                                               100% |   0.0   B/s |   0.0   B |  00m00s
   >>> Already downloaded
   -----------------------------------------------------------------------------------------------------------------------------------------
   [4/7] libreoffice-math-1:24.8.4.2-2.fc41.x86_64                                                  100% |   0.0   B/s |   0.0   B |  00m00s
   >>> Already downloaded
   -----------------------------------------------------------------------------------------------------------------------------------------
   [  5/296] libreoffice-calc-1:24.8.4.2-2.fc41.x86_64                           0% [<=>               ] |   1.0   B/s |   0.0   B |  97d15h
   ```
  • Loading branch information
kontura committed Jan 8, 2025
1 parent 23107e1 commit 3ebf1c1
Showing 1 changed file with 34 additions and 25 deletions.
59 changes: 34 additions & 25 deletions libdnf5/repo/package_downloader.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -146,6 +146,7 @@ void PackageDownloader::download() try {

std::vector<std::unique_ptr<LrPackageTarget>> lr_targets;
lr_targets.reserve(p_impl->targets.size());
std::vector<PackageTarget *> local_targets;
for (auto & pkg_target : p_impl->targets) {
if (use_cache_only && !pkg_target.package.is_available_locally()) {
throw RepoCacheonlyError(
Expand All @@ -164,31 +165,11 @@ void PackageDownloader::download() try {
}

if (pkg_target.package.is_available_locally()) {
// Copy local packages to their destination directories
std::filesystem::path source = pkg_target.package.get_package_path();
std::filesystem::path destination = pkg_target.destination / source.filename();
std::error_code ec;
const bool same_file = std::filesystem::equivalent(source, destination, ec);
if (!same_file) {
std::filesystem::copy(source, destination, std::filesystem::copy_options::overwrite_existing, ec);
}
if (auto * download_callbacks = pkg_target.package.get_base()->get_download_callbacks()) {
std::string msg;
DownloadCallbacks::TransferStatus status;
if (ec) {
status = DownloadCallbacks::TransferStatus::ERROR;
msg = ec.message();
} else {
status = DownloadCallbacks::TransferStatus::ALREADYEXISTS;
if (same_file) {
msg = "Already downloaded";
} else {
msg = fmt::format("Copied from {}", source.string());
}
}
pkg_target.need_call_end_callback = false;
download_callbacks->end(pkg_target.user_cb_data, status, msg.c_str());
}
// First only gather all local package targets, we cannot handle them right away because
// the download callbacks are not fully initialized (they don't contain all required downloads).
// This could cause problems for example when the first added pkg_target is local and we called
// the `end` callback right here the callbacks might think that they are finished.
local_targets.push_back(&pkg_target);
continue;
}

Expand Down Expand Up @@ -216,6 +197,34 @@ void PackageDownloader::download() try {
lr_targets.emplace_back(lr_target);
}

for (auto * local_pkg_target : local_targets) {
// Copy local packages to their destination directories
std::filesystem::path source = local_pkg_target->package.get_package_path();
std::filesystem::path destination = local_pkg_target->destination / source.filename();
std::error_code ec;
const bool same_file = std::filesystem::equivalent(source, destination, ec);
if (!same_file) {
std::filesystem::copy(source, destination, std::filesystem::copy_options::overwrite_existing, ec);
}
if (auto * download_callbacks = local_pkg_target->package.get_base()->get_download_callbacks()) {
std::string msg;
DownloadCallbacks::TransferStatus status;
if (ec) {
status = DownloadCallbacks::TransferStatus::ERROR;
msg = ec.message();
} else {
status = DownloadCallbacks::TransferStatus::ALREADYEXISTS;
if (same_file) {
msg = "Already downloaded";
} else {
msg = fmt::format("Copied from {}", source.string());
}
}
local_pkg_target->need_call_end_callback = false;
download_callbacks->end(local_pkg_target->user_cb_data, status, msg.c_str());
}
}

// Adding items to the end of GSList is slow. We go from the back and add items to the beginning.
GSList * list{nullptr};
for (auto it = lr_targets.rbegin(); it != lr_targets.rend(); ++it) {
Expand Down

0 comments on commit 3ebf1c1

Please sign in to comment.