2023-01-29 18:52:38 +00:00
|
|
|
#include <regex>
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <rapidcheck/gtest.h>
|
|
|
|
|
|
|
|
#include "tests/derived-path.hh"
|
|
|
|
#include "tests/libstore.hh"
|
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
class DerivedPathTest : public LibStoreTest
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
Make the Derived Path family of types inductive for dynamic derivations
We want to be able to write down `foo.drv^bar.drv^baz`:
`foo.drv^bar.drv` is the dynamic derivation (since it is itself a
derivation output, `bar.drv` from `foo.drv`).
To that end, we create `Single{Derivation,BuiltPath}` types, that are
very similar except instead of having multiple outputs (in a set or
map), they have a single one. This is for everything to the left of the
rightmost `^`.
`NixStringContextElem` has an analogous change, and now can reuse
`SingleDerivedPath` at the top level. In fact, if we ever get rid of
`DrvDeep`, `NixStringContextElem` could be replaced with
`SingleDerivedPath` entirely!
Important note: some JSON formats have changed.
We already can *produce* dynamic derivations, but we can't refer to them
directly. Today, we can merely express building or example at the top
imperatively over time by building `foo.drv^bar.drv`, and then with a
second nix invocation doing `<result-from-first>^baz`, but this is not
declarative. The ethos of Nix of being able to write down the full plan
everything you want to do, and then execute than plan with a single
command, and for that we need the new inductive form of these types.
Co-authored-by: Robert Hensing <roberth@users.noreply.github.com>
Co-authored-by: Valentin Gagarin <valentin.gagarin@tweag.io>
2023-01-15 22:39:04 +00:00
|
|
|
/**
|
|
|
|
* Round trip (string <-> data structure) test for
|
|
|
|
* `DerivedPath::Opaque`.
|
|
|
|
*/
|
|
|
|
TEST_F(DerivedPathTest, opaque) {
|
|
|
|
std::string_view opaque = "/nix/store/g1w7hy3qg1w7hy3qg1w7hy3qg1w7hy3q-x";
|
|
|
|
auto elem = DerivedPath::parse(*store, opaque);
|
|
|
|
auto * p = std::get_if<DerivedPath::Opaque>(&elem);
|
|
|
|
ASSERT_TRUE(p);
|
|
|
|
ASSERT_EQ(p->path, store->parseStorePath(opaque));
|
|
|
|
ASSERT_EQ(elem.to_string(*store), opaque);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Round trip (string <-> data structure) test for a simpler
|
|
|
|
* `DerivedPath::Built`.
|
|
|
|
*/
|
|
|
|
TEST_F(DerivedPathTest, built_opaque) {
|
|
|
|
std::string_view built = "/nix/store/g1w7hy3qg1w7hy3qg1w7hy3qg1w7hy3q-x.drv^bar,foo";
|
|
|
|
auto elem = DerivedPath::parse(*store, built);
|
|
|
|
auto * p = std::get_if<DerivedPath::Built>(&elem);
|
|
|
|
ASSERT_TRUE(p);
|
|
|
|
ASSERT_EQ(p->outputs, ((OutputsSpec) OutputsSpec::Names { "foo", "bar" }));
|
|
|
|
ASSERT_EQ(*p->drvPath, ((SingleDerivedPath) SingleDerivedPath::Opaque {
|
|
|
|
.path = store->parseStorePath(built.substr(0, 49)),
|
|
|
|
}));
|
|
|
|
ASSERT_EQ(elem.to_string(*store), built);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Round trip (string <-> data structure) test for a more complex,
|
|
|
|
* inductive `DerivedPath::Built`.
|
|
|
|
*/
|
|
|
|
TEST_F(DerivedPathTest, built_built) {
|
|
|
|
/**
|
|
|
|
* We set these in tests rather than the regular globals so we don't have
|
|
|
|
* to worry about race conditions if the tests run concurrently.
|
|
|
|
*/
|
|
|
|
ExperimentalFeatureSettings mockXpSettings;
|
|
|
|
mockXpSettings.set("experimental-features", "dynamic-derivations ca-derivations");
|
|
|
|
|
|
|
|
std::string_view built = "/nix/store/g1w7hy3qg1w7hy3qg1w7hy3qg1w7hy3q-x.drv^foo^bar,baz";
|
|
|
|
auto elem = DerivedPath::parse(*store, built, mockXpSettings);
|
|
|
|
auto * p = std::get_if<DerivedPath::Built>(&elem);
|
|
|
|
ASSERT_TRUE(p);
|
|
|
|
ASSERT_EQ(p->outputs, ((OutputsSpec) OutputsSpec::Names { "bar", "baz" }));
|
|
|
|
auto * drvPath = std::get_if<SingleDerivedPath::Built>(&*p->drvPath);
|
|
|
|
ASSERT_TRUE(drvPath);
|
|
|
|
ASSERT_EQ(drvPath->output, "foo");
|
|
|
|
ASSERT_EQ(*drvPath->drvPath, ((SingleDerivedPath) SingleDerivedPath::Opaque {
|
|
|
|
.path = store->parseStorePath(built.substr(0, 49)),
|
|
|
|
}));
|
|
|
|
ASSERT_EQ(elem.to_string(*store), built);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Without the right experimental features enabled, we cannot parse a
|
|
|
|
* complex inductive derived path.
|
|
|
|
*/
|
|
|
|
TEST_F(DerivedPathTest, built_built_xp) {
|
|
|
|
ASSERT_THROW(
|
|
|
|
DerivedPath::parse(*store, "/nix/store/g1w7hy3qg1w7hy3qg1w7hy3qg1w7hy3q-x.drv^foo^bar,baz"),
|
|
|
|
MissingExperimentalFeature);
|
2023-01-29 18:52:38 +00:00
|
|
|
}
|
|
|
|
|
2024-05-17 00:02:48 +00:00
|
|
|
/**
|
|
|
|
* Built paths with a non-derivation base should fail parsing.
|
|
|
|
*/
|
|
|
|
TEST_F(DerivedPathTest, non_derivation_base) {
|
|
|
|
ASSERT_THROW(
|
|
|
|
DerivedPath::parse(*store, "/nix/store/g1w7hy3qg1w7hy3qg1w7hy3qg1w7hy3q-x^foo"),
|
|
|
|
InvalidPath);
|
|
|
|
}
|
|
|
|
|
2023-09-19 14:04:00 +00:00
|
|
|
#ifndef COVERAGE
|
|
|
|
|
2023-04-15 00:45:11 +00:00
|
|
|
RC_GTEST_FIXTURE_PROP(
|
|
|
|
DerivedPathTest,
|
|
|
|
prop_legacy_round_rip,
|
|
|
|
(const DerivedPath & o))
|
|
|
|
{
|
|
|
|
RC_ASSERT(o == DerivedPath::parseLegacy(*store, o.to_string_legacy(*store)));
|
|
|
|
}
|
|
|
|
|
2023-01-29 18:52:38 +00:00
|
|
|
RC_GTEST_FIXTURE_PROP(
|
|
|
|
DerivedPathTest,
|
|
|
|
prop_round_rip,
|
|
|
|
(const DerivedPath & o))
|
|
|
|
{
|
|
|
|
RC_ASSERT(o == DerivedPath::parse(*store, o.to_string(*store)));
|
|
|
|
}
|
|
|
|
|
2023-09-19 14:04:00 +00:00
|
|
|
#endif
|
|
|
|
|
2023-01-29 18:52:38 +00:00
|
|
|
}
|