mirror of
https://github.com/NixOS/nix.git
synced 2025-12-24 01:41:08 +01:00
The previous commit hacked it into the output of `nix path-info --json`, this cleans that up my making it an actual field of that data type, and part of the canonical JSON serializers for it (and `ValidPathInfo` and `NarInfo`). Beyond cleaning up the JSON code, this also opens the doors to things like: - Binary caches that contain store objects that don't all belong in the same store directory - Relocatable store objects which carefully don't mention any store directory by absolute path, and instead use relative paths for anything. (#9549)
152 lines
6.4 KiB
C++
152 lines
6.4 KiB
C++
#include <nlohmann/json.hpp>
|
|
#include <gtest/gtest.h>
|
|
|
|
#include "nix/store/path-info.hh"
|
|
|
|
#include "nix/util/tests/characterization.hh"
|
|
#include "nix/store/tests/libstore.hh"
|
|
|
|
namespace nix {
|
|
|
|
using nlohmann::json;
|
|
|
|
class PathInfoTestV1 : public CharacterizationTest, public LibStoreTest
|
|
{
|
|
std::filesystem::path unitTestData = getUnitTestData() / "path-info" / "json-1";
|
|
|
|
std::filesystem::path goldenMaster(PathView testStem) const override
|
|
{
|
|
return unitTestData / (testStem + ".json");
|
|
}
|
|
};
|
|
|
|
class PathInfoTestV2 : public CharacterizationTest, public LibStoreTest
|
|
{
|
|
std::filesystem::path unitTestData = getUnitTestData() / "path-info" / "json-2";
|
|
|
|
std::filesystem::path goldenMaster(PathView testStem) const override
|
|
{
|
|
return unitTestData / (testStem + ".json");
|
|
}
|
|
};
|
|
|
|
static UnkeyedValidPathInfo makeEmpty()
|
|
{
|
|
return {
|
|
"/nix/store",
|
|
Hash::parseSRI("sha256-FePFYIlMuycIXPZbWi7LGEiMmZSX9FMbaQenWBzm1Sc="),
|
|
};
|
|
}
|
|
|
|
static ValidPathInfo makeFullKeyed(const Store & store, bool includeImpureInfo)
|
|
{
|
|
auto info = ValidPathInfo::makeFromCA(
|
|
store,
|
|
"foo",
|
|
FixedOutputInfo{
|
|
.method = FileIngestionMethod::NixArchive,
|
|
.hash = hashString(HashAlgorithm::SHA256, "(...)"),
|
|
|
|
.references =
|
|
{
|
|
.others =
|
|
{
|
|
StorePath{
|
|
"g1w7hy3qg1w7hy3qg1w7hy3qg1w7hy3q-bar",
|
|
},
|
|
},
|
|
.self = true,
|
|
},
|
|
},
|
|
Hash::parseSRI("sha256-FePFYIlMuycIXPZbWi7LGEiMmZSX9FMbaQenWBzm1Sc="));
|
|
info.narSize = 34878;
|
|
if (includeImpureInfo) {
|
|
info.deriver = StorePath{
|
|
"g1w7hy3qg1w7hy3qg1w7hy3qg1w7hy3q-bar.drv",
|
|
};
|
|
info.registrationTime = 23423;
|
|
info.ultimate = true;
|
|
info.sigs = {"asdf", "qwer"};
|
|
}
|
|
return info;
|
|
}
|
|
|
|
static UnkeyedValidPathInfo makeFull(const Store & store, bool includeImpureInfo)
|
|
{
|
|
return makeFullKeyed(store, includeImpureInfo);
|
|
}
|
|
|
|
#define JSON_READ_TEST_V1(STEM, OBJ) \
|
|
TEST_F(PathInfoTestV1, PathInfo_##STEM##_from_json) \
|
|
{ \
|
|
readTest(#STEM, [&](const auto & encoded_) { \
|
|
auto encoded = json::parse(encoded_); \
|
|
UnkeyedValidPathInfo got = UnkeyedValidPathInfo::fromJSON(&*store, encoded); \
|
|
auto expected = OBJ; \
|
|
ASSERT_EQ(got, expected); \
|
|
}); \
|
|
}
|
|
|
|
#define JSON_WRITE_TEST_V1(STEM, OBJ, PURE) \
|
|
TEST_F(PathInfoTestV1, PathInfo_##STEM##_to_json) \
|
|
{ \
|
|
writeTest( \
|
|
#STEM, \
|
|
[&]() -> json { return OBJ.toJSON(&*store, PURE, PathInfoJsonFormat::V1); }, \
|
|
[](const auto & file) { return json::parse(readFile(file)); }, \
|
|
[](const auto & file, const auto & got) { return writeFile(file, got.dump(2) + "\n"); }); \
|
|
}
|
|
|
|
#define JSON_TEST_V1(STEM, OBJ, PURE) \
|
|
JSON_READ_TEST_V1(STEM, OBJ) \
|
|
JSON_WRITE_TEST_V1(STEM, OBJ, PURE)
|
|
|
|
#define JSON_READ_TEST_V2(STEM, OBJ) \
|
|
TEST_F(PathInfoTestV2, PathInfo_##STEM##_from_json) \
|
|
{ \
|
|
readTest(#STEM, [&](const auto & encoded_) { \
|
|
auto encoded = json::parse(encoded_); \
|
|
UnkeyedValidPathInfo got = UnkeyedValidPathInfo::fromJSON(nullptr, encoded); \
|
|
auto expected = OBJ; \
|
|
ASSERT_EQ(got, expected); \
|
|
}); \
|
|
}
|
|
|
|
#define JSON_WRITE_TEST_V2(STEM, OBJ, PURE) \
|
|
TEST_F(PathInfoTestV2, PathInfo_##STEM##_to_json) \
|
|
{ \
|
|
writeTest( \
|
|
#STEM, \
|
|
[&]() -> json { return OBJ.toJSON(nullptr, PURE, PathInfoJsonFormat::V2); }, \
|
|
[](const auto & file) { return json::parse(readFile(file)); }, \
|
|
[](const auto & file, const auto & got) { return writeFile(file, got.dump(2) + "\n"); }); \
|
|
}
|
|
|
|
#define JSON_TEST_V2(STEM, OBJ, PURE) \
|
|
JSON_READ_TEST_V2(STEM, OBJ) \
|
|
JSON_WRITE_TEST_V2(STEM, OBJ, PURE)
|
|
|
|
JSON_TEST_V1(empty_pure, makeEmpty(), false)
|
|
JSON_TEST_V1(empty_impure, makeEmpty(), true)
|
|
JSON_TEST_V1(pure, makeFull(*store, false), false)
|
|
JSON_TEST_V1(impure, makeFull(*store, true), true)
|
|
|
|
// Test that JSON without explicit version field parses as V1
|
|
JSON_READ_TEST_V1(pure_noversion, makeFull(*store, false))
|
|
|
|
JSON_TEST_V2(empty_pure, makeEmpty(), false)
|
|
JSON_TEST_V2(empty_impure, makeEmpty(), true)
|
|
JSON_TEST_V2(pure, makeFull(*store, false), false)
|
|
JSON_TEST_V2(impure, makeFull(*store, true), true)
|
|
|
|
TEST_F(PathInfoTestV2, PathInfo_full_shortRefs)
|
|
{
|
|
ValidPathInfo it = makeFullKeyed(*store, true);
|
|
// it.references = unkeyed.references;
|
|
auto refs = it.shortRefs();
|
|
ASSERT_EQ(refs.size(), 2u);
|
|
ASSERT_EQ(*refs.begin(), "g1w7hy3qg1w7hy3qg1w7hy3qg1w7hy3q-bar");
|
|
ASSERT_EQ(*++refs.begin(), "n5wkd9frr45pa74if5gpz9j7mifg27fh-foo");
|
|
}
|
|
|
|
} // namespace nix
|