2021-10-25 13:53:01 +00:00
|
|
|
#include "experimental-features.hh"
|
2021-10-26 12:29:48 +00:00
|
|
|
#include "util.hh"
|
|
|
|
|
2021-10-25 13:53:01 +00:00
|
|
|
#include "nlohmann/json.hpp"
|
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
2023-04-02 22:11:16 +00:00
|
|
|
struct ExperimentalFeatureDetails
|
|
|
|
{
|
|
|
|
ExperimentalFeature tag;
|
|
|
|
std::string_view name;
|
|
|
|
std::string_view description;
|
2021-10-25 13:53:01 +00:00
|
|
|
};
|
|
|
|
|
2023-04-02 22:11:16 +00:00
|
|
|
constexpr std::array<ExperimentalFeatureDetails, 11> xpFeatureDetails = {{
|
|
|
|
{
|
|
|
|
.tag = Xp::CaDerivations,
|
|
|
|
.name = "ca-derivations",
|
|
|
|
.description = R"(
|
2023-04-04 23:16:10 +00:00
|
|
|
Allow derivations to be content-addressed in order to prevent rebuilds
|
2023-04-02 22:11:16 +00:00
|
|
|
when changes to the derivation do not result in changes to the
|
|
|
|
derivation's output. See
|
2023-04-04 23:16:10 +00:00
|
|
|
[__contentAddressed](@docroot@/language/advanced-attributes.md#adv-attr-__contentAddressed)
|
|
|
|
for details.
|
2023-04-02 22:11:16 +00:00
|
|
|
)",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.tag = Xp::ImpureDerivations,
|
|
|
|
.name = "impure-derivations",
|
|
|
|
.description = R"(
|
|
|
|
Allows derivations to produce non-fixed outputs by setting the `__impure`
|
|
|
|
derivation attribute to `true`. See [these release
|
|
|
|
notes](../release-notes/rl-2.8.md) for an example.
|
|
|
|
)",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.tag = Xp::Flakes,
|
|
|
|
.name = "flakes",
|
|
|
|
.description = R"(
|
2023-04-04 23:16:10 +00:00
|
|
|
Enable flakes. See the manual entry for
|
|
|
|
[`nix flake`](../command-ref/new-cli/nix3-flake.md) for details.
|
2023-04-02 22:11:16 +00:00
|
|
|
)",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.tag = Xp::NixCommand,
|
|
|
|
.name = "nix-command",
|
|
|
|
.description = R"(
|
2023-04-04 23:16:10 +00:00
|
|
|
Enable the new `nix` subcommands. See the manual on
|
|
|
|
[`nix`](@docroot@/command-ref/new-cli/nix.md) for details.
|
2023-04-02 22:11:16 +00:00
|
|
|
)",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.tag = Xp::RecursiveNix,
|
|
|
|
.name = "recursive-nix",
|
|
|
|
.description = R"(
|
|
|
|
Allow Nix derivations to call Nix in order to recursively build derivations.
|
|
|
|
See [this
|
|
|
|
commit](https://github.com/edolstra/nix/commit/1a27aa7d64ffe6fc36cfca4d82bdf51c4d8cf717)
|
|
|
|
for more info.
|
|
|
|
)",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.tag = Xp::NoUrlLiterals,
|
|
|
|
.name = "no-url-literals",
|
|
|
|
.description = R"(
|
|
|
|
Disallows unquoted URLs as part of the Nix language syntax. See [RFC
|
|
|
|
45](https://github.com/NixOS/rfcs/pull/45) for more info.
|
|
|
|
)",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.tag = Xp::FetchClosure,
|
|
|
|
.name = "fetch-closure",
|
|
|
|
.description = R"(
|
2023-04-04 23:16:10 +00:00
|
|
|
Enable the use of the [`fetchClosure`](@docroot@/language/builtins.md#builtins-fetchClosure) built-in function in the Nix language.
|
2023-04-02 22:11:16 +00:00
|
|
|
)",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.tag = Xp::ReplFlake,
|
|
|
|
.name = "repl-flake",
|
|
|
|
.description = R"(
|
2023-04-04 23:16:10 +00:00
|
|
|
Allow passing [installables](@docroot@/command-ref/new-cli/nix.md#installables) to `nix repl`, making its interface consistent with the other experimental commands.
|
2023-04-02 22:11:16 +00:00
|
|
|
)",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.tag = Xp::AutoAllocateUids,
|
|
|
|
.name = "auto-allocate-uids",
|
|
|
|
.description = R"(
|
|
|
|
Allows Nix to automatically pick UIDs for builds, rather than creating
|
2023-04-04 23:16:10 +00:00
|
|
|
`nixbld*` user accounts. See the [`auto-allocate-uids`](#conf-auto-allocate-uids) setting for details.
|
2023-04-02 22:11:16 +00:00
|
|
|
)",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.tag = Xp::Cgroups,
|
|
|
|
.name = "cgroups",
|
|
|
|
.description = R"(
|
|
|
|
Allows Nix to execute builds inside cgroups. See
|
2023-04-04 23:16:10 +00:00
|
|
|
the [`use-cgroups`](#conf-use-cgroups) setting for details.
|
2023-04-02 22:11:16 +00:00
|
|
|
)",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.tag = Xp::DiscardReferences,
|
|
|
|
.name = "discard-references",
|
|
|
|
.description = R"(
|
2023-04-04 23:16:10 +00:00
|
|
|
Allow the use of the [`unsafeDiscardReferences`](@docroot@/language/advanced-attributes.html#adv-attr-unsafeDiscardReferences) attribute in derivations
|
|
|
|
that use [structured attributes](@docroot@/language/advanced-attributes.html#adv-attr-structuredAttrs). This disables scanning of outputs for
|
2023-04-02 22:11:16 +00:00
|
|
|
runtime dependencies.
|
|
|
|
)",
|
|
|
|
},
|
|
|
|
}};
|
|
|
|
|
|
|
|
static_assert(
|
|
|
|
[]() constexpr {
|
|
|
|
for (auto [index, feature] : enumerate(xpFeatureDetails))
|
|
|
|
if (index != (size_t)feature.tag)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}(),
|
|
|
|
"array order does not match enum tag order");
|
|
|
|
|
2021-10-25 13:53:01 +00:00
|
|
|
const std::optional<ExperimentalFeature> parseExperimentalFeature(const std::string_view & name)
|
|
|
|
{
|
|
|
|
using ReverseXpMap = std::map<std::string_view, ExperimentalFeature>;
|
2021-10-26 12:29:48 +00:00
|
|
|
|
2023-04-02 22:11:16 +00:00
|
|
|
static std::unique_ptr<ReverseXpMap> reverseXpMap = [](){
|
2021-10-26 12:29:48 +00:00
|
|
|
auto reverseXpMap = std::make_unique<ReverseXpMap>();
|
2023-04-02 22:11:16 +00:00
|
|
|
for (auto & xpFeature : xpFeatureDetails)
|
|
|
|
(*reverseXpMap)[xpFeature.name] = xpFeature.tag;
|
2021-10-26 12:29:48 +00:00
|
|
|
return reverseXpMap;
|
|
|
|
}();
|
2021-10-25 13:53:01 +00:00
|
|
|
|
2021-10-26 12:29:48 +00:00
|
|
|
if (auto feature = get(*reverseXpMap, name))
|
|
|
|
return *feature;
|
|
|
|
else
|
2021-10-25 13:53:01 +00:00
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2023-04-02 22:11:16 +00:00
|
|
|
std::string_view showExperimentalFeature(const ExperimentalFeature tag)
|
2021-10-25 13:53:01 +00:00
|
|
|
{
|
2023-04-02 22:11:16 +00:00
|
|
|
assert((size_t)tag < xpFeatureDetails.size());
|
|
|
|
return xpFeatureDetails[(size_t)tag].name;
|
2023-03-28 00:12:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string getExperimentalFeaturesList() {
|
|
|
|
std::string experimentalFeaturesList = R"(
|
2023-04-04 23:16:10 +00:00
|
|
|
Experimental features that can be enabled.
|
|
|
|
|
|
|
|
Example:
|
|
|
|
|
|
|
|
```
|
|
|
|
experimental-features = nix-command flakes
|
|
|
|
|
|
|
|
Experimental features available:
|
2023-03-28 00:12:49 +00:00
|
|
|
|
|
|
|
)";
|
|
|
|
|
2023-04-02 22:11:16 +00:00
|
|
|
for (auto & xpFeature : xpFeatureDetails) {
|
|
|
|
experimentalFeaturesList += std::string {}
|
|
|
|
/* length of this first string must be 12, matching the indent of
|
|
|
|
the descriptions in the xpFeatureDetails literal. FIXME compute
|
|
|
|
markdown in a less hacky way. */
|
|
|
|
+ " - "
|
|
|
|
+ "`" + xpFeature.name + "`"
|
|
|
|
+ "\n"
|
|
|
|
+ xpFeature.description
|
|
|
|
+ "\n\n";
|
2023-03-28 00:12:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return experimentalFeaturesList;
|
2021-10-25 13:53:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::set<ExperimentalFeature> parseFeatures(const std::set<std::string> & rawFeatures)
|
|
|
|
{
|
|
|
|
std::set<ExperimentalFeature> res;
|
2023-04-02 22:11:16 +00:00
|
|
|
for (auto & rawFeature : rawFeatures)
|
2021-10-25 13:53:01 +00:00
|
|
|
if (auto feature = parseExperimentalFeature(rawFeature))
|
|
|
|
res.insert(*feature);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
MissingExperimentalFeature::MissingExperimentalFeature(ExperimentalFeature feature)
|
|
|
|
: Error("experimental Nix feature '%1%' is disabled; use '--extra-experimental-features %1%' to override", showExperimentalFeature(feature))
|
|
|
|
, missingFeature(feature)
|
2021-10-26 12:29:48 +00:00
|
|
|
{}
|
2021-10-25 13:53:01 +00:00
|
|
|
|
|
|
|
std::ostream & operator <<(std::ostream & str, const ExperimentalFeature & feature)
|
|
|
|
{
|
|
|
|
return str << showExperimentalFeature(feature);
|
|
|
|
}
|
|
|
|
|
2022-05-03 12:37:28 +00:00
|
|
|
void to_json(nlohmann::json & j, const ExperimentalFeature & feature)
|
|
|
|
{
|
2022-04-20 13:41:01 +00:00
|
|
|
j = showExperimentalFeature(feature);
|
|
|
|
}
|
|
|
|
|
2022-05-03 12:37:28 +00:00
|
|
|
void from_json(const nlohmann::json & j, ExperimentalFeature & feature)
|
|
|
|
{
|
2022-04-20 13:41:01 +00:00
|
|
|
const std::string input = j;
|
|
|
|
const auto parsed = parseExperimentalFeature(input);
|
|
|
|
|
|
|
|
if (parsed.has_value())
|
|
|
|
feature = *parsed;
|
|
|
|
else
|
|
|
|
throw Error("Unknown experimental feature '%s' in JSON input", input);
|
|
|
|
}
|
|
|
|
|
2021-10-25 13:53:01 +00:00
|
|
|
}
|