From 7bc3956b8efbe79c610ab73509b6c89c435656e2 Mon Sep 17 00:00:00 2001 From: Karen Arutyunov Date: Thu, 14 Jul 2022 22:39:46 +0300 Subject: Select next dependency alternative on dependency configuration negotiation failure in pkg-build --- bpkg/package-configuration.cxx | 51 ++- bpkg/package-configuration.hxx | 21 +- bpkg/package-skeleton.cxx | 34 +- bpkg/package-skeleton.hxx | 6 +- bpkg/pkg-build.cxx | 434 ++++++++++++++++++--- .../dependency-alternatives/t11a/foo-0.2.0.tar.gz | Bin 0 -> 464 bytes .../dependency-alternatives/t11a/fox-0.1.0.tar.gz | Bin 0 -> 425 bytes .../dependency-alternatives/t11a/fox-0.2.0.tar.gz | Bin 0 -> 457 bytes tests/pkg-build.testscript | 378 +++++++++++++++++- 9 files changed, 837 insertions(+), 87 deletions(-) create mode 100644 tests/common/dependency-alternatives/t11a/foo-0.2.0.tar.gz create mode 100644 tests/common/dependency-alternatives/t11a/fox-0.1.0.tar.gz create mode 100644 tests/common/dependency-alternatives/t11a/fox-0.2.0.tar.gz diff --git a/bpkg/package-configuration.cxx b/bpkg/package-configuration.cxx index e8ea13a..483e3e0 100644 --- a/bpkg/package-configuration.cxx +++ b/bpkg/package-configuration.cxx @@ -98,12 +98,13 @@ namespace bpkg } } - bool + optional negotiate_configuration ( package_configurations& cfgs, package_skeleton& dept, pair pos, - const small_vector, 1>& depcs) + const small_vector, 1>& depcs, + bool has_alt) { assert (!dept.system); @@ -210,7 +211,7 @@ namespace bpkg { old_cfgs.push_back ( dependent_config_variable_value { - v.name, move (v.value), move (*v.dependent)}); + v.name, move (v.value), move (*v.dependent), v.has_alternative}); // Note that we will not reload it to default in case of require. // @@ -218,7 +219,8 @@ namespace bpkg } else old_cfgs.push_back ( - dependent_config_variable_value {v.name, v.value, *v.dependent}); + dependent_config_variable_value { + v.name, v.value, *v.dependent, v.has_alternative}); } } @@ -261,10 +263,14 @@ namespace bpkg // Step 2: execute the prefer/accept or requires clauses. // if (!(da.require - ? dept.evaluate_require (depc_cfgs, *da.require, pos) + ? dept.evaluate_require (depc_cfgs, *da.require, pos, has_alt) : dept.evaluate_prefer_accept (depc_cfgs, - *da.prefer, *da.accept, pos))) + *da.prefer, *da.accept, pos, + has_alt))) { + if (has_alt) + return nullopt; + diag_record dr (fail); dr << "unable to negotiate acceptable configuration with dependent " @@ -377,7 +383,8 @@ namespace bpkg if (v.origin == variable_origin::buildfile) { new_cfgs.push_back ( - dependent_config_variable_value {v.name, v.value, *v.dependent}); + dependent_config_variable_value { + v.name, v.value, *v.dependent, v.has_alternative}); } } } @@ -441,16 +448,19 @@ namespace bpkg return true; } - diag_record dr (fail); - - dr << "unable to negotiate acceptable configuration between dependents " - << dept.package; + if (has_alt) + return nullopt; // Analyze the O->N changes and determine the problematic dependent(s). // Do we actually know for sure they are all problematic? Well, they // repeatedly changed the values to the ones we don't like, so I guess so. // - small_vector, 1> depts; // Duplicates. + // If it's the other dependent that has an alternative, then we let the + // negotiation continue for one more half-cycle at which point it will be + // while negotiating the configuration of the other dependent that we will + // (again) detect this cycle. + // + small_vector, 1> depts; for (const dependent_config_variable_value& nv: new_cfgs) { if (nv.dependent == dept.package) @@ -465,7 +475,14 @@ namespace bpkg return ov->dependent == pk.get (); }) == depts.end ()) { - dr << ", " << ov->dependent; + if (ov->has_alternative) + { + change_history.push_back (move (old_cfgs)); + change_history.push_back (move (new_cfgs)); + + return true; + } + depts.push_back (ov->dependent); } } @@ -473,6 +490,14 @@ namespace bpkg } } + diag_record dr (fail); + + dr << "unable to negotiate acceptable configuration between dependents " + << dept.package; + + for (const package_key& d: depts) + dr << ", " << d; + dr << " for dependencies "; for (size_t i (0); i != depcs.size (); ++i) diff --git a/bpkg/package-configuration.hxx b/bpkg/package-configuration.hxx index 73f05ff..30cbe0a 100644 --- a/bpkg/package-configuration.hxx +++ b/bpkg/package-configuration.hxx @@ -56,6 +56,18 @@ namespace bpkg // bool confirmed; + // If origin is buildfile and the originating dependent has been + // encountered during the negotiation, then this flag indicates whether + // this dependent has another dependency alternative. + // + // @@ Strictly speaking this is a property of the dependent and + // duplicating it here for each variable is quite dirty (and requires + // us to drag this through skeleton calls). Doing this properly, + // however, will likely require another map with the dependent as a + // key. Maybe one day. + // + bool has_alternative; + public: void undefine () @@ -64,6 +76,7 @@ namespace bpkg value = nullopt; dependent = nullopt; confirmed = false; + has_alternative = false; } string @@ -84,6 +97,7 @@ namespace bpkg string name; optional value; package_key dependent; + bool has_alternative; public: string @@ -194,13 +208,16 @@ namespace bpkg // Negotiate the configuration for the specified dependencies of the // specified dependent. Return true if the configuration has changed. + // Return absent if has_alternative is true and no acceptable configuration + // could be negotiated. // - bool + optional negotiate_configuration ( package_configurations&, package_skeleton& dependent, pair position, - const small_vector, 1>& dependencies); + const small_vector, 1>& dependencies, + bool has_alternative); } #endif // BPKG_PACKAGE_CONFIGURATION_HXX diff --git a/bpkg/package-skeleton.cxx b/bpkg/package-skeleton.cxx index 55db919..fee4bbc 100644 --- a/bpkg/package-skeleton.cxx +++ b/bpkg/package-skeleton.cxx @@ -530,7 +530,8 @@ namespace bpkg case variable_origin::override_: case variable_origin::undefined: { - config_variable_value v {var.name, ol.first, {}, {}, {}, false}; + config_variable_value v { + var.name, ol.first, {}, {}, {}, false, false}; // Override could mean user override from config_vars_ or the // dependent override that we have merged above. @@ -544,6 +545,7 @@ namespace bpkg v.origin = variable_origin::buildfile; v.dependent = move (ov->dependent); v.confirmed = ov->confirmed; + v.has_alternative = ov->has_alternative; } else assert (ov->origin == variable_origin::override_); @@ -659,9 +661,13 @@ namespace bpkg if (config_variable_value* v = cfg.find (var.name)) v->value = move (val); else - cfg.push_back ( - config_variable_value { - var.name, variable_origin::override_, {}, move (val), {}, false}); + cfg.push_back (config_variable_value {var.name, + variable_origin::override_, + {}, + move (val), + {}, + false, + false}); } ctx_ = nullptr; // Free. @@ -1181,7 +1187,8 @@ namespace bpkg evaluate_prefer_accept (const dependency_configurations& cfgs, const string& prefer, const string& accept, - pair indexes) + pair indexes, + bool has_alt) { size_t depends_index (indexes.first); @@ -1420,6 +1427,7 @@ namespace bpkg v.value = move (ns); v.dependent = package; // We are the originating dependent. v.confirmed = true; + v.has_alternative = has_alt; break; } case variable_origin::default_: @@ -1484,7 +1492,9 @@ namespace bpkg bool package_skeleton:: evaluate_require (const dependency_configurations& cfgs, - const string& require, pair indexes) + const string& require, + pair indexes, + bool has_alt) { size_t depends_index (indexes.first); @@ -1732,9 +1742,14 @@ namespace bpkg if (v == nullptr) // cfg.system { - cfg.push_back ( - config_variable_value { - var.name, variable_origin::undefined, {}, {}, {}, false}); + cfg.push_back (config_variable_value {var.name, + variable_origin::undefined, + {}, + {}, + {}, + false, + false}); + v = &cfg.back (); } @@ -1780,6 +1795,7 @@ namespace bpkg v->value = move (ns); v->dependent = package; // We are the originating dependent. v->confirmed = true; + v->has_alternative = has_alt; } } } diff --git a/bpkg/package-skeleton.hxx b/bpkg/package-skeleton.hxx index 7224d1d..9f76221 100644 --- a/bpkg/package-skeleton.hxx +++ b/bpkg/package-skeleton.hxx @@ -141,11 +141,13 @@ namespace bpkg bool evaluate_prefer_accept (const dependency_configurations&, - const string&, const string&, pair); + const string&, const string&, pair, + bool has_alternative); bool evaluate_require (const dependency_configurations&, - const string&, pair); + const string&, pair, + bool has_alternative); // Reset the skeleton to the start of the call sequence. // diff --git a/bpkg/pkg-build.cxx b/bpkg/pkg-build.cxx index f396431..2b1c4f3 100644 --- a/bpkg/pkg-build.cxx +++ b/bpkg/pkg-build.cxx @@ -1336,10 +1336,20 @@ namespace bpkg class dependency: public packages { public: - pair position; // depends + alternative + pair position; // depends + alternative (1-based) - dependency (const pair& pos, packages deps) - : packages (move (deps)), position (pos) {} + // If true, then another dependency alternative is present and that can + // potentially be considered instead of this one (see + // unacceptable_alternatives for details). + // + // Initially nullopt for existing dependents until they are re-evaluated. + // + optional has_alternative; + + dependency (const pair& pos, + packages deps, + optional ha) + : packages (move (deps)), position (pos), has_alternative (ha) {} }; class dependent_info @@ -1377,6 +1387,17 @@ namespace bpkg if (find (d->begin (), d->end (), p) == d->end ()) d->push_back (move (p)); } + + // Set the has_alternative flag for an existing dependent. Note that + // it shouldn't change if already set. + // + if (dep.has_alternative) + { + if (!d->has_alternative) + d->has_alternative = *dep.has_alternative; + else + assert (*d->has_alternative == *dep.has_alternative); + } } else dependencies.push_back (move (dep)); @@ -1422,10 +1443,15 @@ namespace bpkg package_key&& dependent, bool existing, pair position, - packages&& deps) + packages&& deps, + optional has_alternative) : id (i) { - add (move (dependent), existing, position, move (deps)); + add (move (dependent), + existing, + position, + move (deps), + has_alternative); } // Add dependency of an existing dependent. @@ -1439,7 +1465,8 @@ namespace bpkg add (move (dependent), true /* existing */, position, - packages ({move (dep)})); + packages ({move (dep)}), + nullopt /* has_alternative */); } // Add dependencies of a dependent. @@ -1451,7 +1478,8 @@ namespace bpkg add (package_key&& dependent, bool existing, pair position, - packages&& deps) + packages&& deps, + optional has_alternative) { assert (position.first != 0 && position.second != 0); @@ -1463,7 +1491,7 @@ namespace bpkg { dependent_info& ddi (i->second); - ddi.add (dependency (position, move (deps))); + ddi.add (dependency (position, move (deps), has_alternative)); // Conceptually, on the first glance, we can only move from existing // to non-existing (e.g., due to a upgrade/downgrade later) and that @@ -1476,7 +1504,8 @@ namespace bpkg } else { - small_vector ds ({dependency (position, move (deps))}); + small_vector ds ({ + dependency (position, move (deps), has_alternative)}); dependents.emplace (move (dependent), dependent_info {existing, move (ds)}); @@ -1767,7 +1796,8 @@ namespace bpkg add (package_key dependent, bool existing, pair position, - postponed_configuration::packages dependencies) + postponed_configuration::packages dependencies, + optional has_alternative) { tracer trace ("postponed_configurations::add"); @@ -1893,7 +1923,12 @@ namespace bpkg { trace_add (c, true /* shadow */); - c.add (move (dependent), existing, position, move (dependencies)); + c.add (move (dependent), + existing, + position, + move (dependencies), + has_alternative); + break; } } @@ -1946,7 +1981,8 @@ namespace bpkg move (dependent), existing, position, - move (dependencies))); + move (dependencies), + has_alternative)); l5 ([&]{trace << "create " << *ri;}); } @@ -1958,7 +1994,11 @@ namespace bpkg trace_add (c, false /* shadow */); - c.add (move (dependent), existing, position, move (dependencies)); + c.add (move (dependent), + existing, + position, + move (dependencies), + has_alternative); // Try to merge other clusters into this cluster. // @@ -2277,6 +2317,52 @@ namespace bpkg } }; + // Set of dependency alternatives which were found unacceptable by the + // configuration negotiation machinery and need to be ignored on re- + // collection. + // + // Note that while negotiating the dependency alternative configuration for + // a dependent it may turn out that the configuration required by other + // dependents is not acceptable for this dependent. It can also happen that + // this dependent is involved in a negotiation cycle when two dependents + // continuously overwrite each other's configuration during re-negotiation. + // Both situations end up with the failure, unless the dependent has some + // other reused dependency alternative which can be tried instead. In the + // latter case, we note the problematic alternative and re-collect from + // scratch. On re-collection the unacceptable alternatives are ignored, + // similar to the disabled alternatives. + // + struct unacceptable_alternative + { + package_key package; + bpkg::version version; + pair position; // depends + alternative (1-based) + + unacceptable_alternative (package_key pkg, + bpkg::version ver, + pair pos) + : package (move (pkg)), version (move (ver)), position (pos) {} + + bool + operator< (const unacceptable_alternative& v) const + { + if (package != v.package) + return package < v.package; + + if (int r = version.compare (v.version)) + return r < 0; + + return position < v.position; + } + }; + + using unacceptable_alternatives = set; + + struct unaccept_alternative: scratch_collection + { + unaccept_alternative (): scratch_collection ("unacceptable alternative") {} + }; + struct build_packages: build_package_list { build_packages () = default; @@ -2432,6 +2518,7 @@ namespace bpkg postponed_packages* postponed_alts = nullptr, postponed_dependencies* postponed_deps = nullptr, postponed_positions* postponed_poss = nullptr, + unacceptable_alternatives* unacceptable_alts = nullptr, const function& vpb = nullptr) { using std::swap; // ...and not list::swap(). @@ -2441,10 +2528,11 @@ namespace bpkg // See the above notes. // bool recursive (dep_chain != nullptr); - assert ((postponed_repo != nullptr) == recursive && - (postponed_alts != nullptr) == recursive && - (postponed_deps != nullptr) == recursive && - (postponed_poss != nullptr) == recursive); + assert ((postponed_repo != nullptr) == recursive && + (postponed_alts != nullptr) == recursive && + (postponed_deps != nullptr) == recursive && + (postponed_poss != nullptr) == recursive && + (unacceptable_alts != nullptr) == recursive); // Only builds are allowed here. // @@ -2753,7 +2841,8 @@ namespace bpkg 0 /* max_alt_index */, *postponed_deps, postponed_cfgs, - *postponed_poss); + *postponed_poss, + *unacceptable_alts); return &p; } @@ -2827,6 +2916,10 @@ namespace bpkg // is handled by re-collecting packages from scratch, but now with the // knowledge about position this dependent needs to be re-evaluated to. // + // If a dependency alternative configuration cannot be negotiated between + // all the dependents, then unaccept_alternative can be thrown (see + // unacceptable_alternatives for details). + // struct postpone_dependency: scratch_collection { package_key package; @@ -2861,6 +2954,7 @@ namespace bpkg postponed_dependencies& postponed_deps, postponed_configurations& postponed_cfgs, postponed_positions& postponed_poss, + unacceptable_alternatives& unacceptable_alts, pair reeval_pos = make_pair(0, 0)) { @@ -3272,7 +3366,8 @@ namespace bpkg // Try to pre-collect build information (pre-builds) for the // dependencies of an alternative. Optionally, issue diagnostics into - // the specified diag record. + // the specified diag record. In the dry-run mode don't change the + // packages collection state (postponed_repo set, etc). // // Note that rather than considering an alternative as unsatisfactory // (returning no pre-builds) the function can fail in some cases @@ -3360,7 +3455,8 @@ namespace bpkg (const dependency_alternative& da, bool buildtime, const package_prerequisites* prereqs, - diag_record* dr = nullptr) + diag_record* dr = nullptr, + bool dry_run = false) -> precollect_result { prebuilds r; @@ -3850,13 +3946,17 @@ namespace bpkg // assert (dr == nullptr); - l5 ([&]{trace << "rep-postpone dependent " - << pkg.available_name_version_db () - << " due to dependency " << dp - << " and user-specified constraint " - << *dep_constr;}); + if (!dry_run) + { + l5 ([&]{trace << "rep-postpone dependent " + << pkg.available_name_version_db () + << " due to dependency " << dp + << " and user-specified constraint " + << *dep_constr;}); + + postponed_repo->insert (&pkg); + } - postponed_repo->insert (&pkg); return precollect_result (true /* postpone */); } @@ -4084,16 +4184,21 @@ namespace bpkg &postponed_deps, &postponed_cfgs, &postponed_poss, + &unacceptable_alts, &di, reeval, &reeval_pos, &reevaluated, &fail_reeval, + &edas, + &das, + &precollect, &trace, this] (const dependency_alternative& da, size_t dai, - prebuilds&& bs) + prebuilds&& bs, + const package_prerequisites* prereqs) { // Dependency alternative position. // @@ -4262,6 +4367,7 @@ namespace bpkg nullptr /* postponed_alts */, nullptr /* postponed_deps */, nullptr /* postponed_poss */, + nullptr /* unacceptable_alts */, verify)); package_key dpk (b.db, b.available->id.name); @@ -4390,7 +4496,8 @@ namespace bpkg 0 /* max_alt_index */, postponed_deps, postponed_cfgs, - postponed_poss); + postponed_poss, + unacceptable_alts); } // If this dependent has any dependencies with configurations @@ -4403,6 +4510,102 @@ namespace bpkg // if (!cfg_deps.empty ()) { + // First, determine if there is any unprocessed reused dependency + // alternative that we can potentially use instead of the current + // one if it turns out that a configuration for some of its + // dependencies cannot be negotiated between all the dependents + // (see unacceptable_alternatives for details). + // + bool has_alt (false); + { + // Find the index of the current dependency alternative. + // + size_t i (0); + for (; i != edas.size (); ++i) + { + if (&edas[i].first.get () == &da) + break; + } + + // The current dependency alternative must be present in the + // list. + // + assert (i != edas.size ()); + + // Return true if the current alternative is unacceptable. + // + auto unacceptable = + [&pk, &pkg, di, &i, &edas, &unacceptable_alts] () + { + // Convert to 1-base. + // + pair pos (di + 1, edas[i].second + 1); + + return unacceptable_alts.find ( + unacceptable_alternative (pk, + pkg.available->version, + pos)) != + unacceptable_alts.end (); + + }; + + // See if there is any unprocessed reused alternative to the + // right. + // + // Note that this is parallel to the alternative selection + // logic. + // + for (++i; i != edas.size (); ++i) + { + if (unacceptable ()) + continue; + + const dependency_alternative& a (edas[i].first); + + precollect_result r (precollect (a, + das.buildtime, + prereqs, + nullptr /* diag_record */, + true /* dru_run */)); + + if (r.builds && r.reused) + { + has_alt = true; + break; + } + } + + // If there are none and we are in the "recreate dependency + // decisions" mode, then repeat the search in the "make + // dependency decisions" mode. + // + if (!has_alt && prereqs != nullptr) + { + for (i = 0; i != edas.size (); ++i) + { + if (unacceptable ()) + continue; + + const dependency_alternative& a (edas[i].first); + + if (&a != &da) // Skip the current dependency alternative. + { + precollect_result r (precollect (a, + das.buildtime, + nullptr /* prereqs */, + nullptr /* diag_record */, + true /* dru_run */)); + + if (r.builds && r.reused) + { + has_alt = true; + break; + } + } + } + } + } + // Re-evaluation is a special case (it happens during cluster // negotiation; see collect_build_postponed()). // @@ -4417,7 +4620,8 @@ namespace bpkg postponed_cfgs.add (pk, true /* existing */, dp, - cfg_deps).first); + cfg_deps, + has_alt).first); // Can we merge clusters as a result? Seems so. // @@ -4486,7 +4690,11 @@ namespace bpkg // constructing exception. // pair> r ( - postponed_cfgs.add (pk, false /* existing */, dp, cfg_deps)); + postponed_cfgs.add (pk, + false /* existing */, + dp, + cfg_deps, + has_alt)); postponed_configuration& cfg (r.first); @@ -4588,12 +4796,8 @@ namespace bpkg // Similar to initial negotiation, resolve package skeletons // for this dependent and its dependencies. // - package_skeleton* dept; - { - build_package* b (entered_build (pk)); - assert (b != nullptr && b->skeleton); - dept = &*b->skeleton; - } + assert (pkg.skeleton); + package_skeleton& dept (*pkg.skeleton); // If a dependency has already been recursively collected, // then we can no longer call reload_defaults() or @@ -4630,8 +4834,33 @@ namespace bpkg } } - changed = negotiate_configuration ( - cfg.dependency_configurations, *dept, dp, depcs); + optional c ( + negotiate_configuration ( + cfg.dependency_configurations, dept, dp, depcs, has_alt)); + + // If the dependency alternative configuration cannot be + // negotiated for this dependent, then add an entry to + // unacceptable_alts and throw unaccept_alternative to + // recollect from scratch. + // + if (!c) + { + unacceptable_alts.emplace (pk, pkg.available->version, dp); + + l5 ([&]{trace << "unable to cfg-negotiate dependency " + << "alternative " << dp.first << ',' + << dp.second << " for dependent " + << pkg.available_name_version_db () + << ", throwing unaccept_alternative";}); + + // Don't print the "while satisfying..." chain. + // + dep_chain.clear (); + + throw unaccept_alternative (); + } + else + changed = *c; } // If the configuration hasn't changed, then we carry on. @@ -4724,7 +4953,8 @@ namespace bpkg 0 /* max_alt_index */, postponed_deps, postponed_cfgs, - postponed_poss); + postponed_poss, + unacceptable_alts); } else l5 ([&]{trace << "dependency " @@ -4810,7 +5040,7 @@ namespace bpkg // assert (!reeval || prereqs != nullptr); - for (;;) + for (bool unacceptable (false);;) { // The index and pre-collection result of the first satisfactory // alternative. @@ -4843,6 +5073,28 @@ namespace bpkg for (size_t i (0); i != edas.size (); ++i) { + // Skip the unacceptable alternatives. + // + { + // Convert to 1-base. + // + pair pos (di + 1, edas[i].second + 1); + + if (unacceptable_alts.find ( + unacceptable_alternative (pk, ap->version, pos)) != + unacceptable_alts.end ()) + { + unacceptable = true; + + l5 ([&]{trace << "dependency alternative " << pos.first << ',' + << pos.second << " for dependent " + << pkg.available_name_version_db () + << " is unacceptable, skipping";}); + + continue; + } + } + const dependency_alternative& da (edas[i].first); precollect_result r (precollect (da, das.buildtime, prereqs)); @@ -4903,6 +5155,7 @@ namespace bpkg // auto try_select = [postponed_alts, &max_alt_index, &edas, &pkg, + prereqs, reeval, &trace, &postpone, &collect, &select] @@ -4941,7 +5194,7 @@ namespace bpkg // assert (postponed_alts != nullptr); - if (!collect (da, dai, move (*r.builds))) + if (!collect (da, dai, move (*r.builds), prereqs)) { postpone (nullptr); // Already inserted into postponed_cfgs. return true; @@ -4999,11 +5252,16 @@ namespace bpkg if (r.reused || !reused_only) { + // If there are any unacceptable alternatives, then the + // remaining one should be reused. + // + assert (!unacceptable || r.reused); + const auto& eda (edas[first_alt->first]); const dependency_alternative& da (eda.first); size_t dai (eda.second); - if (!collect (da, dai, move (*r.builds))) + if (!collect (da, dai, move (*r.builds), prereqs)) { postpone (nullptr); // Already inserted into postponed_cfgs. break; @@ -5033,6 +5291,11 @@ namespace bpkg continue; } + // We shouldn't end up with the "no alternative to select" case if + // any alternatives are unacceptable. + // + assert (!unacceptable); + // Issue diagnostics and fail if there are no satisfactory // alternatives. // @@ -5167,7 +5430,8 @@ namespace bpkg size_t max_alt_index, postponed_dependencies& postponed_deps, postponed_configurations& postponed_cfgs, - postponed_positions& postponed_poss) + postponed_positions& postponed_poss, + unacceptable_alternatives& unacceptable_alts) { auto mi (map_.find (db, name)); assert (mi != map_.end ()); @@ -5187,7 +5451,8 @@ namespace bpkg max_alt_index, postponed_deps, postponed_cfgs, - postponed_poss); + postponed_poss, + unacceptable_alts); } // Collect the repointed dependents and their replaced prerequisites, @@ -5208,6 +5473,7 @@ namespace bpkg postponed_dependencies& postponed_deps, postponed_configurations& postponed_cfgs, postponed_positions& postponed_poss, + unacceptable_alternatives& unacceptable_alts, const function& fdb, const function& apc) { @@ -5291,7 +5557,8 @@ namespace bpkg &postponed_repo, &postponed_alts, &postponed_deps, - &postponed_poss); + &postponed_poss, + &unacceptable_alts); } } @@ -5473,6 +5740,7 @@ namespace bpkg postponed_configurations& postponed_cfgs, strings& postponed_cfgs_history, postponed_positions& postponed_poss, + unacceptable_alternatives& unacceptable_alts, const function& fdb, const repointed_dependents& rpt_depts, const function& apc, @@ -5958,6 +6226,7 @@ namespace bpkg postponed_deps, postponed_cfgs, postponed_poss, + unacceptable_alts, ed.dependency_position); ed.reevaluated = true; @@ -6004,6 +6273,7 @@ namespace bpkg pair pos; small_vector, 1> depcs; + bool has_alt; { // A non-negotiated cluster must only have one depends position // for each dependent. @@ -6015,6 +6285,14 @@ namespace bpkg pos = ds.position; + // Note that an existing dependent which initially doesn't have + // the has_alternative flag present should obtain it as a part of + // re-evaluation at this time. + // + assert (ds.has_alternative); + + has_alt = *ds.has_alternative; + depcs.reserve (ds.size ()); for (const package_key& pk: ds) { @@ -6027,8 +6305,31 @@ namespace bpkg } } - if (negotiate_configuration ( - pcfg->dependency_configurations, *dept, pos, depcs)) + optional changed ( + negotiate_configuration ( + pcfg->dependency_configurations, *dept, pos, depcs, has_alt)); + + // If the dependency alternative configuration cannot be negotiated + // for this dependent, then add an entry to unacceptable_alts and + // throw unaccept_alternative to recollect from scratch. + // + if (!changed) + { + assert (dept->available != nullptr); // Can't be system. + + const package_key& p (dept->package); + const version& v (dept->available->version); + + unacceptable_alts.emplace (p, v, pos); + + l5 ([&]{trace << "unable to cfg-negotiate dependency alternative " + << pos.first << ',' << pos.second << " for " + << "dependent " << package_string (p.name, v) + << p.db << ", throwing unaccept_alternative";}); + + throw unaccept_alternative (); + } + else if (*changed) { if (i != b) { @@ -6101,8 +6402,8 @@ namespace bpkg const package_configuration& pc ( pcfg->dependency_configurations[p]); - // Skip the verification if this is a system package - // without skeleton info. + // Skip the verification if this is a system package without + // skeleton info. // pair pr (b->skeleton->available != nullptr ? b->skeleton->verify_sensible (pc) @@ -6141,7 +6442,8 @@ namespace bpkg 0 /* max_alt_index */, postponed_deps, postponed_cfgs, - postponed_poss); + postponed_poss, + unacceptable_alts); } else l5 ([&]{trace << "dependency " << b->available_name_version_db () @@ -6262,7 +6564,8 @@ namespace bpkg 0 /* max_alt_index */, postponed_deps, postponed_cfgs, - postponed_poss); + postponed_poss, + unacceptable_alts); } // Negotiated (so can only be rolled back). @@ -6310,7 +6613,8 @@ namespace bpkg 0 /* max_alt_index */, postponed_deps, postponed_cfgs, - postponed_poss); + postponed_poss, + unacceptable_alts); } // Save the potential new dependency alternative-related postponements. @@ -6373,6 +6677,7 @@ namespace bpkg postponed_cfgs, postponed_cfgs_history, postponed_poss, + unacceptable_alts, fdb, rpt_depts, apc, @@ -6657,7 +6962,8 @@ namespace bpkg i, postponed_deps, postponed_cfgs, - postponed_poss); + postponed_poss, + unacceptable_alts); prog = (pas.find (p) == pas.end () || ndep != p->dependencies->size ()); @@ -6895,7 +7201,8 @@ namespace bpkg 0, postponed_deps, postponed_cfgs, - postponed_poss); + postponed_poss, + unacceptable_alts); assert (false); // Can't be here. } @@ -6917,7 +7224,8 @@ namespace bpkg 0, postponed_deps, postponed_cfgs, - postponed_poss); + postponed_poss, + unacceptable_alts); assert (false); // Can't be here. } @@ -10621,9 +10929,10 @@ namespace bpkg }; vector deps; - replaced_versions replaced_vers; - postponed_dependencies postponed_deps; - postponed_positions postponed_poss; + replaced_versions replaced_vers; + postponed_dependencies postponed_deps; + postponed_positions postponed_poss; + unacceptable_alternatives unacceptable_alts; // Map the repointed dependents to the replacement flags (see // repointed_dependents for details), unless --no-move is specified. @@ -10918,6 +11227,7 @@ namespace bpkg replaced_vers.clear (); postponed_deps.clear (); postponed_poss.clear (); + unacceptable_alts.clear (); scratch_exe = false; } @@ -11008,7 +11318,8 @@ namespace bpkg 0 /* max_alt_index */, postponed_deps, postponed_cfgs, - postponed_poss); + postponed_poss, + unacceptable_alts); } else { @@ -11062,6 +11373,7 @@ namespace bpkg postponed_deps, postponed_cfgs, postponed_poss, + unacceptable_alts, find_prereq_database, add_priv_cfg); } @@ -11142,7 +11454,8 @@ namespace bpkg &postponed_repo, &postponed_alts, &postponed_deps, - &postponed_poss); + &postponed_poss, + &unacceptable_alts); } } @@ -11181,6 +11494,7 @@ namespace bpkg postponed_cfgs, postponed_cfgs_history, postponed_poss, + unacceptable_alts, find_prereq_database, rpt_depts, add_priv_cfg); diff --git a/tests/common/dependency-alternatives/t11a/foo-0.2.0.tar.gz b/tests/common/dependency-alternatives/t11a/foo-0.2.0.tar.gz new file mode 100644 index 0000000..59bac4b Binary files /dev/null and b/tests/common/dependency-alternatives/t11a/foo-0.2.0.tar.gz differ diff --git a/tests/common/dependency-alternatives/t11a/fox-0.1.0.tar.gz b/tests/common/dependency-alternatives/t11a/fox-0.1.0.tar.gz new file mode 100644 index 0000000..19e7307 Binary files /dev/null and b/tests/common/dependency-alternatives/t11a/fox-0.1.0.tar.gz differ diff --git a/tests/common/dependency-alternatives/t11a/fox-0.2.0.tar.gz b/tests/common/dependency-alternatives/t11a/fox-0.2.0.tar.gz new file mode 100644 index 0000000..6f11b23 Binary files /dev/null and b/tests/common/dependency-alternatives/t11a/fox-0.2.0.tar.gz differ diff --git a/tests/pkg-build.testscript b/tests/pkg-build.testscript index 3533b8d..6a33757 100644 --- a/tests/pkg-build.testscript +++ b/tests/pkg-build.testscript @@ -180,7 +180,14 @@ # | |-- libbaz-1.0.0.tar.gz # | |-- libbox-0.1.0.tar.gz # | |-- libbox-1.0.0.tar.gz +# | |-- foo-0.1.0.tar.gz -> libfoo {require {config.libfoo.extras=true}} +# | |-- foo-0.2.0.tar.gz -> libfoo {require {config.libfoo.extras=true}} | libbar # | |-- foo-1.0.0.tar.gz -> libfoo {require {config.libfoo.extras=true}} +# | |-- fox-0.1.0.tar.gz -> libfoo {prefer {config.libfoo.extras=true} accept (false)} | +# | | libbar +# | |-- fox-0.2.0.tar.gz -> libfoo {prefer {config.libfoo.extras=false} accept (!$config.libfoo.extras)} | +# | | libfoo {prefer {config.libfoo.extras=true} accept (true)} | +# | | libbar {require {config.libbar.extras=true}} # | |-- fox-1.0.0.tar.gz -> libfoo {require {config.libfoo.extras=true}} # | |-- fux-1.0.0.tar.gz -> libfoo # | |-- fix-1.0.0.tar.gz -> foo {require {config.foo.extras=true}} @@ -5866,7 +5873,7 @@ test.options += --no-progress $pkg_status -r >>EOO; !libfoo configured !0.1.0 available 1.0.0 !fix configured !0.1.0 available 1.0.0 - foo configured 0.1.0 available 1.0.0 + foo configured 0.1.0 available 1.0.0 0.2.0 !libfoo configured !0.1.0 available 1.0.0 EOO @@ -6865,6 +6872,7 @@ test.options += --no-progress %.* trace: collect_build_prerequisites: cfg-postpone dependency libbar/1.0.0 of dependent box/1.0.0 trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent box/1.0.0 + %.* trace: postponed_configurations::add: add {box 1,1: libbar libfoo} to {foo | libfoo->{foo/1,1}}! trace: postponed_configurations::add: merge {bar | libbar->{bar/1,1}}! into {box foo | libfoo->{box/1,1 foo/1,1} libbar->{box/1,1}}! %.* @@ -6899,6 +6907,7 @@ test.options += --no-progress %.* trace: collect_build_prerequisites: cfg-postpone dependency libbar/1.0.0 of dependent box/1.0.0 trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent box/1.0.0 + %.* trace: postponed_configurations::add: add {box 1,1: libbar libfoo} to {foo | libfoo->{foo/1,1}}! trace: postponed_configurations::add: merge {bar | libbar->{bar/1,1}}! into {box foo | libfoo->{box/1,1 foo/1,1} libbar->{box/1,1}}! trace: collect_build_prerequisites: configuration for cfg-postponed dependencies of dependent box/1.0.0 is negotiated @@ -7389,6 +7398,7 @@ test.options += --no-progress trace: collect_build_prerequisites: resume boo/1.0.0 %.* trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent boo/1.0.0 + %.* trace: postponed_configurations::add: create {boo | libfoo->{boo/1,2}} trace: collect_build_prerequisites: postpone boo/1.0.0 trace: collect_build_postponed (1): begin {boo | libfoo->{boo/1,2}} @@ -7622,6 +7632,7 @@ test.options += --no-progress trace: collect_build_prerequisites: resume boo/1.0.0 %.* trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent boo/1.0.0 + %.* trace: postponed_configurations::add: add {boo 1,2: libfoo} to {foo | libfoo->{foo/1,1}}! %.* trace: collect_build_prerequisites: configuration for cfg-postponed dependencies of dependent boo/1.0.0 is negotiated @@ -7664,6 +7675,7 @@ test.options += --no-progress %.* trace: collect_build: add boo/1.0.0 trace: collect_build_prerequisites: cfg-postpone dependency boo/1.0.0 of dependent biz/1.0.0 + %.* trace: postponed_configurations::add: create {biz | boo->{biz/1,1}} trace: collect_build_prerequisites: postpone biz/1.0.0 trace: collect_build_postponed (0): begin @@ -7710,6 +7722,7 @@ test.options += --no-progress trace: collect_build_prerequisites: resume boo/1.0.0 %.* trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent boo/1.0.0 + %.* trace: postponed_configurations::add: add {boo 1,2: libfoo} to {foo | libfoo->{foo/1,1}}! trace: collect_build_prerequisites: configuration for cfg-postponed dependencies of dependent boo/1.0.0 is negotiated trace: collect_build_prerequisites: dependency libfoo/1.0.0 of dependent boo/1.0.0 is already (being) recursively collected, skipping @@ -11208,6 +11221,369 @@ test.options += --no-progress } } + : unaccept-alternative + : + { + +$clone_cfg + + : unacceptable + : + { + $clone_cfg; + + $* foo fox/0.1.0 ?libbar 2>>~%EOE%; + %.* + trace: pkg_build: refine package collection/plan execution from scratch + %.* + trace: collect_build: add foo/1.0.0 + trace: collect_build: add fox/0.1.0 + trace: collect_build_prerequisites: begin foo/1.0.0 + %.* + trace: collect_build: add libfoo/1.0.0 + trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent foo/1.0.0 + trace: postponed_configurations::add: create {foo | libfoo->{foo/1,1}} + trace: collect_build_prerequisites: postpone foo/1.0.0 + trace: collect_build_prerequisites: begin fox/0.1.0 + %.* + trace: collect_build_prerequisites: alt-postpone dependent fox/0.1.0 since max index is reached: 0 + info: dependency alternative: libfoo + { + prefer + { + config.libfoo.extras = true + } + + accept (false) + } + trace: collect_build_prerequisites: postpone fox/0.1.0 + trace: collect_build_postponed (0): begin + trace: collect_build_postponed (1): begin {foo | libfoo->{foo/1,1}} + %.* + trace: collect_build_postponed (1): cfg-negotiate begin {foo | libfoo->{foo/1,1}} + %.* + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependencies + trace: collect_build_prerequisites: begin libfoo/1.0.0 + trace: collect_build_prerequisites: end libfoo/1.0.0 + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependents + trace: collect_build_postponed (1): select cfg-negotiated dependency alternative for dependent foo/1.0.0 + trace: collect_build_prerequisites: resume foo/1.0.0 + trace: collect_build_prerequisites: end foo/1.0.0 + trace: collect_build_postponed (1): cfg-negotiate end {foo | libfoo->{foo/1,1}}! + trace: collect_build_postponed (1): index 1 collect alt-postponed fox/0.1.0 + trace: collect_build_prerequisites: resume fox/0.1.0 + %.* + trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent fox/0.1.0 + %.* + trace: postponed_configurations::add: add {fox 1,1: libfoo} to {foo | libfoo->{foo/1,1}}! + %.* + trace: collect_build_prerequisites: unable to cfg-negotiate dependency alternative 1,1 for dependent fox/0.1.0, throwing unaccept_alternative + trace: pkg_build: collection failed due to unacceptable alternative, retry from scratch + %.* + trace: pkg_build: refine package collection/plan execution from scratch + %.* + trace: collect_build: add foo/1.0.0 + trace: collect_build: add fox/0.1.0 + trace: collect_build_prerequisites: begin foo/1.0.0 + %.* + trace: collect_build: add libfoo/1.0.0 + trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent foo/1.0.0 + trace: postponed_configurations::add: create {foo | libfoo->{foo/1,1}} + trace: collect_build_prerequisites: postpone foo/1.0.0 + trace: collect_build_prerequisites: begin fox/0.1.0 + trace: collect_build_prerequisites: dependency alternative 1,1 for dependent fox/0.1.0 is unacceptable, skipping + %.* + trace: collect_build_prerequisites: no cfg-clause for dependency libbar/1.0.0 of dependent fox/0.1.0 + trace: collect_build_prerequisites: begin libbar/1.0.0 + trace: collect_build_prerequisites: end libbar/1.0.0 + trace: collect_build_prerequisites: end fox/0.1.0 + trace: collect_build_postponed (0): begin + trace: collect_build_postponed (1): begin {foo | libfoo->{foo/1,1}} + %.* + trace: collect_build_postponed (1): cfg-negotiate begin {foo | libfoo->{foo/1,1}} + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependencies + trace: collect_build_prerequisites: begin libfoo/1.0.0 + trace: collect_build_prerequisites: end libfoo/1.0.0 + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependents + trace: collect_build_postponed (1): select cfg-negotiated dependency alternative for dependent foo/1.0.0 + trace: collect_build_prerequisites: resume foo/1.0.0 + trace: collect_build_prerequisites: end foo/1.0.0 + trace: collect_build_postponed (1): cfg-negotiate end {foo | libfoo->{foo/1,1}}! + trace: collect_build_postponed (1): end {foo | libfoo->{foo/1,1}} + trace: collect_build_postponed (0): end + trace: execute_plan: simulate: yes + %.* + trace: evaluate_dependency: libbar/1.0.0: unchanged + %.* + trace: evaluate_dependency: libbar/1.0.0: unchanged + %.* + build plan: + new libfoo/1.0.0 (required by foo) + config.libfoo.extras=true (set by foo) + new foo/1.0.0 + new libbar/1.0.0 (required by fox) + new fox/0.1.0 + trace: execute_plan: simulate: no + %.* + EOE + + $pkg_status -r >>EOO; + !foo configured 1.0.0 + libfoo configured 1.0.0 + !fox configured !0.1.0 available 1.0.0 0.2.0 + libbar configured 1.0.0 + EOO + + $pkg_drop foo fox + } + + : negotiation-cycle1 + : + { + $clone_cfg; + + $* foo fox/0.2.0 2>>~%EOE%; + %.* + trace: pkg_build: refine package collection/plan execution from scratch + %.* + trace: collect_build: add foo/1.0.0 + trace: collect_build: add fox/0.2.0 + trace: collect_build_prerequisites: begin foo/1.0.0 + %.* + trace: collect_build: add libfoo/1.0.0 + trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent foo/1.0.0 + trace: postponed_configurations::add: create {foo | libfoo->{foo/1,1}} + trace: collect_build_prerequisites: postpone foo/1.0.0 + trace: collect_build_prerequisites: begin fox/0.2.0 + %.* + trace: collect_build_prerequisites: alt-postpone dependent fox/0.2.0 since max index is reached: 0 + info: dependency alternative: libfoo + { + prefer + { + config.libfoo.extras = false + } + + accept (!$config.libfoo.extras) + } + trace: collect_build_prerequisites: postpone fox/0.2.0 + trace: collect_build_postponed (0): begin + trace: collect_build_postponed (1): begin {foo | libfoo->{foo/1,1}} + %.* + trace: collect_build_postponed (1): cfg-negotiate begin {foo | libfoo->{foo/1,1}} + %.* + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependencies + trace: collect_build_prerequisites: begin libfoo/1.0.0 + trace: collect_build_prerequisites: end libfoo/1.0.0 + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependents + trace: collect_build_postponed (1): select cfg-negotiated dependency alternative for dependent foo/1.0.0 + trace: collect_build_prerequisites: resume foo/1.0.0 + trace: collect_build_prerequisites: end foo/1.0.0 + trace: collect_build_postponed (1): cfg-negotiate end {foo | libfoo->{foo/1,1}}! + trace: collect_build_postponed (1): index 1 collect alt-postponed fox/0.2.0 + trace: collect_build_prerequisites: resume fox/0.2.0 + %.* + trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent fox/0.2.0 + %.* + trace: postponed_configurations::add: add {fox 1,1: libfoo} to {foo | libfoo->{foo/1,1}}! + %.* + trace: collect_build_prerequisites: cfg-postponing dependent fox/0.2.0 involves (being) negotiated configurations and results in {foo fox | libfoo->{foo/1,1 fox/1,1}}!, throwing retry_configuration + trace: collect_build_postponed (0): cfg-negotiation of {foo | libfoo->{foo/1,1}} failed due to dependent fox, refining configuration + trace: collect_build_postponed (1): begin {foo | libfoo->{foo/1,1}} + %.* + trace: collect_build_postponed (1): cfg-negotiate begin {foo | libfoo->{foo/1,1}} + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependencies + trace: collect_build_prerequisites: begin libfoo/1.0.0 + trace: collect_build_prerequisites: end libfoo/1.0.0 + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependents + trace: collect_build_postponed (1): select cfg-negotiated dependency alternative for dependent foo/1.0.0 + trace: collect_build_prerequisites: resume foo/1.0.0 + trace: collect_build_prerequisites: end foo/1.0.0 + trace: collect_build_postponed (1): cfg-negotiate end {foo | libfoo->{foo/1,1}}! + trace: collect_build_postponed (1): index 1 collect alt-postponed fox/0.2.0 + trace: collect_build_prerequisites: resume fox/0.2.0 + %.* + trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent fox/0.2.0 + %.* + trace: postponed_configurations::add: add {fox 1,1: libfoo} to {foo | libfoo->{foo/1,1}}! + trace: collect_build_prerequisites: cfg-postponing dependent fox/0.2.0 involves (being) negotiated configurations and results in {foo fox | libfoo->{foo/1,1 fox/1,1}}!, throwing retry_configuration + trace: collect_build_postponed (0): cfg-negotiation of {foo | libfoo->{foo/1,1}} failed due to dependent fox, refining configuration + trace: collect_build_postponed (1): begin {foo | libfoo->{foo/1,1}} + %.* + trace: collect_build_postponed (1): cfg-negotiate begin {foo | libfoo->{foo/1,1}} + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependencies + trace: collect_build_prerequisites: begin libfoo/1.0.0 + trace: collect_build_prerequisites: end libfoo/1.0.0 + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependents + trace: collect_build_postponed (1): select cfg-negotiated dependency alternative for dependent foo/1.0.0 + trace: collect_build_prerequisites: resume foo/1.0.0 + trace: collect_build_prerequisites: end foo/1.0.0 + trace: collect_build_postponed (1): cfg-negotiate end {foo | libfoo->{foo/1,1}}! + trace: collect_build_postponed (1): index 1 collect alt-postponed fox/0.2.0 + trace: collect_build_prerequisites: resume fox/0.2.0 + %.* + trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent fox/0.2.0 + %.* + trace: postponed_configurations::add: add {fox 1,1: libfoo} to {foo | libfoo->{foo/1,1}}! + trace: collect_build_prerequisites: unable to cfg-negotiate dependency alternative 1,1 for dependent fox/0.2.0, throwing unaccept_alternative + trace: pkg_build: collection failed due to unacceptable alternative, retry from scratch + %.* + trace: pkg_build: refine package collection/plan execution from scratch + %.* + trace: collect_build: add foo/1.0.0 + trace: collect_build: add fox/0.2.0 + trace: collect_build_prerequisites: begin foo/1.0.0 + %.* + trace: collect_build: add libfoo/1.0.0 + trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent foo/1.0.0 + trace: postponed_configurations::add: create {foo | libfoo->{foo/1,1}} + trace: collect_build_prerequisites: postpone foo/1.0.0 + trace: collect_build_prerequisites: begin fox/0.2.0 + trace: collect_build_prerequisites: dependency alternative 1,1 for dependent fox/0.2.0 is unacceptable, skipping + %.* + trace: collect_build_prerequisites: alt-postpone dependent fox/0.2.0 since max index is reached: 1 + info: dependency alternative: libfoo + { + prefer + { + config.libfoo.extras = true + } + + accept (true) + } + trace: collect_build_prerequisites: postpone fox/0.2.0 + trace: collect_build_postponed (0): begin + trace: collect_build_postponed (1): begin {foo | libfoo->{foo/1,1}} + %.* + trace: collect_build_postponed (1): cfg-negotiate begin {foo | libfoo->{foo/1,1}} + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependencies + trace: collect_build_prerequisites: begin libfoo/1.0.0 + trace: collect_build_prerequisites: end libfoo/1.0.0 + trace: collect_build_postponed (1): recursively collect cfg-negotiated dependents + trace: collect_build_postponed (1): select cfg-negotiated dependency alternative for dependent foo/1.0.0 + trace: collect_build_prerequisites: resume foo/1.0.0 + trace: collect_build_prerequisites: end foo/1.0.0 + trace: collect_build_postponed (1): cfg-negotiate end {foo | libfoo->{foo/1,1}}! + trace: collect_build_postponed (1): index 1 collect alt-postponed fox/0.2.0 + trace: collect_build_prerequisites: resume fox/0.2.0 + trace: collect_build_prerequisites: dependency alternative 1,1 for dependent fox/0.2.0 is unacceptable, skipping + %.* + trace: collect_build_prerequisites: alt-postpone dependent fox/0.2.0 since max index is reached: 1 + info: dependency alternative: libfoo + { + prefer + { + config.libfoo.extras = true + } + + accept (true) + } + trace: collect_build_prerequisites: postpone fox/0.2.0 + trace: collect_build_postponed (1): index 2 collect alt-postponed fox/0.2.0 + trace: collect_build_prerequisites: resume fox/0.2.0 + trace: collect_build_prerequisites: dependency alternative 1,1 for dependent fox/0.2.0 is unacceptable, skipping + %.* + trace: collect_build_prerequisites: cfg-postpone dependency libfoo/1.0.0 of dependent fox/0.2.0 + %.* + trace: postponed_configurations::add: add {fox 1,2: libfoo} to {foo | libfoo->{foo/1,1}}! + trace: collect_build_prerequisites: configuration for cfg-postponed dependencies of dependent fox/0.2.0 is negotiated + trace: collect_build_prerequisites: dependency libfoo/1.0.0 of dependent fox/0.2.0 is already (being) recursively collected, skipping + trace: collect_build_prerequisites: end fox/0.2.0 + trace: collect_build_postponed (1): end {foo | libfoo->{foo/1,1}} + trace: collect_build_postponed (0): end + %.* + build plan: + new libfoo/1.0.0 (required by foo, fox) + config.libfoo.extras=true (set by foo) + new foo/1.0.0 + new fox/0.2.0 + trace: execute_plan: simulate: no + %.* + EOE + + $pkg_status -r >>EOO; + !foo configured 1.0.0 + libfoo configured 1.0.0 + !fox configured !0.2.0 available 1.0.0 + libfoo configured 1.0.0 + EOO + + $pkg_drop foo fox + } + + : negotiation-cycle2 + : + : Similar to the above but peforms one more negotiation half-cycle (see + : negotiate_configuration() for details). + : + { + $clone_cfg; + + # Note that this 'one more half-cycle' case doesn't reproduce always + # and depends on the negotiate_configuration() calls order for the fox + # and foo dependents. This order is not actually deterministic since + # depends on the order of these packages in the postponed_packages set + # (of the set type). Thus, the 'one more half-cycle' + # case only takes place if the order of the packages as they appear on + # the command line is preserved. + # + $* fox/0.2.0 foo/0.2.0 ?libfoo 2>>~%EOE%; + %.* + trace: pkg_build: refine package collection/plan execution from scratch + %.* + trace: collect_build: add fox/0.2.0 + trace: collect_build: add foo/0.2.0 + trace: collect_build_prerequisites: begin fox/0.2.0 + %.* + trace: collect_build_prerequisites: alt-postpone dependent fox/0.2.0 since max index is reached: 0 + info: dependency alternative: libfoo + { + prefer + { + config.libfoo.extras = false + } + + accept (!$config.libfoo.extras) + } + trace: collect_build_prerequisites: postpone fox/0.2.0 + trace: collect_build_prerequisites: begin foo/0.2.0 + %.* + trace: collect_build_prerequisites: alt-postpone dependent foo/0.2.0 since max index is reached: 0 + info: dependency alternative: libfoo + { + require + { + config.libfoo.extras = true + } + } + trace: collect_build_prerequisites: postpone foo/0.2.0 + %.* + %.+: unable to cfg-negotiate dependency alternative 1,1 for dependent fox/0.2.0, throwing unaccept_alternative% + trace: pkg_build: collection failed due to unacceptable alternative, retry from scratch + %.* + trace: pkg_build: refine package collection/plan execution from scratch + %.* + trace: collect_build_prerequisites: dependency alternative 1,1 for dependent fox/0.2.0 is unacceptable, skipping + %.* + build plan: + new libfoo/1.0.0 (required by foo, fox) + config.libfoo.extras=true (set by foo) + new fox/0.2.0 + new foo/0.2.0 + trace: execute_plan: simulate: no + %.* + EOE + + $pkg_status -r >>EOO; + !fox configured !0.2.0 available 1.0.0 + libfoo configured 1.0.0 + !foo configured !0.2.0 available 1.0.0 + libfoo configured 1.0.0 + EOO + + $pkg_drop foo fox + } + } + : all-repo-packages : : Don't match the tracing but just make sure that pkg-build doesn't crash -- cgit v1.1