mirror of
https://github.com/NixOS/nix.git
synced 2025-11-15 06:52:43 +01:00
Old code is now just used for `nix build` --- there is no CLI breaking change. Test the new format, too. The new format is not currently used, but will be used going forward, for example in the C API. Progress on #13570
295 lines
8.9 KiB
C++
295 lines
8.9 KiB
C++
#include "nix/store/derived-path.hh"
|
|
#include "nix/store/derivations.hh"
|
|
#include "nix/store/store-api.hh"
|
|
#include "nix/util/comparator.hh"
|
|
#include "nix/util/json-utils.hh"
|
|
|
|
#include <optional>
|
|
|
|
namespace nix {
|
|
|
|
// Custom implementation to avoid `ref` ptr equality
|
|
GENERATE_CMP_EXT(, std::strong_ordering, SingleDerivedPathBuilt, *me->drvPath, me->output);
|
|
|
|
// Custom implementation to avoid `ref` ptr equality
|
|
|
|
// TODO no `GENERATE_CMP_EXT` because no `std::set::operator<=>` on
|
|
// Darwin, per header.
|
|
GENERATE_EQUAL(, DerivedPathBuilt ::, DerivedPathBuilt, *me->drvPath, me->outputs);
|
|
GENERATE_ONE_CMP(, bool, DerivedPathBuilt ::, <, DerivedPathBuilt, *me->drvPath, me->outputs);
|
|
|
|
std::string DerivedPath::Opaque::to_string(const StoreDirConfig & store) const
|
|
{
|
|
return store.printStorePath(path);
|
|
}
|
|
|
|
std::string SingleDerivedPath::Built::to_string(const StoreDirConfig & store) const
|
|
{
|
|
return drvPath->to_string(store) + "^" + output;
|
|
}
|
|
|
|
std::string SingleDerivedPath::Built::to_string_legacy(const StoreDirConfig & store) const
|
|
{
|
|
return drvPath->to_string(store) + "!" + output;
|
|
}
|
|
|
|
std::string DerivedPath::Built::to_string(const StoreDirConfig & store) const
|
|
{
|
|
return drvPath->to_string(store) + '^' + outputs.to_string();
|
|
}
|
|
|
|
std::string DerivedPath::Built::to_string_legacy(const StoreDirConfig & store) const
|
|
{
|
|
return drvPath->to_string_legacy(store) + "!" + outputs.to_string();
|
|
}
|
|
|
|
std::string SingleDerivedPath::to_string(const StoreDirConfig & store) const
|
|
{
|
|
return std::visit([&](const auto & req) { return req.to_string(store); }, raw());
|
|
}
|
|
|
|
std::string DerivedPath::to_string(const StoreDirConfig & store) const
|
|
{
|
|
return std::visit([&](const auto & req) { return req.to_string(store); }, raw());
|
|
}
|
|
|
|
std::string SingleDerivedPath::to_string_legacy(const StoreDirConfig & store) const
|
|
{
|
|
return std::visit(
|
|
overloaded{
|
|
[&](const SingleDerivedPath::Built & req) { return req.to_string_legacy(store); },
|
|
[&](const SingleDerivedPath::Opaque & req) { return req.to_string(store); },
|
|
},
|
|
this->raw());
|
|
}
|
|
|
|
std::string DerivedPath::to_string_legacy(const StoreDirConfig & store) const
|
|
{
|
|
return std::visit(
|
|
overloaded{
|
|
[&](const DerivedPath::Built & req) { return req.to_string_legacy(store); },
|
|
[&](const DerivedPath::Opaque & req) { return req.to_string(store); },
|
|
},
|
|
this->raw());
|
|
}
|
|
|
|
DerivedPath::Opaque DerivedPath::Opaque::parse(const StoreDirConfig & store, std::string_view s)
|
|
{
|
|
return {store.parseStorePath(s)};
|
|
}
|
|
|
|
void drvRequireExperiment(const SingleDerivedPath & drv, const ExperimentalFeatureSettings & xpSettings)
|
|
{
|
|
std::visit(
|
|
overloaded{
|
|
[&](const SingleDerivedPath::Opaque &) {
|
|
// plain drv path; no experimental features required.
|
|
},
|
|
[&](const SingleDerivedPath::Built &) { xpSettings.require(Xp::DynamicDerivations); },
|
|
},
|
|
drv.raw());
|
|
}
|
|
|
|
SingleDerivedPath::Built SingleDerivedPath::Built::parse(
|
|
const StoreDirConfig & store,
|
|
ref<const SingleDerivedPath> drv,
|
|
OutputNameView output,
|
|
const ExperimentalFeatureSettings & xpSettings)
|
|
{
|
|
drvRequireExperiment(*drv, xpSettings);
|
|
return {
|
|
.drvPath = drv,
|
|
.output = std::string{output},
|
|
};
|
|
}
|
|
|
|
DerivedPath::Built DerivedPath::Built::parse(
|
|
const StoreDirConfig & store,
|
|
ref<const SingleDerivedPath> drv,
|
|
OutputNameView outputsS,
|
|
const ExperimentalFeatureSettings & xpSettings)
|
|
{
|
|
drvRequireExperiment(*drv, xpSettings);
|
|
return {
|
|
.drvPath = drv,
|
|
.outputs = OutputsSpec::parse(outputsS),
|
|
};
|
|
}
|
|
|
|
static SingleDerivedPath parseWithSingle(
|
|
const StoreDirConfig & store,
|
|
std::string_view s,
|
|
std::string_view separator,
|
|
const ExperimentalFeatureSettings & xpSettings)
|
|
{
|
|
size_t n = s.rfind(separator);
|
|
return n == s.npos
|
|
? (SingleDerivedPath) SingleDerivedPath::Opaque::parse(store, s)
|
|
: (SingleDerivedPath) SingleDerivedPath::Built::parse(
|
|
store,
|
|
make_ref<const SingleDerivedPath>(parseWithSingle(store, s.substr(0, n), separator, xpSettings)),
|
|
s.substr(n + 1),
|
|
xpSettings);
|
|
}
|
|
|
|
SingleDerivedPath SingleDerivedPath::parse(
|
|
const StoreDirConfig & store, std::string_view s, const ExperimentalFeatureSettings & xpSettings)
|
|
{
|
|
return parseWithSingle(store, s, "^", xpSettings);
|
|
}
|
|
|
|
SingleDerivedPath SingleDerivedPath::parseLegacy(
|
|
const StoreDirConfig & store, std::string_view s, const ExperimentalFeatureSettings & xpSettings)
|
|
{
|
|
return parseWithSingle(store, s, "!", xpSettings);
|
|
}
|
|
|
|
static DerivedPath parseWith(
|
|
const StoreDirConfig & store,
|
|
std::string_view s,
|
|
std::string_view separator,
|
|
const ExperimentalFeatureSettings & xpSettings)
|
|
{
|
|
size_t n = s.rfind(separator);
|
|
return n == s.npos
|
|
? (DerivedPath) DerivedPath::Opaque::parse(store, s)
|
|
: (DerivedPath) DerivedPath::Built::parse(
|
|
store,
|
|
make_ref<const SingleDerivedPath>(parseWithSingle(store, s.substr(0, n), separator, xpSettings)),
|
|
s.substr(n + 1),
|
|
xpSettings);
|
|
}
|
|
|
|
DerivedPath
|
|
DerivedPath::parse(const StoreDirConfig & store, std::string_view s, const ExperimentalFeatureSettings & xpSettings)
|
|
{
|
|
return parseWith(store, s, "^", xpSettings);
|
|
}
|
|
|
|
DerivedPath DerivedPath::parseLegacy(
|
|
const StoreDirConfig & store, std::string_view s, const ExperimentalFeatureSettings & xpSettings)
|
|
{
|
|
return parseWith(store, s, "!", xpSettings);
|
|
}
|
|
|
|
DerivedPath DerivedPath::fromSingle(const SingleDerivedPath & req)
|
|
{
|
|
return std::visit(
|
|
overloaded{
|
|
[&](const SingleDerivedPath::Opaque & o) -> DerivedPath { return o; },
|
|
[&](const SingleDerivedPath::Built & b) -> DerivedPath {
|
|
return DerivedPath::Built{
|
|
.drvPath = b.drvPath,
|
|
.outputs = OutputsSpec::Names{b.output},
|
|
};
|
|
},
|
|
},
|
|
req.raw());
|
|
}
|
|
|
|
const StorePath & SingleDerivedPath::Built::getBaseStorePath() const
|
|
{
|
|
return drvPath->getBaseStorePath();
|
|
}
|
|
|
|
const StorePath & DerivedPath::Built::getBaseStorePath() const
|
|
{
|
|
return drvPath->getBaseStorePath();
|
|
}
|
|
|
|
template<typename DP>
|
|
static inline const StorePath & getBaseStorePath_(const DP & derivedPath)
|
|
{
|
|
return std::visit(
|
|
overloaded{
|
|
[&](const typename DP::Built & bfd) -> auto & { return bfd.drvPath->getBaseStorePath(); },
|
|
[&](const typename DP::Opaque & bo) -> auto & { return bo.path; },
|
|
},
|
|
derivedPath.raw());
|
|
}
|
|
|
|
const StorePath & SingleDerivedPath::getBaseStorePath() const
|
|
{
|
|
return getBaseStorePath_(*this);
|
|
}
|
|
|
|
const StorePath & DerivedPath::getBaseStorePath() const
|
|
{
|
|
return getBaseStorePath_(*this);
|
|
}
|
|
|
|
} // namespace nix
|
|
|
|
namespace nlohmann {
|
|
|
|
void adl_serializer<SingleDerivedPath::Opaque>::to_json(json & json, const SingleDerivedPath::Opaque & o)
|
|
{
|
|
json = o.path;
|
|
}
|
|
|
|
SingleDerivedPath::Opaque adl_serializer<SingleDerivedPath::Opaque>::from_json(const json & json)
|
|
{
|
|
return SingleDerivedPath::Opaque{json};
|
|
}
|
|
|
|
void adl_serializer<SingleDerivedPath::Built>::to_json(json & json, const SingleDerivedPath::Built & sdpb)
|
|
{
|
|
json = {
|
|
{"drvPath", *sdpb.drvPath},
|
|
{"output", sdpb.output},
|
|
};
|
|
}
|
|
|
|
void adl_serializer<DerivedPath::Built>::to_json(json & json, const DerivedPath::Built & dbp)
|
|
{
|
|
json = {
|
|
{"drvPath", *dbp.drvPath},
|
|
{"outputs", dbp.outputs},
|
|
};
|
|
}
|
|
|
|
SingleDerivedPath::Built adl_serializer<SingleDerivedPath::Built>::from_json(const json & json0)
|
|
{
|
|
auto & json = getObject(json0);
|
|
return {
|
|
.drvPath = make_ref<SingleDerivedPath>(static_cast<SingleDerivedPath>(valueAt(json, "drvPath"))),
|
|
.output = getString(valueAt(json, "output")),
|
|
};
|
|
}
|
|
|
|
DerivedPath::Built adl_serializer<DerivedPath::Built>::from_json(const json & json0)
|
|
{
|
|
auto & json = getObject(json0);
|
|
return {
|
|
.drvPath = make_ref<SingleDerivedPath>(static_cast<SingleDerivedPath>(valueAt(json, "drvPath"))),
|
|
.outputs = adl_serializer<OutputsSpec>::from_json(valueAt(json, "outputs")),
|
|
};
|
|
}
|
|
|
|
void adl_serializer<SingleDerivedPath>::to_json(json & json, const SingleDerivedPath & sdp)
|
|
{
|
|
std::visit([&](const auto & buildable) { json = buildable; }, sdp.raw());
|
|
}
|
|
|
|
void adl_serializer<DerivedPath>::to_json(json & json, const DerivedPath & sdp)
|
|
{
|
|
std::visit([&](const auto & buildable) { json = buildable; }, sdp.raw());
|
|
}
|
|
|
|
SingleDerivedPath adl_serializer<SingleDerivedPath>::from_json(const json & json)
|
|
{
|
|
if (json.is_string())
|
|
return static_cast<SingleDerivedPath::Opaque>(json);
|
|
else
|
|
return static_cast<SingleDerivedPath::Built>(json);
|
|
}
|
|
|
|
DerivedPath adl_serializer<DerivedPath>::from_json(const json & json)
|
|
{
|
|
if (json.is_string())
|
|
return static_cast<DerivedPath::Opaque>(json);
|
|
else
|
|
return static_cast<DerivedPath::Built>(json);
|
|
}
|
|
|
|
} // namespace nlohmann
|