// file : libbuild2/version/init.cxx -*- C++ -*- // license : MIT; see accompanying LICENSE file #include <libbuild2/version/init.hxx> #include <cstring> // strchr() #include <libbutl/manifest-parser.hxx> #include <libbuild2/scope.hxx> #include <libbuild2/variable.hxx> #include <libbuild2/diagnostics.hxx> #include <libbuild2/config/utility.hxx> #include <libbuild2/dist/module.hxx> #include <libbuild2/version/rule.hxx> #include <libbuild2/version/module.hxx> #include <libbuild2/version/utility.hxx> #include <libbuild2/version/snapshot.hxx> using namespace std; using namespace butl; namespace build2 { namespace version { static const path manifest_file ("manifest"); static const in_rule in_rule_; static const manifest_install_rule manifest_install_rule_; static void dist_callback (const path&, const scope&, void*); void boot_post (scope& rs, const location&, module_boot_post_extra& extra) { // If the dist module is used, set its dist.package and register the // post-processing callback. // if (auto* dm = rs.find_module<dist::module> (dist::module::name)) { // Don't touch if dist.package was set by the user. // value& val (rs.assign (dm->var_dist_package)); if (!val) { auto& m (extra.module_as<module> ()); const standard_version& v (m.version); // We've already verified in boot() it is named. // string p (project (rs).string ()); p += '-'; p += v.string (); val = move (p); // Only register the post-processing callback if this is a rewritten // snapshot. // if (m.rewritten) dm->register_callback (dir_path (".") / manifest_file, &dist_callback, &m); } } } void boot (scope& rs, const location& l, module_boot_extra& extra) { tracer trace ("version::boot"); l5 ([&]{trace << "for " << rs;}); context& ctx (rs.ctx); // Extract the version from the manifest file. As well as summary and // url while at it. // // Also, as a sanity check, verify the package name matches the build // system project name. // string sum; string url; standard_version v; dependencies ds; { path f (rs.src_path () / manifest_file); try { if (!file_exists (f)) fail (l) << "no manifest file in " << rs.src_path (); ifdstream ifs (f); manifest_parser p (ifs, f.string ()); manifest_name_value nv (p.next ()); if (!nv.name.empty () || nv.value != "1") fail (l) << "unsupported manifest format in " << f; for (nv = p.next (); !nv.empty (); nv = p.next ()) { if (nv.name == "name") { const project_name& pn (project (rs)); if (pn.empty ()) fail (l) << "version module loaded in unnamed project"; if (nv.value != pn.string ()) { path bf (rs.src_path () / rs.root_extra->bootstrap_file); location ml (f, nv.value_line, nv.value_column); location bl (bf); fail (ml) << "package name " << nv.value << " does not match " << "build system project name " << pn << info (bl) << "build system project name specified here"; } } if (nv.name == "summary") sum = move (nv.value); else if (nv.name == "url") url = move (nv.value); else if (nv.name == "version") { try { // Allow the package stub versions in the 0+<revision> form. // While not standard, we want to use the version module for // packaging stubs. // v = standard_version (nv.value, standard_version::allow_stub); } catch (const invalid_argument& e) { fail << "invalid standard version '" << nv.value << "': " << e; } } else if (nv.name == "depends") { string v (move (nv.value)); // Parse the dependency and add it to the map (see // bpkg::dependency_alternatives class for dependency syntax). // // Note that currently we only consider simple dependencies: // singe package without alternatives, clauses, or newlines. // In the future, if/when we add full support, we will likely // keep this as a fast path. // // Also note that we don't do exhaustive validation here leaving // it to the package manager. // Get rid of the comment. // // Note that we can potentially mis-detect the comment // separator, since ';' can be a part of some of the dependency // alternative clauses. If that's the case, we will skip the // dependency later. // size_t p; if ((p = v.find (';')) != string::npos) v.resize (p); // Skip the dependency if it is not a simple one. // // Note that we will check for the presence of the reflect // clause later since `=` can also be in the constraint. // if (v.find_first_of ("{?|\n") != string::npos) continue; // Find the beginning of the dependency package name, skipping // the build-time marker, if present. // bool buildtime (v[0] == '*'); size_t b (buildtime ? v.find_first_not_of (" \t", 1) : 0); if (b == string::npos) fail (l) << "invalid dependency " << v << ": no package name"; // Find the end of the dependency package name. // p = v.find_first_of (" \t=<>[(~^", b); // Dependency name (without leading/trailing white-spaces). // string n (v, b, p == string::npos ? p : p - b); string vc; // Empty if no constraint is specified // Position to the first non-whitespace character after the // dependency name, which, if present, can be a part of the // version constraint or the reflect clause. // if (p != string::npos) p = v.find_first_not_of (" \t", p); if (p != string::npos) { // Check if this is definitely not a version constraint and // drop this dependency if that's the case. // if (strchr ("=<>[(~^", v[p]) == nullptr) continue; // Ok, we have a constraint, check that there is no reflect // clause after it (the only other valid `=` in a constraint // is in the immediately following character as part of // `==`, `<=`, or `>=`). // if (v.size () > p + 2 && v.find ('=', p + 2) != string::npos) continue; vc.assign (v, p, string::npos); trim (vc); } // Finally, add the dependency to the map. // try { package_name pn (move (n)); string v (pn.variable ()); ds.emplace (move (v), dependency {move (pn), move (vc), buildtime}); } catch (const invalid_argument& e) { fail (l) << "invalid dependency package name '" << n << "': " << e; } } } } catch (const manifest_parsing& e) { location l (f, e.line, e.column); fail (l) << e.description; } catch (const io_error& e) { fail (l) << "unable to read from " << f << ": " << e; } catch (const system_error& e) // EACCES, etc. { fail (l) << "unable to access manifest " << f << ": " << e; } if (v.empty ()) fail (l) << "no version in " << f; } // If this is the latest snapshot (i.e., the -a.1.z kind), then load the // snapshot number and id (e.g., commit date and id from git). // bool committed (true); bool rewritten (false); if (v.snapshot () && v.snapshot_sn == standard_version::latest_sn) { snapshot ss (extract_snapshot (rs)); if (!ss.empty ()) { v.snapshot_sn = ss.sn; v.snapshot_id = move (ss.id); committed = ss.committed; rewritten = true; } else committed = false; } // If there is a dependency on the build system itself, check it (so // there is no need for explicit using build@X.Y.Z). // { auto i (ds.find ("build2")); if (i != ds.end () && i->second.buildtime && !i->second.constraint.empty ()) try { check_build_version ( standard_version_constraint (i->second.constraint, v), l); } catch (const invalid_argument& e) { fail (l) << "invalid version constraint for dependency build2 " << i->second.constraint << ": " << e; } } // Set all the version.* variables. // // Note also that we have "gifted" the config.version variable name to // the config module. // auto set = [&rs] (const char* var, auto val) { using T = decltype (val); rs.assign<T> (var, move (val)); }; if (!sum.empty ()) rs.assign (ctx.var_project_summary, move (sum)); if (!url.empty ()) rs.assign (ctx.var_project_url, move (url)); set ("version", v.string ()); // Project version (var_version). set ("version.project", v.string_project ()); set ("version.project_number", v.version); // Enough of project version for unique identification (can be used in // places like soname, etc). // set ("version.project_id", v.string_project_id ()); set ("version.stub", v.stub ()); // bool set ("version.epoch", uint64_t (v.epoch)); set ("version.major", uint64_t (v.major ())); set ("version.minor", uint64_t (v.minor ())); set ("version.patch", uint64_t (v.patch ())); optional<uint16_t> a (v.alpha ()); optional<uint16_t> b (v.beta ()); set ("version.alpha", a.has_value ()); set ("version.beta", b.has_value ()); set ("version.pre_release", v.pre_release ().has_value ()); set ("version.pre_release_string", v.string_pre_release ()); set ("version.pre_release_number", uint64_t (a ? *a : b ? *b : 0)); set ("version.snapshot", v.snapshot ()); // bool set ("version.snapshot_sn", v.snapshot_sn); // uint64 set ("version.snapshot_id", v.snapshot_id); // string set ("version.snapshot_string", v.string_snapshot ()); set ("version.snapshot_committed", committed); // bool set ("version.revision", uint64_t (v.revision)); // Create the module instance. // extra.set_module ( new module (project (rs), move (v), committed, rewritten, move (ds))); // Initialize second (dist.package, etc). // extra.post = &boot_post; extra.init = module_boot_init::before_second; } bool init (scope& rs, scope&, const location& l, bool first, bool, module_init_extra&) { tracer trace ("version::init"); if (!first) fail (l) << "multiple version module initializations"; // Load in.base (in.* variables, in{} target type). // load_module (rs, rs, "in.base", l); // Register rules. // rs.insert_rule<file> (perform_update_id, "version.in", in_rule_); rs.insert_rule<file> (perform_clean_id, "version.in", in_rule_); rs.insert_rule<file> (configure_update_id, "version.in", in_rule_); if (cast_false<bool> (rs["install.booted"])) { rs.insert_rule<manifest> ( perform_install_id, "version.install", manifest_install_rule_); } return true; } static void dist_callback (const path& f, const scope& rs, void* data) { module& m (*static_cast<module*> (data)); // Complain if this is an uncommitted snapshot. // if (!m.committed && !cast_false<bool> (rs["config.dist.uncommitted"])) fail << "distribution of uncommitted project " << rs.src_path () << info << "specify config.dist.uncommitted=true to force"; // The plan is simple: fixing up the version in a temporary file then // move it to the original. // auto_rmfile t (fixup_manifest (rs.ctx, f, path::temp_path ("manifest"), m.version)); mvfile (t.path, f, verb_never); t.cancel (); } static const module_functions mod_functions[] = { // NOTE: don't forget to also update the documentation in init.hxx if // changing anything here. {"version", boot, init}, {nullptr, nullptr, nullptr} }; const module_functions* build2_version_load () { return mod_functions; } } }