mirror of
https://github.com/NixOS/nix.git
synced 2025-11-08 19:46:02 +01:00
Fix #14480 This method is not well-defined for arbitrary stores, which do not have a notion of a "real path" -- it is only well-defined for local file systems stores, which do have exactly that notion, and so it is moved to that sub-interface instead. Some call-sites had to be fixed up for this, but in all cases the changes are positive. Using `getFSSourceAccessor` allows for more other stores to work properly. `nix-channel` was straight-up wrong in the case of redirected local stores. And the building logic with remote building and a non-local store is also fixed, properly gating some deletions on store type. Co-authored-by: Robert Hensing <robert@roberthensing.nl>
261 lines
7.1 KiB
C++
261 lines
7.1 KiB
C++
#include "nix_api_store.h"
|
|
#include "nix_api_store_internal.h"
|
|
#include "nix_api_util.h"
|
|
#include "nix_api_util_internal.h"
|
|
|
|
#include "nix/store/path.hh"
|
|
#include "nix/store/store-api.hh"
|
|
#include "nix/store/store-open.hh"
|
|
#include "nix/store/build-result.hh"
|
|
#include "nix/store/local-fs-store.hh"
|
|
|
|
#include "nix/store/globals.hh"
|
|
|
|
extern "C" {
|
|
|
|
nix_err nix_libstore_init(nix_c_context * context)
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
nix::initLibStore();
|
|
}
|
|
NIXC_CATCH_ERRS
|
|
}
|
|
|
|
nix_err nix_libstore_init_no_load_config(nix_c_context * context)
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
nix::initLibStore(false);
|
|
}
|
|
NIXC_CATCH_ERRS
|
|
}
|
|
|
|
Store * nix_store_open(nix_c_context * context, const char * uri, const char *** params)
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
std::string uri_str = uri ? uri : "";
|
|
|
|
if (uri_str.empty())
|
|
return new Store{nix::openStore()};
|
|
|
|
if (!params)
|
|
return new Store{nix::openStore(uri_str)};
|
|
|
|
nix::Store::Config::Params params_map;
|
|
for (size_t i = 0; params[i] != nullptr; i++) {
|
|
params_map[params[i][0]] = params[i][1];
|
|
}
|
|
return new Store{nix::openStore(uri_str, params_map)};
|
|
}
|
|
NIXC_CATCH_ERRS_NULL
|
|
}
|
|
|
|
void nix_store_free(Store * store)
|
|
{
|
|
delete store;
|
|
}
|
|
|
|
nix_err nix_store_get_uri(nix_c_context * context, Store * store, nix_get_string_callback callback, void * user_data)
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
auto res = store->ptr->config.getReference().render(/*withParams=*/true);
|
|
return call_nix_get_string_callback(res, callback, user_data);
|
|
}
|
|
NIXC_CATCH_ERRS
|
|
}
|
|
|
|
nix_err
|
|
nix_store_get_storedir(nix_c_context * context, Store * store, nix_get_string_callback callback, void * user_data)
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
return call_nix_get_string_callback(store->ptr->storeDir, callback, user_data);
|
|
}
|
|
NIXC_CATCH_ERRS
|
|
}
|
|
|
|
nix_err
|
|
nix_store_get_version(nix_c_context * context, Store * store, nix_get_string_callback callback, void * user_data)
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
auto res = store->ptr->getVersion();
|
|
return call_nix_get_string_callback(res.value_or(""), callback, user_data);
|
|
}
|
|
NIXC_CATCH_ERRS
|
|
}
|
|
|
|
bool nix_store_is_valid_path(nix_c_context * context, Store * store, const StorePath * path)
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
return store->ptr->isValidPath(path->path);
|
|
}
|
|
NIXC_CATCH_ERRS_RES(false);
|
|
}
|
|
|
|
nix_err nix_store_real_path(
|
|
nix_c_context * context, Store * store, StorePath * path, nix_get_string_callback callback, void * user_data)
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
auto store2 = store->ptr.dynamic_pointer_cast<nix::LocalFSStore>();
|
|
auto res = store2 ? store2->toRealPath(path->path) : store->ptr->printStorePath(path->path);
|
|
return call_nix_get_string_callback(res, callback, user_data);
|
|
}
|
|
NIXC_CATCH_ERRS
|
|
}
|
|
|
|
StorePath * nix_store_parse_path(nix_c_context * context, Store * store, const char * path)
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
nix::StorePath s = store->ptr->parseStorePath(path);
|
|
return new StorePath{std::move(s)};
|
|
}
|
|
NIXC_CATCH_ERRS_NULL
|
|
}
|
|
|
|
nix_err nix_store_get_fs_closure(
|
|
nix_c_context * context,
|
|
Store * store,
|
|
const StorePath * store_path,
|
|
bool flip_direction,
|
|
bool include_outputs,
|
|
bool include_derivers,
|
|
void * userdata,
|
|
void (*callback)(nix_c_context * context, void * userdata, const StorePath * store_path))
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
const auto nixStore = store->ptr;
|
|
|
|
nix::StorePathSet set;
|
|
nixStore->computeFSClosure(store_path->path, set, flip_direction, include_outputs, include_derivers);
|
|
|
|
if (callback) {
|
|
for (const auto & path : set) {
|
|
const StorePath tmp{path};
|
|
callback(context, userdata, &tmp);
|
|
if (context && context->last_err_code != NIX_OK)
|
|
return context->last_err_code;
|
|
}
|
|
}
|
|
}
|
|
NIXC_CATCH_ERRS
|
|
}
|
|
|
|
nix_err nix_store_realise(
|
|
nix_c_context * context,
|
|
Store * store,
|
|
StorePath * path,
|
|
void * userdata,
|
|
void (*callback)(void * userdata, const char *, const StorePath *))
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
|
|
const std::vector<nix::DerivedPath> paths{nix::DerivedPath::Built{
|
|
.drvPath = nix::makeConstantStorePathRef(path->path), .outputs = nix::OutputsSpec::All{}}};
|
|
|
|
const auto nixStore = store->ptr;
|
|
auto results = nixStore->buildPathsWithResults(paths, nix::bmNormal, nixStore);
|
|
|
|
assert(results.size() == 1);
|
|
|
|
// Check if any builds failed
|
|
for (auto & result : results) {
|
|
if (auto * failureP = result.tryGetFailure())
|
|
failureP->rethrow();
|
|
}
|
|
|
|
if (callback) {
|
|
for (const auto & result : results) {
|
|
if (auto * success = result.tryGetSuccess()) {
|
|
for (const auto & [outputName, realisation] : success->builtOutputs) {
|
|
StorePath p{realisation.outPath};
|
|
callback(userdata, outputName.c_str(), &p);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
NIXC_CATCH_ERRS
|
|
}
|
|
|
|
void nix_store_path_name(const StorePath * store_path, nix_get_string_callback callback, void * user_data)
|
|
{
|
|
std::string_view name = store_path->path.name();
|
|
callback(name.data(), name.size(), user_data);
|
|
}
|
|
|
|
void nix_store_path_free(StorePath * sp)
|
|
{
|
|
delete sp;
|
|
}
|
|
|
|
void nix_derivation_free(nix_derivation * drv)
|
|
{
|
|
delete drv;
|
|
}
|
|
|
|
StorePath * nix_store_path_clone(const StorePath * p)
|
|
{
|
|
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 = static_cast<nix::Derivation>(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)
|
|
{
|
|
if (context)
|
|
context->last_err_code = NIX_OK;
|
|
try {
|
|
nix::RealisedPath::Set paths;
|
|
paths.insert(path->path);
|
|
nix::copyClosure(*srcStore->ptr, *dstStore->ptr, paths);
|
|
}
|
|
NIXC_CATCH_ERRS
|
|
}
|
|
|
|
} // extern "C"
|