aboutsummaryrefslogtreecommitdiff
path: root/bpkg
diff options
context:
space:
mode:
authorKaren Arutyunov <karen@codesynthesis.com>2023-09-06 23:41:08 +0300
committerKaren Arutyunov <karen@codesynthesis.com>2023-09-25 11:35:25 +0300
commit1fe2a36968940021593bf980fee8da1255cd3b5e (patch)
tree9123d32cf2163031f0c7302f6599ce00ce23fe9c /bpkg
parent0369b5e6ed827b9416514bef54d4997c67a1953d (diff)
Fix configuration negotiation so that existing dependents with config clauses are always considered for negotiation
Diffstat (limited to 'bpkg')
-rw-r--r--bpkg/pkg-build-collect.cxx172
-rw-r--r--bpkg/pkg-build-collect.hxx72
2 files changed, 201 insertions, 43 deletions
diff --git a/bpkg/pkg-build-collect.cxx b/bpkg/pkg-build-collect.cxx
index b3b50aa..4db5e7d 100644
--- a/bpkg/pkg-build-collect.cxx
+++ b/bpkg/pkg-build-collect.cxx
@@ -1681,9 +1681,12 @@ namespace bpkg
// given that the deviated dependents are not very common, we just
// postpone their re-collection.
//
+ l5 ([&]{trace << "schedule re-collection of deviated "
+ << "existing dependent " << *ed.selected
+ << ed.db;});
+
recollect_existing_dependent (options,
ed,
- true /* reconfigure */,
replaced_vers,
postponed_recs,
postponed_cfgs);
@@ -3652,6 +3655,87 @@ namespace bpkg
throw merge_configuration {cfg.depth};
}
+ // Note that there can be some non-negotiated clusters which
+ // have been merged based on the shadow cluster into the
+ // resulting (being) negotiated cluster. If we had negotiated
+ // such non-negotiated clusters normally, we would query
+ // existing dependents for the dependencies they contain and
+ // consider them in the negotiation process by re-evaluating
+ // them (see collect_build_postponed() for details). But if we
+ // force-merge a non-negotiated cluster into the (being)
+ // negotiated cluster then the existing dependents of its
+ // dependencies won't participate in the negotiation, unless we
+ // take care of that now. We will recognize such dependencies as
+ // not yet (being) recursively collected and re-collect their
+ // existing dependents, if any.
+ //
+ vector<existing_dependent> depts;
+ string deps_trace;
+
+ for (const package_key& d: cfg.dependencies)
+ {
+ build_package* p (entered_build (d));
+
+ // Must be collected at least non-recursively.
+ //
+ assert (p != nullptr);
+
+ if (p->recursive_collection)
+ continue;
+
+ bool add_deps_trace (verb >= 5);
+
+ for (existing_dependent& ed:
+ query_existing_dependents (trace,
+ options,
+ d.db,
+ d.name,
+ fdb,
+ rpt_depts,
+ replaced_vers))
+ {
+ if (add_deps_trace)
+ {
+ deps_trace += p->available_name_version_db () + ' ';
+
+ // Make sure the dependency is only listed once in the
+ // trace record.
+ //
+ add_deps_trace = false;
+ }
+
+ // Add the existing dependent to the list, suppressing
+ // duplicates.
+ //
+ if (find_if (depts.begin (), depts.end (),
+ [&ed] (const existing_dependent& d)
+ {
+ return d.selected->name == ed.selected->name &&
+ d.db == ed.db;
+ }) == depts.end ())
+ {
+ depts.push_back (move (ed));
+ }
+ }
+ }
+
+ if (!depts.empty ())
+ {
+ l5 ([&]{trace << "cfg-postponing dependent "
+ << pkg.available_name_version_db ()
+ << " adds not (being) collected dependencies "
+ << deps_trace << "with not (being) collected "
+ << "existing dependents to (being) negotiated "
+ << "cluster and results in " << cfg
+ << ", throwing recollect_existing_dependents";});
+
+ // Don't print the "while satisfying..." chain.
+ //
+ dep_chain.clear ();
+
+ throw recollect_existing_dependents {cfg.depth, move (depts)};
+ }
+
// Up-negotiate the configuration and if it has changed, throw
// retry_configuration to the try/catch frame corresponding to
// the negotiation of the outermost merged cluster in order to
@@ -4809,12 +4893,6 @@ namespace bpkg
postponed_configurations postponed_cfgs_;
};
- struct recollect_dependent
- {
- size_t depth;
- existing_dependent dependent;
- };
-
size_t depth (pcfg != nullptr ? pcfg->depth : 0);
string t ("collect_build_postponed (" + to_string (depth) + ')');
@@ -4976,9 +5054,12 @@ namespace bpkg
}
else
{
+ l5 ([&]{trace << "schedule re-collection of deviated "
+ << "existing dependent " << *ed.selected
+ << ed.db;});
+
recollect_existing_dependent (o,
ed,
- true /* reconfigure */,
replaced_vers,
postponed_recs,
postponed_cfgs);
@@ -5049,9 +5130,10 @@ namespace bpkg
l5 ([&]{trace << "re-evaluation of existing dependent "
<< b->available_name_version_db () << " failed "
<< "due to merge configuration cycle for "
- << *pcfg << ", throwing recollect_dependent";});
+ << *pcfg << ", throwing "
+ << "recollect_existing_dependents";});
- throw recollect_dependent {e.depth, move (ed)};
+ throw recollect_existing_dependents {e.depth, {move (ed)}};
}
ed.reevaluated = true;
@@ -5470,6 +5552,35 @@ namespace bpkg
{
if (!p->recursive_collection)
{
+ package_key pk (p->db, p->name ());
+
+ auto pi (postponed_deps.find (pk));
+ if (pi != postponed_deps.end ())
+ {
+ l5 ([&]{trace << "skip re-collection of dep-postponed package "
+ << pk;});
+
+ // Note that here we would re-collect the package without
+ // specifying any configuration for it.
+ //
+ pi->second.wout_config = true;
+
+ continue;
+ }
+ else
+ {
+ const postponed_configuration* pcfg (
+ postponed_cfgs.find_dependency (pk));
+
+ if (pcfg != nullptr)
+ {
+ l5 ([&]{trace << "skip re-collection of dep-postponed package "
+ << pk << " since already in cluster " << *pcfg;});
+
+ continue;
+ }
+ }
+
build_package_refs dep_chain;
collect_build_prerequisites (o,
*p,
@@ -5750,7 +5861,7 @@ namespace bpkg
pc->set_shadow_cluster (move (shadow));
}
- catch (const recollect_dependent& e)
+ catch (const recollect_existing_dependents& e)
{
// If this is not "our problem", then keep looking.
//
@@ -5784,18 +5895,23 @@ namespace bpkg
//
pc->shadow_cluster.clear ();
- l5 ([&]{trace << "cfg-negotiation of " << *pc << " failed due "
- << "to merge configuration cycle while "
- << "re-evaluating existing dependent "
- << *e.dependent.selected << e.dependent.db
- << ", scheduling its re-collection";});
-
- recollect_existing_dependent (o,
- e.dependent,
- false /* reconfigure */,
- replaced_vers,
- postponed_recs,
- postponed_cfgs);
+ l5 ([&]{trace << "cfg-negotiation of " << *pc << " failed due to "
+ << "some existing dependents related problem, "
+ << "scheduling their re-collection";});
+
+ for (const existing_dependent& ed: e.dependents)
+ {
+ l5 ([&]{trace << "schedule re-collection of "
+ << (!ed.dependency ? "deviated " : "")
+ << "existing dependent " << *ed.selected
+ << ed.db;});
+
+ recollect_existing_dependent (o,
+ ed,
+ replaced_vers,
+ postponed_recs,
+ postponed_cfgs);
+ }
}
}
}
@@ -6732,8 +6848,9 @@ namespace bpkg
catch (const reeval_deviated&)
{
r.push_back (
- existing_dependent {
- ddb, move (dsp), nullopt, {}, package_key {db, name}, nullopt});
+ existing_dependent {ddb, move (dsp),
+ nullopt, {},
+ package_key {db, name}, nullopt});
}
}
}
@@ -6856,7 +6973,6 @@ namespace bpkg
void build_packages::
recollect_existing_dependent (const pkg_build_options& o,
const existing_dependent& ed,
- bool reconfigure,
replaced_versions& replaced_vers,
postponed_packages& postponed_recs,
postponed_configurations& postponed_cfgs)
@@ -6867,7 +6983,9 @@ namespace bpkg
uint16_t flags (build_package::build_recollect);
- if (reconfigure)
+ // Reconfigure the deviated dependents.
+ //
+ if (!ed.dependency)
flags |= build_package::adjust_reconfigure;
build_package p {
diff --git a/bpkg/pkg-build-collect.hxx b/bpkg/pkg-build-collect.hxx
index 5b4db84..1c3bc09 100644
--- a/bpkg/pkg-build-collect.hxx
+++ b/bpkg/pkg-build-collect.hxx
@@ -10,7 +10,6 @@
#include <forward_list>
#include <bpkg/types.hxx>
-#include <bpkg/forward.hxx> // database, linked_databases
#include <bpkg/utility.hxx>
#include <bpkg/package.hxx>
@@ -19,6 +18,7 @@
#include <bpkg/common-options.hxx>
#include <bpkg/pkg-build-options.hxx>
+#include <bpkg/database.hxx>
#include <bpkg/pkg-configure.hxx> // find_database_function()
#include <bpkg/package-skeleton.hxx>
#include <bpkg/system-package-manager.hxx>
@@ -441,7 +441,28 @@ namespace bpkg
// This, in particular, may end up in resolving different dependency
// packages and affect the dependent and dependency configurations.
//
- using postponed_packages = std::set<build_package*>;
+ // - Postponed recollection of configured dependents for resolving merge
+ // configuration cycles and as a fallback for missed re-evaluations due to
+ // the shadow-based configuration clusters merge (see
+ // collect_build_prerequisites() for details).
+ //
+ // For the sake of testing, make sure the order in the set is stable.
+ //
+ struct compare_build_package
+ {
+ bool
+ operator() (const build_package* x, const build_package* y) const
+ {
+ const package_name& nx (x->name ());
+ const package_name& ny (y->name ());
+
+ if (int d = nx.compare (ny))
+ return d < 0;
+
+ return x->db.get () < y->db.get ();
+ }
+ };
+ using postponed_packages = std::set<build_package*, compare_build_package>;
// Base for exception types that indicate an inability to collect a package
// build because it was collected prematurely (version needs to be replaced,
@@ -1173,20 +1194,20 @@ namespace bpkg
// is {0,0}).
//
// - For an existing dependent being re-collected due to the selected
- // dependency alternatives deviation, which may be caused by its
- // dependency up/downgrade (see postponed prerequisites collection for
- // details).
+ // dependency alternatives deviation, etc which may be caused by its
+ // dependency up/downgrade (see postponed_packages and
+ // build_package::build_recollect flag for details).
//
// Note that for these cases, as it was said above, we can potentially
// fail if the dependent is an orphan, but this is exactly what we need to
// do in that case, since we won't be able to re-collect its dependencies.
//
// Only a single true dependency alternative can be selected per function
- // call. Such an alternative can only be selected if its index in the
- // postponed alternatives list is less than the specified maximum (used by
- // the heuristics that determines in which order to process packages with
- // alternatives; if 0 is passed, then no true alternative will be
- // selected).
+ // call, unless we are (pre-)re-evaluating. Such an alternative can only
+ // be selected if its index in the postponed alternatives list is less
+ // than the specified maximum (used by the heuristics that determines in
+ // which order to process packages with alternatives; if 0 is passed, then
+ // no true alternative will be selected).
//
// The idea here is to postpone the true alternatives selection till the
// end of the packages collection and then try to optimize the overall
@@ -1203,8 +1224,11 @@ namespace bpkg
// exception. This exception is handled via re-collecting packages from
// scratch, but now with the knowledge about premature dependency
// collection. If some dependency already belongs to some non or being
- // negotiated cluster then throw merge_configuration. If some dependency
- // configuration has already been negotiated between some other
+ // negotiated cluster then throw merge_configuration. If some dependencies
+ // have existing dependents with config clauses which have not been
+ // considered for the configuration negotiation yet, then throw
+ // recollect_existing_dependents exception to re-collect these dependents.
+ // If configuration has already been negotiated between some other
// dependents, then up-negotiate the configuration and throw
// retry_configuration exception so that the configuration refinement can
// be performed. See the collect lambda implementation for details on the
@@ -1476,6 +1500,17 @@ namespace bpkg
optional<pair<size_t, size_t>> orig_dependency_position;
};
+ // This exception is thrown by collect_build_prerequisites() and
+ // collect_build_postponed() to resolve different kinds of existing
+ // dependent re-evaluation related cycles by re-collecting the problematic
+ // dependents from scratch.
+ //
+ struct recollect_existing_dependents
+ {
+ size_t depth;
+ vector<existing_dependent> dependents;
+ };
+
vector<existing_dependent>
query_existing_dependents (
tracer&,
@@ -1509,14 +1544,19 @@ namespace bpkg
replaced_versions&,
postponed_configurations&);
- // Non-recursively collect the deviated existing dependent previously
- // returned by the query_existing_dependents() function call and add it to
- // the postponed package recollections list.
+ // Non-recursively collect an existing dependent previously returned by
+ // the query_existing_dependents() function call with the
+ // build_package::build_recollect flag and add it to the postponed package
+ // recollections list. Also add the build_package::adjust_reconfigure flag
+ // for the deviated dependents (existing_dependent::dependency is absent).
+ //
+ // Note that after this function call the existing dependent may not be
+ // returned as a result by the query_existing_dependents() function
+ // anymore (due to the build_package::build_recollect flag presence).
//
void
recollect_existing_dependent (const pkg_build_options&,
const existing_dependent&,
- bool reconfigure,
replaced_versions&,
postponed_packages& postponed_recs,
postponed_configurations&);