aboutsummaryrefslogtreecommitdiff
path: root/bbot/worker/worker.cxx
diff options
context:
space:
mode:
authorKaren Arutyunov <karen@codesynthesis.com>2024-10-30 18:17:18 +0200
committerKaren Arutyunov <karen@codesynthesis.com>2024-11-04 10:45:54 +0200
commit2fb3a346e0a74d746fd88aa1f4f2988cb5e20c2d (patch)
tree0261567c5cfb567fb246ccba0fc107d12d8d3ee9 /bbot/worker/worker.cxx
parent144aeccfdcb12af2477a159bd57b67b85b2acddf (diff)
Add support for configuration-specific dependencies
Diffstat (limited to 'bbot/worker/worker.cxx')
-rw-r--r--bbot/worker/worker.cxx421
1 files changed, 331 insertions, 90 deletions
diff --git a/bbot/worker/worker.cxx b/bbot/worker/worker.cxx
index 810797b..e5210f3 100644
--- a/bbot/worker/worker.cxx
+++ b/bbot/worker/worker.cxx
@@ -883,6 +883,33 @@ run_bpkg (step_id step,
verbosity, cmd, forward<A> (a)...);
}
+template <typename V, typename... A>
+static result_status
+run_bpkg (step_id step,
+ const V& envvars,
+ tracer& t,
+ string& log,
+ const function<pre_run_function>& pre_run,
+ const regexes& warn_detect,
+ const optional<step_id>& bkp_step,
+ const optional<result_status>& bkp_status,
+ const strings& aux_env,
+ string& last_cmd,
+ const char* verbosity,
+ const string& cmd, A&&... a)
+{
+ return run_cmd (step,
+ t,
+ log,
+ pre_run,
+ nullptr /* out_str */, path () /* out_file */,
+ warn_detect,
+ "bpkg " + cmd,
+ bkp_step, bkp_status, aux_env, last_cmd,
+ process_env ("bpkg", envvars),
+ verbosity, cmd, forward<A> (a)...);
+}
+
template <typename... A>
static result_status
run_bpkg (step_id step,
@@ -1664,6 +1691,38 @@ build (size_t argc, const char* argv[])
if (!rm.status)
break;
+ // If we end up with multiple current configurations (root and install;
+ // see below), then when running the bpkg-pkg-build command we need to
+ // specify the configuration for each package explicitly via
+ // --config-uuid.
+ //
+ // While it's tempting to use the --config-name option instead of
+ // --config-uuid, that wouldn't work well for multiple current
+ // configurations. For --config-name the configuration search is carried
+ // out among configurations explicitly linked to the main configuration
+ // only. That's in contrast to --config-uuid, when the whole configuration
+ // cluster is searched (see bpkg-pkg-build implementation for details).
+ //
+ // Also, the build package configuration may contain the
+ // configuration-specific dependencies specified using the
+ // package-specific --config-uuid option. Thus, we do not generate random
+ // UUIDs but use some predefined values, which the user can refer to in
+ // the *-build-config package manifest values.
+ //
+ // Note, though, that currently there is no way to customarily configure a
+ // dependency in the private host and build2 configurations specifically.
+ // In the future we may support that by, for example, creating such a
+ // host/build2 configuration explicitly (making it not private) if any
+ // dependency needs to be configured in it.
+ //
+ const char* target_uuid ("00000000-0000-0000-0000-000000000001");
+ const char* host_uuid ("00000000-0000-0000-0000-000000000002");
+ const char* module_uuid ("00000000-0000-0000-0000-000000000003");
+ const char* install_uuid ("00000000-0000-0000-0000-000000000004");
+ const char* target_installed_uuid ("00000000-0000-0000-0000-000000000005");
+ const char* host_installed_uuid ("00000000-0000-0000-0000-000000000006");
+ const char* module_installed_uuid ("00000000-0000-0000-0000-000000000007");
+
// Parse the build package configuration represented as a whitespace
// separated list of the following potentially quoted bpkg-pkg-build
// command arguments:
@@ -1707,6 +1766,12 @@ build (size_t argc, const char* argv[])
// other than the main and external test package names are silently
// ignored.
//
+ // - The configuration-specific dependencies, potentially with the
+ // dependency-specific configuration variables (pkg_config_deps).
+ // Configured in configurations specified by their package-specific
+ // --config-uuid=<uuid> options (see the above *_uuid variables for the
+ // recognized configuration UUIDs).
+ //
multimap<string, string> pkg_args;
strings pkg_config_opts;
strings pkg_config_vars;
@@ -1714,6 +1779,87 @@ build (size_t argc, const char* argv[])
vector<pair<string, strings>> pkg_config_glob_deps; // ?sys:<pkg>
map<string, strings> pkg_config_pkgs; // <pkg>
+ // { --config-uuid=<uuid>... }+ (?[sys:]<pkg> | sys:<pkg>)
+ //
+ // Maps configuration UUIDs to dependencies, potentially, with the
+ // package-specific configuration variables.
+ //
+ map<string, vector<pair<string, strings>>> pkg_config_deps;
+
+ // Return true if any configuration-specific dependencies need to be
+ // configured in the specified configuration.
+ //
+ auto config_deps = [&pkg_config_deps] (const char* conf_uuid)
+ {
+ return pkg_config_deps.find (conf_uuid) != pkg_config_deps.end ();
+ };
+
+ // Add configuration-specific dependencies with the dependency-specific
+ // configuration variables, if present, to the arguments list. Return true
+ // if any dependency has been added to the list.
+ //
+ auto add_config_deps = [&pkg_config_deps] (const char* conf_uuid,
+ strings& args)
+ {
+ bool r (false);
+ auto i (pkg_config_deps.find (conf_uuid));
+
+ if (i != pkg_config_deps.end ())
+ {
+ const vector<pair<string, strings>>& deps (i->second);
+
+ // Add dependencies which have some configuration variables specified
+ // and count the number of others.
+ //
+ // ({ --config-uuid <uuid> <var>... }+ <pkg>)...
+ //
+ size_t no_vars (0);
+ for (const pair<string, strings>& d: deps)
+ {
+ if (!d.second.empty ())
+ {
+ args.push_back ("{");
+ args.push_back ("--config-uuid");
+ args.push_back (conf_uuid);
+ args.insert (args.end (), d.second.begin (), d.second.end ());
+ args.push_back ("}+");
+
+ args.push_back (d.first);
+ }
+ else
+ ++no_vars;
+ }
+
+ // Add dependencies which have no configuration variables specified.
+ //
+ // { --config-uuid <uuid> }+ { <pkg>... }
+ //
+ if (no_vars != 0)
+ {
+ args.push_back ("{");
+ args.push_back ("--config-uuid");
+ args.push_back (conf_uuid);
+ args.push_back ("}+");
+
+ if (no_vars != 1)
+ args.push_back ("{");
+
+ for (const pair<string, strings>& d: deps)
+ {
+ if (d.second.empty ())
+ args.push_back (d.first);
+ }
+
+ if (no_vars != 1)
+ args.push_back ("}");
+ }
+
+ r = true;
+ }
+
+ return r;
+ };
+
if (!tm.package_config.empty ())
{
struct abort {};
@@ -1830,20 +1976,46 @@ build (size_t argc, const char* argv[])
// regardless whether it is prefixed with '?' or not.
//
strings vars;
+ strings uuids;
while (ag.more ())
{
string da (unquote (ag.next ()));
- if (!var (da))
- fail ("argument is not a configuration variable for "
- "dependency " + a + ": '" + da + '\'');
- vars.push_back (move (da));
+ if (da.compare (0, 14, "--config-uuid=") == 0)
+ {
+ string uuid (da, 14);
+
+ if (uuid != target_uuid &&
+ uuid != host_uuid &&
+ uuid != module_uuid &&
+ uuid != install_uuid &&
+ uuid != target_installed_uuid &&
+ uuid != host_installed_uuid &&
+ uuid != module_installed_uuid)
+ fail ("unrecognized configuration UUID '" + uuid + '\'');
+
+ uuids.push_back (move (uuid));
+ }
+ else if (var (da))
+ {
+ vars.push_back (move (da));
+ }
+ else
+ {
+ fail ("argument is not a configuration variable nor "
+ "configuration UUID for dependency " + a + ": '" + da +
+ '\'');
+ }
}
- // Add the system dependency packages (prefixed with `?sys:`) to
- // a separate list, to specify them globally on the
- // bpkg-pkg-build command line for configuring them in all the
- // (being) created configurations.
+ // Add the system dependency packages (prefixed with `?sys:`) with
+ // no configuration specified to a separate list, to be specified
+ // globally on the bpkg-pkg-build command line for configuring
+ // them in all the (being) created configurations. Add the
+ // configuration-specific dependency packages to the
+ // per-configuration lists, to be specified on the bpkg-pkg-build
+ // command line for configuring them in the specific
+ // configurations.
//
// Note, though, that we will handle the build-to-hold system
// packages (prefixed with `sys:`) in the same way as non system
@@ -1856,30 +2028,51 @@ build (size_t argc, const char* argv[])
// sure that these dependencies are also auto-configured for the
// private configurations potentially created by bpkg-pkg-build.
//
- // Also note that in the future we may allow package-specific
- // --config-uuid options to only configure such packages in the
- // specified configurations. We may also invent the special
- // 00000000-0000-0000-0000-000000000005 configuration id to, for
- // example, only configure them at the
- // bpkg.test-separate-installed.configure.build step.
+
+ // If the specified dependency is configuration-specific, then add
+ // it to the configuration-specific lists and to the specified list
+ // otherwise.
+ //
+ // Note: moves out entries from uuids list.
//
- if (a.compare (0, 5, "?sys:") == 0) // Global system dependency.
+ auto add_dep = [&a, &vars, &uuids, &pkg_config_deps]
+ (vector<pair<string, strings>>& non_cfg_spec_deps)
{
- pkg_config_glob_deps.push_back (make_pair (move (a),
- move (vars)));
+ if (!uuids.empty ())
+ {
+ for (string& uuid: uuids)
+ pkg_config_deps[move (uuid)].emplace_back (a, vars);
+ }
+ else
+ non_cfg_spec_deps.emplace_back (move (a), move (vars));
+ };
+
+ // Configuration-specific or global package dependency.
+ //
+ if (a.compare (0, 5, "?sys:") == 0)
+ {
+ add_dep (pkg_config_glob_deps);
}
- else if (a[0] == '?' || // Main package dependency.
- a.compare (0, 4, "sys:") == 0)
+ //
+ // Configuration-specific or main package dependency.
+ //
+ else if (a[0] == '?' || a.compare (0, 4, "sys:") == 0)
{
- pkg_config_main_deps.push_back (make_pair (move (a),
- move (vars)));
+ add_dep (pkg_config_main_deps);
}
- else // Build-to-hold package.
+ //
+ // Build-to-hold package.
+ //
+ else
{
if (vars.empty ())
fail ("no configuration variables specified for package '" +
a + '\'');
+ if (!uuids.empty ())
+ fail ("configuration UUID cannot be specified for "
+ "build-to-hold package '" + a + '\'');
+
auto i (pkg_config_pkgs.find (a));
if (i == pkg_config_pkgs.end ())
@@ -2448,9 +2641,9 @@ build (size_t argc, const char* argv[])
create_host ? host_conf :
module_conf);
- // Note that bpkg doesn't support configuring bootstrap module
- // dependents well, not distinguishing such modules from regular ones
- // (see pkg_configure() for details). Thus, we need to pass the
+ // Note that bpkg doesn't support configuring bootstrap module dependents
+ // well, not distinguishing such modules from regular ones (see
+ // pkg_configure() for details). Thus, we need to pass the
// !config.import.* global override wherever required ourselves.
//
// Also note that since this override is global, it may only be specified
@@ -2463,6 +2656,19 @@ build (size_t argc, const char* argv[])
bootstrap_import = "!config.import." + pkg_var + '=' +
(rwd / main_pkg_conf).string ();
+ // Log the configuration UUID as a comment.
+ //
+ auto log_conf_uuid = [] (tracer& t, string& log,
+ const char* uuid,
+ const char* name)
+ {
+ string s (uuid);
+ s += " - ";
+ s += name;
+
+ log_comment (t, log, s);
+ };
+
// Configure.
//
{
@@ -2487,26 +2693,6 @@ build (size_t argc, const char* argv[])
//
change_wd (trace, &r.log, rwd);
- // If we end up with multiple current configurations (root and install)
- // then when running the bpkg-pkg-build command we need to specify the
- // configuration for each package explicitly via --config-uuid.
- //
- // While it's tempting to use the --config-name option instead of
- // --config-uuid, that wouldn't work well for multiple current
- // configurations. For --config-name the configuration search is carried
- // out among configurations explicitly linked to the main configuration
- // only. That's in contrast to --config-uuid, when the whole
- // configuration cluster is searched (see bpkg-pkg-build implementation
- // for details).
- //
- // Let's not generate random UUIDs but use some predefined values which
- // we can easily recognize in the build logs.
- //
- const char* target_uuid ("00000000-0000-0000-0000-000000000001");
- const char* host_uuid ("00000000-0000-0000-0000-000000000002");
- const char* module_uuid ("00000000-0000-0000-0000-000000000003");
- const char* install_uuid ("00000000-0000-0000-0000-000000000004");
-
// Let's however distinguish the target package as a simple common case
// and simplify the configuration creation and packages configuration
// commands making them more readable in the build log. For this simple
@@ -2534,7 +2720,9 @@ build (size_t argc, const char* argv[])
"-V",
"create",
"-d", target_conf,
- !target_pkg ? cstrings ({"--uuid", target_uuid}) : cstrings (),
+ (!target_pkg || config_deps (target_uuid)
+ ? cstrings ({"--uuid", target_uuid})
+ : cstrings ()),
step_args (modules, s, f1, f2),
step_args (env_args, s, f1, f2),
step_args (tgt_args, s, f1, f2),
@@ -3073,11 +3261,13 @@ build (size_t argc, const char* argv[])
// <pkg-config-args>
// <pkg-config-opts>
// --
- // { <pkg-config-vars>|config.<pkg-name>.develop=false }+ <pkg>
- // { <rtt-config-vars>|config.<runtime-test-name>.develop=false }+ <runtime-test>...
- // { <dep-config-vars> }+ <main-dep>...
- // <main-dep>...
- // <glob-dep>...
+ // { <pkg-config-vars>|config.<pkg-name>.develop=false }+ <pkg>
+ // { <rtt-config-vars>|config.<runtime-test-name>.develop=false }+ <runtime-test>...
+ // { <dep-config-vars> }+ <main-dep>...
+ // <main-dep>...
+ // <glob-dep>...
+ // { <target-config> <dep-config-vars> }+ <dep>...
+ // { <target-config> }+ { <dep>... }
//
step_id s (step_id::bpkg_target_configure_build);
step_id f1 (step_id::b_configure);
@@ -3197,6 +3387,8 @@ build (size_t argc, const char* argv[])
// { <build-config> <install-config> <dep-config-vars> }+ <main-dep>...
// { <build-config> <install-config> }+ { <main-dep>... }
// <glob-dep>...
+ // { <(target|host|module|install)-config> <dep-config-vars> }+ <dep>...
+ // { <(target|host|module|install)-config> }+ { <dep>... }...
//
// Main package configuration name.
@@ -3395,6 +3587,8 @@ build (size_t argc, const char* argv[])
// Add dependencies which have some configuration variables
// specified and count the number of others.
//
+ // ({ --config-uuid=<uuid>... <var>... }+ <pkg>)...
+ //
size_t no_vars (0);
for (const pair<string, strings>& d: pkg_config_main_deps)
{
@@ -3423,6 +3617,8 @@ build (size_t argc, const char* argv[])
// Add dependencies which have no configuration variables specified.
//
+ // { --config-uuid=<uuid>... }+ { <pkg>... }
+ //
if (no_vars != 0)
{
pkgs.push_back ("{");
@@ -3458,6 +3654,24 @@ build (size_t argc, const char* argv[])
for (const pair<string, strings>& d: pkg_config_glob_deps)
pkgs.push_back (d.first);
+ // Only log configuration UUIDs if they are specified on the command
+ // line.
+ //
+ bool log_uuids (!target_pkg);
+
+ // Add the configuration-specific dependencies.
+ //
+ // Note that while it's tempting to optimize by omitting the
+ // --config-uuid option for a simple target configuration only case,
+ // that would change the semantics since ?sys:<dependency> would
+ // potentially be configured in the private host and module
+ // configurations in this case.
+ //
+ log_uuids = add_config_deps (target_uuid, pkgs) || log_uuids;
+ log_uuids = add_config_deps (host_uuid, pkgs) || log_uuids;
+ log_uuids = add_config_deps (module_uuid, pkgs) || log_uuids;
+ log_uuids = add_config_deps (install_uuid, pkgs) || log_uuids;
+
// Finally, configure all the packages.
//
{
@@ -3466,40 +3680,27 @@ build (size_t argc, const char* argv[])
optional<string> dependency_checksum;
- // Only log configuration UUIDs if they are specified on the command
- // line.
- //
- function<pre_run_function> log_uuids (
- [&r, &trace,
- target_uuid, host_uuid, module_uuid, install_uuid,
- target_pkg] ()
- {
- if (!target_pkg)
- {
- auto log = [&r, &trace] (const char* uuid, const char* name)
- {
- string s (uuid);
- s += " - ";
- s += name;
+ function<pre_run_function> log_uuids_func;
- log_comment (trace, r.log, s);
- };
-
- log_comment (trace, r.log, "");
-
- log (target_uuid, "target");
- log (host_uuid, "host");
- log (module_uuid, "module");
- log (install_uuid, "install");
-
- log_comment (trace, r.log, "");
- }
- });
+ if (log_uuids)
+ {
+ log_uuids_func = [&r, &trace,
+ target_uuid, host_uuid, module_uuid, install_uuid,
+ &log_conf_uuid] ()
+ {
+ log_comment (trace, r.log, "");
+ log_conf_uuid (trace, r.log, target_uuid, "target");
+ log_conf_uuid (trace, r.log, host_uuid, "host");
+ log_conf_uuid (trace, r.log, module_uuid, "module");
+ log_conf_uuid (trace, r.log, install_uuid, "install");
+ log_comment (trace, r.log, "");
+ };
+ }
r.status |= run_bpkg (
b,
trace, r.log,
- log_uuids,
+ log_uuids_func,
dependency_checksum,
wre,
bkp_step, bkp_status, aux_env, last_cmd,
@@ -5062,6 +5263,9 @@ build (size_t argc, const char* argv[])
"-V",
"create",
"-d", target_conf,
+ (config_deps (target_installed_uuid)
+ ? cstrings ({"--uuid", target_installed_uuid})
+ : cstrings ()),
step_args (modules, s, f),
step_args (env_args, s, f),
step_args (tgt_args, s, f),
@@ -5100,6 +5304,9 @@ build (size_t argc, const char* argv[])
"-d", host_conf,
"--type", "host",
"--name", "host",
+ (config_deps (host_installed_uuid)
+ ? cstrings ({"--uuid", host_installed_uuid})
+ : cstrings ()),
step_args (modules, s, f),
step_args (env_args, s, f),
step_args (tgt_args, s, f),
@@ -5144,7 +5351,10 @@ build (size_t argc, const char* argv[])
"--existing",
"-d", module_conf,
"--type", "build2",
- "--name", "module");
+ "--name", "module",
+ (config_deps (module_installed_uuid)
+ ? cstrings ({"--uuid", module_installed_uuid})
+ : cstrings ()));
if (!r.status)
break;
@@ -5370,14 +5580,14 @@ build (size_t argc, const char* argv[])
// Configure all the packages using a single bpkg-pkg-build command.
//
- // bpkg build --configure-only <env-config-args>
- // <tgt-config-args>
- // <pkg-config-args>
- // { <config> <rtt-config-vars> }+ <runtime-test>...
- // { <config> }+ { <runtime-test>... }
- // { <btt-config-vars> }+ <buildtime-test>...
- // ?sys:<pkg>
- // <glob-dep>...
+ // bpkg build --configure-only <env-config-args> <tgt-config-args> <pkg-config-args>
+ // { <build-installed-config> <rtt-config-vars> }+ <runtime-test>...
+ // { <build-installed-config> }+ { <runtime-test>... }
+ // { <btt-config-vars> }+ <buildtime-test>...
+ // ?sys:<pkg>
+ // <glob-dep>...
+ // { <(target|host|module)-installed-config> <dep-config-vars> }+ <dep>...
+ // { <(target|host|module)-installed-config> }+ { <dep>... }...
//
strings pkgs;
@@ -5497,6 +5707,17 @@ build (size_t argc, const char* argv[])
for (const pair<string, strings>& d: pkg_config_glob_deps)
pkgs.push_back (d.first);
+ // Only log configuration UUIDs if they are specified on the
+ // command line.
+ //
+ bool log_uuids (false);
+
+ // Add the configuration-specific dependencies.
+ //
+ log_uuids = add_config_deps (target_installed_uuid, pkgs) || log_uuids;
+ log_uuids = add_config_deps (host_installed_uuid, pkgs) || log_uuids;
+ log_uuids = add_config_deps (module_installed_uuid, pkgs) || log_uuids;
+
// Finally, configure all the test packages.
//
{
@@ -5513,10 +5734,30 @@ build (size_t argc, const char* argv[])
step_id f (step_id::bpkg_test_separate_installed_configure_build);
+ function<pre_run_function> log_uuids_func;
+
+ if (log_uuids)
+ {
+ log_uuids_func = [&r, &trace,
+ target_installed_uuid,
+ host_installed_uuid,
+ module_installed_uuid,
+ &log_conf_uuid] ()
+ {
+ log_comment (trace, r.log, "");
+ log_conf_uuid (trace, r.log, target_installed_uuid, "target installed");
+ log_conf_uuid (trace, r.log, host_installed_uuid, "host installed");
+ log_conf_uuid (trace, r.log, module_installed_uuid, "module installed");
+ log_comment (trace, r.log, "");
+ };
+ }
+
r.status |= run_bpkg (
b,
envvars,
- trace, r.log, wre,
+ trace, r.log,
+ log_uuids_func,
+ wre,
bkp_step, bkp_status, aux_env, last_cmd,
"-v",
"build",