mirror of
https://github.com/NixOS/nix.git
synced 2025-11-09 12:06:01 +01:00
Merge pull request #13907 from obsidiansystems/store-building-unit-test
C API for building
This commit is contained in:
commit
0b19468368
13 changed files with 270 additions and 47 deletions
|
|
@ -1,43 +1,14 @@
|
||||||
#include <gtest/gtest.h>
|
#include <gtest/gtest.h>
|
||||||
#include <cstdlib>
|
|
||||||
#include "nix/store/globals.hh"
|
#include "nix/store/tests/test-main.hh"
|
||||||
#include "nix/util/logging.hh"
|
|
||||||
|
|
||||||
using namespace nix;
|
using namespace nix;
|
||||||
|
|
||||||
int main(int argc, char ** argv)
|
int main(int argc, char ** argv)
|
||||||
{
|
{
|
||||||
if (argc > 1 && std::string_view(argv[1]) == "__build-remote") {
|
auto res = testMainForBuidingPre(argc, argv);
|
||||||
printError("test-build-remote: not supported in libexpr unit tests");
|
if (!res)
|
||||||
return 1;
|
return res;
|
||||||
}
|
|
||||||
|
|
||||||
// Disable build hook. We won't be testing remote builds in these unit tests. If we do, fix the above build hook.
|
|
||||||
settings.buildHook = {};
|
|
||||||
|
|
||||||
#ifdef __linux__ // should match the conditional around sandboxBuildDir declaration.
|
|
||||||
|
|
||||||
// When building and testing nix within the host's Nix sandbox, our store dir will be located in the host's
|
|
||||||
// sandboxBuildDir, e.g.: Host
|
|
||||||
// storeDir = /nix/store
|
|
||||||
// sandboxBuildDir = /build
|
|
||||||
// This process
|
|
||||||
// storeDir = /build/foo/bar/store
|
|
||||||
// sandboxBuildDir = /build
|
|
||||||
// However, we have a rule that the store dir must not be inside the storeDir, so we need to pick a different
|
|
||||||
// sandboxBuildDir.
|
|
||||||
settings.sandboxBuildDir = "/test-build-dir-instead-of-usual-build-dir";
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef __APPLE__
|
|
||||||
// Avoid this error, when already running in a sandbox:
|
|
||||||
// sandbox-exec: sandbox_apply: Operation not permitted
|
|
||||||
settings.sandboxMode = smDisabled;
|
|
||||||
setEnv("_NIX_TEST_NO_SANDBOX", "1");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// For pipe operator tests in trivial.cc
|
|
||||||
experimentalFeatureSettings.set("experimental-features", "pipe-operators");
|
|
||||||
|
|
||||||
::testing::InitGoogleTest(&argc, argv);
|
::testing::InitGoogleTest(&argc, argv);
|
||||||
return RUN_ALL_TESTS();
|
return RUN_ALL_TESTS();
|
||||||
|
|
|
||||||
|
|
@ -166,11 +166,44 @@ void nix_store_path_free(StorePath * sp)
|
||||||
delete sp;
|
delete sp;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void nix_derivation_free(nix_derivation * drv)
|
||||||
|
{
|
||||||
|
delete drv;
|
||||||
|
}
|
||||||
|
|
||||||
StorePath * nix_store_path_clone(const StorePath * p)
|
StorePath * nix_store_path_clone(const StorePath * p)
|
||||||
{
|
{
|
||||||
return new StorePath{p->path};
|
return new StorePath{p->path};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nix_derivation * nix_derivation_from_json(nix_c_context * context, Store * store, const char * json)
|
||||||
|
{
|
||||||
|
if (context)
|
||||||
|
context->last_err_code = NIX_OK;
|
||||||
|
try {
|
||||||
|
auto drv = nix::Derivation::fromJSON(*store->ptr, nlohmann::json::parse(json));
|
||||||
|
|
||||||
|
auto drvPath = nix::writeDerivation(*store->ptr, drv, nix::NoRepair, /* read only */ true);
|
||||||
|
|
||||||
|
drv.checkInvariants(*store->ptr, drvPath);
|
||||||
|
|
||||||
|
return new nix_derivation{drv};
|
||||||
|
}
|
||||||
|
NIXC_CATCH_ERRS_NULL
|
||||||
|
}
|
||||||
|
|
||||||
|
StorePath * nix_add_derivation(nix_c_context * context, Store * store, nix_derivation * derivation)
|
||||||
|
{
|
||||||
|
if (context)
|
||||||
|
context->last_err_code = NIX_OK;
|
||||||
|
try {
|
||||||
|
auto ret = nix::writeDerivation(*store->ptr, derivation->drv, nix::NoRepair);
|
||||||
|
|
||||||
|
return new StorePath{ret};
|
||||||
|
}
|
||||||
|
NIXC_CATCH_ERRS_NULL
|
||||||
|
}
|
||||||
|
|
||||||
nix_err nix_store_copy_closure(nix_c_context * context, Store * srcStore, Store * dstStore, StorePath * path)
|
nix_err nix_store_copy_closure(nix_c_context * context, Store * srcStore, Store * dstStore, StorePath * path)
|
||||||
{
|
{
|
||||||
if (context)
|
if (context)
|
||||||
|
|
|
||||||
|
|
@ -23,6 +23,8 @@ extern "C" {
|
||||||
typedef struct Store Store;
|
typedef struct Store Store;
|
||||||
/** @brief Nix store path */
|
/** @brief Nix store path */
|
||||||
typedef struct StorePath StorePath;
|
typedef struct StorePath StorePath;
|
||||||
|
/** @brief Nix Derivation */
|
||||||
|
typedef struct nix_derivation nix_derivation;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Initializes the Nix store library
|
* @brief Initializes the Nix store library
|
||||||
|
|
@ -207,6 +209,32 @@ nix_err nix_store_realise(
|
||||||
nix_err
|
nix_err
|
||||||
nix_store_get_version(nix_c_context * context, Store * store, nix_get_string_callback callback, void * user_data);
|
nix_store_get_version(nix_c_context * context, Store * store, nix_get_string_callback callback, void * user_data);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Create a `nix_derivation` from a JSON representation of that derivation.
|
||||||
|
*
|
||||||
|
* @param[out] context Optional, stores error information.
|
||||||
|
* @param[in] store nix store reference.
|
||||||
|
* @param[in] json JSON of the derivation as a string.
|
||||||
|
*/
|
||||||
|
nix_derivation * nix_derivation_from_json(nix_c_context * context, Store * store, const char * json);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Add the given `nix_derivation` to the given store
|
||||||
|
*
|
||||||
|
* @param[out] context Optional, stores error information.
|
||||||
|
* @param[in] store nix store reference. The derivation will be inserted here.
|
||||||
|
* @param[in] derivation nix_derivation to insert into the given store.
|
||||||
|
*/
|
||||||
|
StorePath * nix_add_derivation(nix_c_context * context, Store * store, nix_derivation * derivation);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Deallocate a `nix_derivation`
|
||||||
|
*
|
||||||
|
* Does not fail.
|
||||||
|
* @param[in] drv the derivation to free
|
||||||
|
*/
|
||||||
|
void nix_derivation_free(nix_derivation * drv);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Copy the closure of `path` from `srcStore` to `dstStore`.
|
* @brief Copy the closure of `path` from `srcStore` to `dstStore`.
|
||||||
*
|
*
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,7 @@
|
||||||
#ifndef NIX_API_STORE_INTERNAL_H
|
#ifndef NIX_API_STORE_INTERNAL_H
|
||||||
#define NIX_API_STORE_INTERNAL_H
|
#define NIX_API_STORE_INTERNAL_H
|
||||||
#include "nix/store/store-api.hh"
|
#include "nix/store/store-api.hh"
|
||||||
|
#include "nix/store/derivations.hh"
|
||||||
|
|
||||||
extern "C" {
|
extern "C" {
|
||||||
|
|
||||||
|
|
@ -14,6 +15,11 @@ struct StorePath
|
||||||
nix::StorePath path;
|
nix::StorePath path;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct nix_derivation
|
||||||
|
{
|
||||||
|
nix::Derivation drv;
|
||||||
|
};
|
||||||
|
|
||||||
} // extern "C"
|
} // extern "C"
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -9,4 +9,5 @@ headers = files(
|
||||||
'outputs-spec.hh',
|
'outputs-spec.hh',
|
||||||
'path.hh',
|
'path.hh',
|
||||||
'protocol.hh',
|
'protocol.hh',
|
||||||
|
'test-main.hh',
|
||||||
)
|
)
|
||||||
|
|
|
||||||
|
|
@ -12,33 +12,32 @@
|
||||||
#include <gtest/gtest.h>
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
namespace nixC {
|
namespace nixC {
|
||||||
class nix_api_store_test : public nix_api_util_context
|
|
||||||
|
class nix_api_store_test_base : public nix_api_util_context
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
nix_api_store_test()
|
nix_api_store_test_base()
|
||||||
{
|
{
|
||||||
nix_libstore_init(ctx);
|
nix_libstore_init(ctx);
|
||||||
init_local_store();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
~nix_api_store_test() override
|
~nix_api_store_test_base() override
|
||||||
{
|
{
|
||||||
nix_store_free(store);
|
if (exists(std::filesystem::path{nixDir})) {
|
||||||
|
for (auto & path : std::filesystem::recursive_directory_iterator(nixDir)) {
|
||||||
for (auto & path : std::filesystem::recursive_directory_iterator(nixDir)) {
|
std::filesystem::permissions(path, std::filesystem::perms::owner_all);
|
||||||
std::filesystem::permissions(path, std::filesystem::perms::owner_all);
|
}
|
||||||
|
std::filesystem::remove_all(nixDir);
|
||||||
}
|
}
|
||||||
std::filesystem::remove_all(nixDir);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Store * store;
|
|
||||||
std::string nixDir;
|
std::string nixDir;
|
||||||
std::string nixStoreDir;
|
std::string nixStoreDir;
|
||||||
std::string nixStateDir;
|
std::string nixStateDir;
|
||||||
std::string nixLogDir;
|
std::string nixLogDir;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void init_local_store()
|
Store * open_local_store()
|
||||||
{
|
{
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
// no `mkdtemp` with MinGW
|
// no `mkdtemp` with MinGW
|
||||||
|
|
@ -66,11 +65,37 @@ protected:
|
||||||
|
|
||||||
const char ** params[] = {p1, p2, p3, nullptr};
|
const char ** params[] = {p1, p2, p3, nullptr};
|
||||||
|
|
||||||
store = nix_store_open(ctx, "local", params);
|
auto * store = nix_store_open(ctx, "local", params);
|
||||||
if (!store) {
|
if (!store) {
|
||||||
std::string errMsg = nix_err_msg(nullptr, ctx, nullptr);
|
std::string errMsg = nix_err_msg(nullptr, ctx, nullptr);
|
||||||
ASSERT_NE(store, nullptr) << "Could not open store: " << errMsg;
|
EXPECT_NE(store, nullptr) << "Could not open store: " << errMsg;
|
||||||
|
assert(store);
|
||||||
};
|
};
|
||||||
|
return store;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class nix_api_store_test : public nix_api_store_test_base
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
nix_api_store_test()
|
||||||
|
: nix_api_store_test_base{}
|
||||||
|
{
|
||||||
|
init_local_store();
|
||||||
|
};
|
||||||
|
|
||||||
|
~nix_api_store_test() override
|
||||||
|
{
|
||||||
|
nix_store_free(store);
|
||||||
|
}
|
||||||
|
|
||||||
|
Store * store;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void init_local_store()
|
||||||
|
{
|
||||||
|
store = open_local_store();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
} // namespace nixC
|
} // namespace nixC
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,13 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
///@file
|
||||||
|
|
||||||
|
namespace nix {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Call this for a GTest test suite that will including performing Nix
|
||||||
|
* builds, before running tests.
|
||||||
|
*/
|
||||||
|
int testMainForBuidingPre(int argc, char ** argv);
|
||||||
|
|
||||||
|
} // namespace nix
|
||||||
|
|
@ -34,6 +34,7 @@ sources = files(
|
||||||
'derived-path.cc',
|
'derived-path.cc',
|
||||||
'outputs-spec.cc',
|
'outputs-spec.cc',
|
||||||
'path.cc',
|
'path.cc',
|
||||||
|
'test-main.cc',
|
||||||
)
|
)
|
||||||
|
|
||||||
subdir('include/nix/store/tests')
|
subdir('include/nix/store/tests')
|
||||||
|
|
|
||||||
47
src/libstore-test-support/test-main.cc
Normal file
47
src/libstore-test-support/test-main.cc
Normal file
|
|
@ -0,0 +1,47 @@
|
||||||
|
#include <cstdlib>
|
||||||
|
|
||||||
|
#include "nix/store/globals.hh"
|
||||||
|
#include "nix/util/logging.hh"
|
||||||
|
|
||||||
|
#include "nix/store/tests/test-main.hh"
|
||||||
|
|
||||||
|
namespace nix {
|
||||||
|
|
||||||
|
int testMainForBuidingPre(int argc, char ** argv)
|
||||||
|
{
|
||||||
|
if (argc > 1 && std::string_view(argv[1]) == "__build-remote") {
|
||||||
|
printError("test-build-remote: not supported in libexpr unit tests");
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Disable build hook. We won't be testing remote builds in these unit tests. If we do, fix the above build hook.
|
||||||
|
settings.buildHook = {};
|
||||||
|
|
||||||
|
// No substituters, unless a test specifically requests.
|
||||||
|
settings.substituters = {};
|
||||||
|
|
||||||
|
#ifdef __linux__ // should match the conditional around sandboxBuildDir declaration.
|
||||||
|
|
||||||
|
// When building and testing nix within the host's Nix sandbox, our store dir will be located in the host's
|
||||||
|
// sandboxBuildDir, e.g.: Host
|
||||||
|
// storeDir = /nix/store
|
||||||
|
// sandboxBuildDir = /build
|
||||||
|
// This process
|
||||||
|
// storeDir = /build/foo/bar/store
|
||||||
|
// sandboxBuildDir = /build
|
||||||
|
// However, we have a rule that the store dir must not be inside the storeDir, so we need to pick a different
|
||||||
|
// sandboxBuildDir.
|
||||||
|
settings.sandboxBuildDir = "/test-build-dir-instead-of-usual-build-dir";
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __APPLE__
|
||||||
|
// Avoid this error, when already running in a sandbox:
|
||||||
|
// sandbox-exec: sandbox_apply: Operation not permitted
|
||||||
|
settings.sandboxMode = smDisabled;
|
||||||
|
setEnv("_NIX_TEST_NO_SANDBOX", "1");
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return EXIT_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace nix
|
||||||
23
src/libstore-tests/data/derivation/ca/self-contained.json
Normal file
23
src/libstore-tests/data/derivation/ca/self-contained.json
Normal file
|
|
@ -0,0 +1,23 @@
|
||||||
|
{
|
||||||
|
"args": [
|
||||||
|
"-c",
|
||||||
|
"echo $name foo > $out"
|
||||||
|
],
|
||||||
|
"builder": "/bin/sh",
|
||||||
|
"env": {
|
||||||
|
"builder": "/bin/sh",
|
||||||
|
"name": "myname",
|
||||||
|
"out": "/1rz4g4znpzjwh1xymhjpm42vipw92pr73vdgl6xs1hycac8kf2n9",
|
||||||
|
"system": "x86_64-linux"
|
||||||
|
},
|
||||||
|
"inputDrvs": {},
|
||||||
|
"inputSrcs": [],
|
||||||
|
"name": "myname",
|
||||||
|
"outputs": {
|
||||||
|
"out": {
|
||||||
|
"hashAlgo": "sha256",
|
||||||
|
"method": "nar"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"system": "x86_64-linux"
|
||||||
|
}
|
||||||
15
src/libstore-tests/main.cc
Normal file
15
src/libstore-tests/main.cc
Normal file
|
|
@ -0,0 +1,15 @@
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
|
#include "nix/store/tests/test-main.hh"
|
||||||
|
|
||||||
|
using namespace nix;
|
||||||
|
|
||||||
|
int main(int argc, char ** argv)
|
||||||
|
{
|
||||||
|
auto res = testMainForBuidingPre(argc, argv);
|
||||||
|
if (res)
|
||||||
|
return res;
|
||||||
|
|
||||||
|
::testing::InitGoogleTest(&argc, argv);
|
||||||
|
return RUN_ALL_TESTS();
|
||||||
|
}
|
||||||
|
|
@ -66,6 +66,7 @@ sources = files(
|
||||||
'local-overlay-store.cc',
|
'local-overlay-store.cc',
|
||||||
'local-store.cc',
|
'local-store.cc',
|
||||||
'machines.cc',
|
'machines.cc',
|
||||||
|
'main.cc',
|
||||||
'nar-info-disk-cache.cc',
|
'nar-info-disk-cache.cc',
|
||||||
'nar-info.cc',
|
'nar-info.cc',
|
||||||
'nix_api_store.cc',
|
'nix_api_store.cc',
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,10 @@
|
||||||
|
#include <fstream>
|
||||||
|
|
||||||
#include "nix_api_util.h"
|
#include "nix_api_util.h"
|
||||||
#include "nix_api_store.h"
|
#include "nix_api_store.h"
|
||||||
|
|
||||||
#include "nix/store/tests/nix_api_store.hh"
|
#include "nix/store/tests/nix_api_store.hh"
|
||||||
|
#include "nix/store/globals.hh"
|
||||||
#include "nix/util/tests/string_callback.hh"
|
#include "nix/util/tests/string_callback.hh"
|
||||||
#include "nix/util/url.hh"
|
#include "nix/util/url.hh"
|
||||||
|
|
||||||
|
|
@ -197,4 +200,60 @@ TEST_F(nix_api_util_context, nix_store_real_path_binary_cache)
|
||||||
ASSERT_STREQ(path_raw.c_str(), rp.c_str());
|
ASSERT_STREQ(path_raw.c_str(), rp.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename F>
|
||||||
|
struct LambdaAdapter
|
||||||
|
{
|
||||||
|
F fun;
|
||||||
|
|
||||||
|
template<typename... Args>
|
||||||
|
static inline auto call(LambdaAdapter<F> * ths, Args... args)
|
||||||
|
{
|
||||||
|
return ths->fun(args...);
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename... Args>
|
||||||
|
static auto call_void(void * ths, Args... args)
|
||||||
|
{
|
||||||
|
return call(static_cast<LambdaAdapter<F> *>(ths), args...);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
TEST_F(nix_api_store_test_base, build_from_json)
|
||||||
|
{
|
||||||
|
// FIXME get rid of these
|
||||||
|
nix::experimentalFeatureSettings.set("extra-experimental-features", "ca-derivations");
|
||||||
|
nix::settings.substituters = {};
|
||||||
|
|
||||||
|
auto * store = open_local_store();
|
||||||
|
|
||||||
|
std::filesystem::path unitTestData{getenv("_NIX_TEST_UNIT_DATA")};
|
||||||
|
|
||||||
|
std::ifstream t{unitTestData / "derivation/ca/self-contained.json"};
|
||||||
|
std::stringstream buffer;
|
||||||
|
buffer << t.rdbuf();
|
||||||
|
|
||||||
|
auto * drv = nix_derivation_from_json(ctx, store, buffer.str().c_str());
|
||||||
|
assert_ctx_ok();
|
||||||
|
ASSERT_NE(drv, nullptr);
|
||||||
|
|
||||||
|
auto * drvPath = nix_add_derivation(ctx, store, drv);
|
||||||
|
assert_ctx_ok();
|
||||||
|
ASSERT_NE(drv, nullptr);
|
||||||
|
|
||||||
|
auto cb = LambdaAdapter{.fun = [&](const char * outname, const StorePath * outPath) {
|
||||||
|
auto is_valid_path = nix_store_is_valid_path(ctx, store, outPath);
|
||||||
|
ASSERT_EQ(is_valid_path, true);
|
||||||
|
}};
|
||||||
|
|
||||||
|
auto ret = nix_store_realise(
|
||||||
|
ctx, store, drvPath, static_cast<void *>(&cb), decltype(cb)::call_void<const char *, const StorePath *>);
|
||||||
|
assert_ctx_ok();
|
||||||
|
ASSERT_EQ(ret, NIX_OK);
|
||||||
|
|
||||||
|
// Clean up
|
||||||
|
nix_store_path_free(drvPath);
|
||||||
|
nix_derivation_free(drv);
|
||||||
|
nix_store_free(store);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace nixC
|
} // namespace nixC
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue