2016-03-24 12:41:00 +02:00
|
|
|
#include "crypto.hh"
|
2006-11-30 19:43:04 +02:00
|
|
|
#include "globals.hh"
|
2016-03-24 12:41:00 +02:00
|
|
|
#include "store-api.hh"
|
2006-11-30 20:35:36 +02:00
|
|
|
#include "util.hh"
|
2016-04-20 15:12:38 +03:00
|
|
|
#include "nar-info-disk-cache.hh"
|
2016-07-19 01:50:27 +03:00
|
|
|
#include "thread-pool.hh"
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
#include "json.hh"
|
2016-08-10 17:44:39 +03:00
|
|
|
#include "derivations.hh"
|
2020-03-30 17:04:18 +03:00
|
|
|
#include "url.hh"
|
2006-11-30 19:43:04 +02:00
|
|
|
|
2016-09-16 19:54:14 +03:00
|
|
|
#include <future>
|
|
|
|
|
2019-07-10 20:46:15 +03:00
|
|
|
|
2006-11-30 19:43:04 +02:00
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
2016-06-01 15:49:12 +03:00
|
|
|
bool Store::isInStore(const Path & path) const
|
2006-11-30 19:43:04 +02:00
|
|
|
{
|
2016-06-01 15:49:12 +03:00
|
|
|
return isInDir(path, storeDir);
|
2006-11-30 19:43:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-06-01 15:49:12 +03:00
|
|
|
Path Store::toStorePath(const Path & path) const
|
2006-11-30 19:43:04 +02:00
|
|
|
{
|
|
|
|
if (!isInStore(path))
|
2017-07-30 14:27:57 +03:00
|
|
|
throw Error(format("path '%1%' is not in the Nix store") % path);
|
2016-06-01 15:49:12 +03:00
|
|
|
Path::size_type slash = path.find('/', storeDir.size() + 1);
|
2006-11-30 19:43:04 +02:00
|
|
|
if (slash == Path::npos)
|
|
|
|
return path;
|
|
|
|
else
|
|
|
|
return Path(path, 0, slash);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-16 20:11:47 +02:00
|
|
|
Path Store::followLinksToStore(std::string_view _path) const
|
2007-11-29 18:18:24 +02:00
|
|
|
{
|
2019-12-16 20:11:47 +02:00
|
|
|
Path path = absPath(std::string(_path));
|
2007-11-29 18:18:24 +02:00
|
|
|
while (!isInStore(path)) {
|
|
|
|
if (!isLink(path)) break;
|
|
|
|
string target = readLink(path);
|
|
|
|
path = absPath(target, dirOf(path));
|
|
|
|
}
|
|
|
|
if (!isInStore(path))
|
2020-03-30 17:04:18 +03:00
|
|
|
throw NotInStore("path '%1%' is not in the Nix store", path);
|
2007-11-29 18:18:24 +02:00
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-16 20:11:47 +02:00
|
|
|
StorePath Store::followLinksToStorePath(std::string_view path) const
|
2007-11-29 18:18:24 +02:00
|
|
|
{
|
2019-12-05 20:11:09 +02:00
|
|
|
return parseStorePath(toStorePath(followLinksToStore(path)));
|
2016-02-15 13:49:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-16 20:11:47 +02:00
|
|
|
StorePathWithOutputs Store::followLinksToStorePathWithOutputs(std::string_view path) const
|
|
|
|
{
|
|
|
|
auto [path2, outputs] = nix::parsePathWithOutputs(path);
|
|
|
|
return StorePathWithOutputs(followLinksToStorePath(path2), std::move(outputs));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-15 13:49:01 +02:00
|
|
|
string storePathToHash(const Path & path)
|
|
|
|
{
|
2016-06-01 15:49:12 +03:00
|
|
|
auto base = baseNameOf(path);
|
|
|
|
assert(base.size() >= storePathHashLen);
|
|
|
|
return string(base, 0, storePathHashLen);
|
2011-07-20 21:10:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-03 17:06:30 +02:00
|
|
|
/* Store paths have the following form:
|
|
|
|
|
|
|
|
<store>/<h>-<name>
|
|
|
|
|
|
|
|
where
|
|
|
|
|
|
|
|
<store> = the location of the Nix store, usually /nix/store
|
2015-07-17 20:24:28 +03:00
|
|
|
|
2008-12-03 17:06:30 +02:00
|
|
|
<name> = a human readable name for the path, typically obtained
|
|
|
|
from the name attribute of the derivation, or the name of the
|
2011-07-20 21:10:47 +03:00
|
|
|
source file from which the store path is created. For derivation
|
|
|
|
outputs other than the default "out" output, the string "-<id>"
|
|
|
|
is suffixed to <name>.
|
2015-07-17 20:24:28 +03:00
|
|
|
|
2008-12-03 17:06:30 +02:00
|
|
|
<h> = base-32 representation of the first 160 bits of a SHA-256
|
|
|
|
hash of <s>; the hash part of the store name
|
2015-07-17 20:24:28 +03:00
|
|
|
|
2008-12-03 17:06:30 +02:00
|
|
|
<s> = the string "<type>:sha256:<h2>:<store>:<name>";
|
|
|
|
note that it includes the location of the store as well as the
|
|
|
|
name to make sure that changes to either of those are reflected
|
|
|
|
in the hash (e.g. you won't get /nix/store/<h>-name1 and
|
|
|
|
/nix/store/<h>-name2 with equal hash parts).
|
2015-07-17 20:24:28 +03:00
|
|
|
|
2008-12-03 17:06:30 +02:00
|
|
|
<type> = one of:
|
|
|
|
"text:<r1>:<r2>:...<rN>"
|
|
|
|
for plain text files written to the store using
|
|
|
|
addTextToStore(); <r1> ... <rN> are the references of the
|
|
|
|
path.
|
|
|
|
"source"
|
|
|
|
for paths copied to the store using addToStore() when recursive
|
|
|
|
= true and hashAlgo = "sha256"
|
2011-07-20 21:10:47 +03:00
|
|
|
"output:<id>"
|
2008-12-03 17:06:30 +02:00
|
|
|
for either the outputs created by derivations, OR paths copied
|
|
|
|
to the store using addToStore() with recursive != true or
|
|
|
|
hashAlgo != "sha256" (in that case "source" is used; it's
|
2011-07-20 21:10:47 +03:00
|
|
|
silly, but it's done that way for compatibility). <id> is the
|
|
|
|
name of the output (usually, "out").
|
2008-12-03 17:06:30 +02:00
|
|
|
|
|
|
|
<h2> = base-16 representation of a SHA-256 hash of:
|
|
|
|
if <type> = "text:...":
|
|
|
|
the string written to the resulting store path
|
|
|
|
if <type> = "source":
|
|
|
|
the serialisation of the path from which this store path is
|
|
|
|
copied, as returned by hashPath()
|
2016-03-24 12:27:58 +02:00
|
|
|
if <type> = "output:<id>":
|
2008-12-03 17:06:30 +02:00
|
|
|
for non-fixed derivation outputs:
|
|
|
|
the derivation (see hashDerivationModulo() in
|
|
|
|
primops.cc)
|
|
|
|
for paths copied by addToStore() or produced by fixed-output
|
|
|
|
derivations:
|
|
|
|
the string "fixed:out:<rec><algo>:<hash>:", where
|
2016-03-24 12:27:58 +02:00
|
|
|
<rec> = "r:" for recursive (path) hashes, or "" for flat
|
2008-12-03 17:06:30 +02:00
|
|
|
(file) hashes
|
|
|
|
<algo> = "md5", "sha1" or "sha256"
|
|
|
|
<hash> = base-16 representation of the path or flat hash of
|
|
|
|
the contents of the path (or expected contents of the
|
|
|
|
path for fixed-output derivations)
|
|
|
|
|
|
|
|
It would have been nicer to handle fixed-output derivations under
|
|
|
|
"source", e.g. have something like "source:<rec><algo>", but we're
|
|
|
|
stuck with this for now...
|
|
|
|
|
|
|
|
The main reason for this way of computing names is to prevent name
|
|
|
|
collisions (for security). For instance, it shouldn't be feasible
|
|
|
|
to come up with a derivation whose output path collides with the
|
|
|
|
path for a copied source. The former would have a <s> starting with
|
2017-05-11 15:02:03 +03:00
|
|
|
"output:out:", while the latter would have a <s> starting with
|
2008-12-03 17:06:30 +02:00
|
|
|
"source:".
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
StorePath Store::makeStorePath(const string & type,
|
|
|
|
const Hash & hash, std::string_view name) const
|
2006-11-30 19:43:04 +02:00
|
|
|
{
|
|
|
|
/* e.g., "source:sha256:1abc...:/nix/store:foo.tar.gz" */
|
2020-03-29 01:22:10 +02:00
|
|
|
string s = type + ":" + hash.to_string(Base::Base16) + ":" + storeDir + ":" + std::string(name);
|
|
|
|
auto h = compressHash(hashString(HashType::SHA256, s), 20);
|
2019-12-05 20:11:09 +02:00
|
|
|
return StorePath::make(h.hash, name);
|
2006-11-30 19:43:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
StorePath Store::makeOutputPath(const string & id,
|
|
|
|
const Hash & hash, std::string_view name) const
|
2011-07-20 21:10:47 +03:00
|
|
|
{
|
|
|
|
return makeStorePath("output:" + id, hash,
|
2019-12-05 20:11:09 +02:00
|
|
|
std::string(name) + (id == "out" ? "" : "-" + id));
|
2011-07-20 21:10:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
static std::string makeType(
|
|
|
|
const Store & store,
|
|
|
|
string && type,
|
|
|
|
const StorePathSet & references,
|
|
|
|
bool hasSelfReference = false)
|
2018-03-30 01:56:13 +03:00
|
|
|
{
|
|
|
|
for (auto & i : references) {
|
|
|
|
type += ":";
|
2019-12-05 20:11:09 +02:00
|
|
|
type += store.printStorePath(i);
|
2018-03-30 01:56:13 +03:00
|
|
|
}
|
2019-12-05 20:11:09 +02:00
|
|
|
if (hasSelfReference) type += ":self";
|
2019-11-26 22:07:44 +02:00
|
|
|
return std::move(type);
|
2018-03-30 01:56:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
StorePath Store::makeFixedOutputPath(
|
2020-03-23 05:43:07 +02:00
|
|
|
FileIngestionMethod method,
|
2019-12-05 20:11:09 +02:00
|
|
|
const Hash & hash,
|
|
|
|
std::string_view name,
|
|
|
|
const StorePathSet & references,
|
|
|
|
bool hasSelfReference) const
|
2006-11-30 19:43:04 +02:00
|
|
|
{
|
2020-06-02 00:12:50 +03:00
|
|
|
if (hash.type == HashType::SHA256 && method == FileIngestionMethod::Recursive) {
|
2019-12-05 20:11:09 +02:00
|
|
|
return makeStorePath(makeType(*this, "source", references, hasSelfReference), hash, name);
|
2018-03-30 01:56:13 +03:00
|
|
|
} else {
|
|
|
|
assert(references.empty());
|
2020-06-02 00:12:50 +03:00
|
|
|
return makeStorePath("output:out", hashString(HashType::SHA256,
|
2020-03-23 05:43:07 +02:00
|
|
|
"fixed:out:" + makeFileIngestionPrefix(method) +
|
2020-06-02 00:12:50 +03:00
|
|
|
hash.to_string(Base::Base16) + ":"), name);
|
2018-03-30 01:56:13 +03:00
|
|
|
}
|
2006-11-30 19:43:04 +02:00
|
|
|
}
|
|
|
|
|
2006-12-01 20:00:01 +02:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
StorePath Store::makeTextPath(std::string_view name, const Hash & hash,
|
|
|
|
const StorePathSet & references) const
|
2016-08-03 14:17:11 +03:00
|
|
|
{
|
2020-03-29 01:22:10 +02:00
|
|
|
assert(hash.type == HashType::SHA256);
|
2016-08-03 14:17:11 +03:00
|
|
|
/* Stuff the references (if any) into the type. This is a bit
|
|
|
|
hacky, but we can't put them in `s' since that would be
|
|
|
|
ambiguous. */
|
2019-12-05 20:11:09 +02:00
|
|
|
return makeStorePath(makeType(*this, "text", references), hash, name);
|
2016-08-03 14:17:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
std::pair<StorePath, Hash> Store::computeStorePathForPath(std::string_view name,
|
2020-05-27 21:04:20 +03:00
|
|
|
const Path & srcPath, FileIngestionMethod method, HashType hashAlgo, PathFilter & filter) const
|
2006-12-01 20:00:01 +02:00
|
|
|
{
|
2020-05-27 21:04:20 +03:00
|
|
|
Hash h = method == FileIngestionMethod::Recursive
|
2020-03-31 01:31:51 +03:00
|
|
|
? hashPath(hashAlgo, srcPath, filter).first
|
|
|
|
: hashFile(hashAlgo, srcPath);
|
2020-05-27 21:04:20 +03:00
|
|
|
return std::make_pair(makeFixedOutputPath(method, h, name), h);
|
2006-12-01 20:00:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
StorePath Store::computeStorePathForText(const string & name, const string & s,
|
|
|
|
const StorePathSet & references) const
|
2006-12-01 20:00:01 +02:00
|
|
|
{
|
2020-03-29 01:22:10 +02:00
|
|
|
return makeTextPath(name, hashString(HashType::SHA256, s), references);
|
2006-12-01 20:00:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-06-01 15:49:12 +03:00
|
|
|
Store::Store(const Params & params)
|
2017-04-13 16:55:38 +03:00
|
|
|
: Config(params)
|
|
|
|
, state({(size_t) pathInfoCacheSize})
|
2016-06-01 15:49:12 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-20 15:12:38 +03:00
|
|
|
std::string Store::getUri()
|
|
|
|
{
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2020-03-11 21:04:47 +02:00
|
|
|
bool Store::PathInfoCacheValue::isKnownNow()
|
|
|
|
{
|
|
|
|
std::chrono::duration ttl = didExist()
|
|
|
|
? std::chrono::seconds(settings.ttlPositiveNarInfoCache)
|
|
|
|
: std::chrono::seconds(settings.ttlNegativeNarInfoCache);
|
|
|
|
|
|
|
|
return std::chrono::steady_clock::now() < time_point + ttl;
|
|
|
|
}
|
2016-04-20 15:12:38 +03:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
bool Store::isValidPath(const StorePath & storePath)
|
2016-02-15 15:48:38 +02:00
|
|
|
{
|
2019-12-05 20:11:09 +02:00
|
|
|
auto hashPart = storePathToHash(printStorePath(storePath));
|
2016-04-21 18:53:47 +03:00
|
|
|
|
2016-04-19 19:50:15 +03:00
|
|
|
{
|
|
|
|
auto state_(state.lock());
|
2016-04-21 18:53:47 +03:00
|
|
|
auto res = state_->pathInfoCache.get(hashPart);
|
2020-03-11 21:04:47 +02:00
|
|
|
if (res && res->isKnownNow()) {
|
2016-04-19 19:50:15 +03:00
|
|
|
stats.narInfoReadAverted++;
|
2020-03-11 21:04:47 +02:00
|
|
|
return res->didExist();
|
2016-04-19 19:50:15 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-20 15:12:38 +03:00
|
|
|
if (diskCache) {
|
2016-04-21 18:53:47 +03:00
|
|
|
auto res = diskCache->lookupNarInfo(getUri(), hashPart);
|
2016-04-20 15:12:38 +03:00
|
|
|
if (res.first != NarInfoDiskCache::oUnknown) {
|
2016-04-21 18:53:47 +03:00
|
|
|
stats.narInfoReadAverted++;
|
2016-04-20 15:12:38 +03:00
|
|
|
auto state_(state.lock());
|
2016-04-21 18:53:47 +03:00
|
|
|
state_->pathInfoCache.upsert(hashPart,
|
2020-03-11 21:04:47 +02:00
|
|
|
res.first == NarInfoDiskCache::oInvalid ? PathInfoCacheValue{} : PathInfoCacheValue { .value = res.second });
|
2016-04-20 15:12:38 +03:00
|
|
|
return res.first == NarInfoDiskCache::oValid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-20 18:39:05 +03:00
|
|
|
bool valid = isValidPathUncached(storePath);
|
2016-04-20 15:12:38 +03:00
|
|
|
|
2016-06-20 18:39:05 +03:00
|
|
|
if (diskCache && !valid)
|
|
|
|
// FIXME: handle valid = true case.
|
|
|
|
diskCache->upsertNarInfo(getUri(), hashPart, 0);
|
|
|
|
|
|
|
|
return valid;
|
2016-04-19 19:50:15 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-07 20:22:48 +02:00
|
|
|
/* Default implementation for stores that only implement
|
|
|
|
queryPathInfoUncached(). */
|
2019-12-05 20:11:09 +02:00
|
|
|
bool Store::isValidPathUncached(const StorePath & path)
|
2017-02-07 20:22:48 +02:00
|
|
|
{
|
|
|
|
try {
|
|
|
|
queryPathInfo(path);
|
|
|
|
return true;
|
|
|
|
} catch (InvalidPath &) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
ref<const ValidPathInfo> Store::queryPathInfo(const StorePath & storePath)
|
2016-09-16 19:54:14 +03:00
|
|
|
{
|
2018-09-25 19:54:16 +03:00
|
|
|
std::promise<ref<const ValidPathInfo>> promise;
|
2016-09-16 19:54:14 +03:00
|
|
|
|
|
|
|
queryPathInfo(storePath,
|
2018-09-25 19:54:16 +03:00
|
|
|
{[&](std::future<ref<const ValidPathInfo>> result) {
|
2018-03-27 23:16:01 +03:00
|
|
|
try {
|
|
|
|
promise.set_value(result.get());
|
|
|
|
} catch (...) {
|
|
|
|
promise.set_exception(std::current_exception());
|
|
|
|
}
|
|
|
|
}});
|
2016-09-16 19:54:14 +03:00
|
|
|
|
|
|
|
return promise.get_future().get();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
void Store::queryPathInfo(const StorePath & storePath,
|
2018-09-25 19:54:16 +03:00
|
|
|
Callback<ref<const ValidPathInfo>> callback) noexcept
|
2016-04-19 19:50:15 +03:00
|
|
|
{
|
2019-09-03 14:00:55 +03:00
|
|
|
std::string hashPart;
|
2016-04-21 18:53:47 +03:00
|
|
|
|
2016-09-16 19:54:14 +03:00
|
|
|
try {
|
2019-12-05 20:11:09 +02:00
|
|
|
hashPart = storePathToHash(printStorePath(storePath));
|
2016-09-16 19:54:14 +03:00
|
|
|
|
|
|
|
{
|
|
|
|
auto res = state.lock()->pathInfoCache.get(hashPart);
|
2020-03-11 21:04:47 +02:00
|
|
|
if (res && res->isKnownNow()) {
|
2016-09-16 19:54:14 +03:00
|
|
|
stats.narInfoReadAverted++;
|
2020-03-11 21:04:47 +02:00
|
|
|
if (!res->didExist())
|
2019-12-05 20:11:09 +02:00
|
|
|
throw InvalidPath("path '%s' is not valid", printStorePath(storePath));
|
2020-03-11 21:04:47 +02:00
|
|
|
return callback(ref<const ValidPathInfo>(res->value));
|
2016-09-16 19:54:14 +03:00
|
|
|
}
|
2016-04-19 19:50:15 +03:00
|
|
|
}
|
|
|
|
|
2016-09-16 19:54:14 +03:00
|
|
|
if (diskCache) {
|
|
|
|
auto res = diskCache->lookupNarInfo(getUri(), hashPart);
|
|
|
|
if (res.first != NarInfoDiskCache::oUnknown) {
|
|
|
|
stats.narInfoReadAverted++;
|
|
|
|
{
|
|
|
|
auto state_(state.lock());
|
|
|
|
state_->pathInfoCache.upsert(hashPart,
|
2020-03-11 21:04:47 +02:00
|
|
|
res.first == NarInfoDiskCache::oInvalid ? PathInfoCacheValue{} : PathInfoCacheValue{ .value = res.second });
|
2016-09-16 19:54:14 +03:00
|
|
|
if (res.first == NarInfoDiskCache::oInvalid ||
|
2019-12-05 20:11:09 +02:00
|
|
|
res.second->path != storePath)
|
|
|
|
throw InvalidPath("path '%s' is not valid", printStorePath(storePath));
|
2016-09-16 19:54:14 +03:00
|
|
|
}
|
2018-09-25 19:54:16 +03:00
|
|
|
return callback(ref<const ValidPathInfo>(res.second));
|
2016-09-16 19:54:14 +03:00
|
|
|
}
|
2016-04-20 15:12:38 +03:00
|
|
|
}
|
2016-09-16 19:54:14 +03:00
|
|
|
|
2018-03-27 23:16:01 +03:00
|
|
|
} catch (...) { return callback.rethrow(); }
|
2016-04-20 15:12:38 +03:00
|
|
|
|
2019-09-03 13:51:35 +03:00
|
|
|
auto callbackPtr = std::make_shared<decltype(callback)>(std::move(callback));
|
|
|
|
|
2016-09-16 19:54:14 +03:00
|
|
|
queryPathInfoUncached(storePath,
|
2019-12-05 20:11:09 +02:00
|
|
|
{[this, storePath{printStorePath(storePath)}, hashPart, callbackPtr](std::future<std::shared_ptr<const ValidPathInfo>> fut) {
|
2016-04-19 19:50:15 +03:00
|
|
|
|
2018-03-27 23:16:01 +03:00
|
|
|
try {
|
|
|
|
auto info = fut.get();
|
2016-04-20 15:12:38 +03:00
|
|
|
|
2018-03-27 23:16:01 +03:00
|
|
|
if (diskCache)
|
|
|
|
diskCache->upsertNarInfo(getUri(), hashPart, info);
|
2016-04-19 19:50:15 +03:00
|
|
|
|
2018-03-27 23:16:01 +03:00
|
|
|
{
|
|
|
|
auto state_(state.lock());
|
2020-03-11 21:04:47 +02:00
|
|
|
state_->pathInfoCache.upsert(hashPart, PathInfoCacheValue { .value = info });
|
2018-03-27 23:16:01 +03:00
|
|
|
}
|
2016-09-16 19:54:14 +03:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
if (!info || info->path != parseStorePath(storePath)) {
|
2018-03-27 23:16:01 +03:00
|
|
|
stats.narInfoMissing++;
|
|
|
|
throw InvalidPath("path '%s' is not valid", storePath);
|
|
|
|
}
|
2016-04-19 19:50:15 +03:00
|
|
|
|
2018-09-25 19:54:16 +03:00
|
|
|
(*callbackPtr)(ref<const ValidPathInfo>(info));
|
2019-09-03 13:51:35 +03:00
|
|
|
} catch (...) { callbackPtr->rethrow(); }
|
2018-03-27 23:16:01 +03:00
|
|
|
}});
|
2016-02-15 15:48:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
StorePathSet Store::queryValidPaths(const StorePathSet & paths, SubstituteFlag maybeSubstitute)
|
2016-10-07 20:20:47 +03:00
|
|
|
{
|
2016-10-07 20:43:36 +03:00
|
|
|
struct State
|
|
|
|
{
|
|
|
|
size_t left;
|
2019-12-05 20:11:09 +02:00
|
|
|
StorePathSet valid;
|
2016-10-07 20:43:36 +03:00
|
|
|
std::exception_ptr exc;
|
|
|
|
};
|
2016-10-07 20:20:47 +03:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
Sync<State> state_(State{paths.size(), StorePathSet()});
|
2016-10-07 20:20:47 +03:00
|
|
|
|
2016-10-07 20:43:36 +03:00
|
|
|
std::condition_variable wakeup;
|
2017-10-25 18:13:49 +03:00
|
|
|
ThreadPool pool;
|
2016-10-07 20:43:36 +03:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
auto doQuery = [&](const Path & path) {
|
2017-10-25 18:51:45 +03:00
|
|
|
checkInterrupt();
|
2019-12-05 20:11:09 +02:00
|
|
|
queryPathInfo(parseStorePath(path), {[path, this, &state_, &wakeup](std::future<ref<const ValidPathInfo>> fut) {
|
2018-03-27 23:16:01 +03:00
|
|
|
auto state(state_.lock());
|
|
|
|
try {
|
|
|
|
auto info = fut.get();
|
2019-12-05 20:11:09 +02:00
|
|
|
state->valid.insert(parseStorePath(path));
|
2018-03-27 23:16:01 +03:00
|
|
|
} catch (InvalidPath &) {
|
|
|
|
} catch (...) {
|
|
|
|
state->exc = std::current_exception();
|
|
|
|
}
|
|
|
|
assert(state->left);
|
|
|
|
if (!--state->left)
|
|
|
|
wakeup.notify_one();
|
|
|
|
}});
|
2017-10-25 18:13:49 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
for (auto & path : paths)
|
2019-12-05 20:11:09 +02:00
|
|
|
pool.enqueue(std::bind(doQuery, printStorePath(path))); // FIXME
|
2017-10-25 18:13:49 +03:00
|
|
|
|
|
|
|
pool.process();
|
2016-10-07 20:43:36 +03:00
|
|
|
|
|
|
|
while (true) {
|
|
|
|
auto state(state_.lock());
|
|
|
|
if (!state->left) {
|
|
|
|
if (state->exc) std::rethrow_exception(state->exc);
|
2019-12-05 20:11:09 +02:00
|
|
|
return std::move(state->valid);
|
2016-10-07 20:43:36 +03:00
|
|
|
}
|
|
|
|
state.wait(wakeup);
|
|
|
|
}
|
2016-10-07 20:20:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-29 20:17:36 +02:00
|
|
|
/* Return a string accepted by decodeValidPathInfo() that
|
|
|
|
registers the specified paths as valid. Note: it's the
|
|
|
|
responsibility of the caller to provide a closure. */
|
2019-12-05 20:11:09 +02:00
|
|
|
string Store::makeValidityRegistration(const StorePathSet & paths,
|
2008-01-29 20:17:36 +02:00
|
|
|
bool showDerivers, bool showHash)
|
|
|
|
{
|
|
|
|
string s = "";
|
|
|
|
|
2015-07-17 20:24:28 +03:00
|
|
|
for (auto & i : paths) {
|
2019-12-05 20:11:09 +02:00
|
|
|
s += printStorePath(i) + "\n";
|
2015-07-17 20:24:28 +03:00
|
|
|
|
2016-04-19 19:50:15 +03:00
|
|
|
auto info = queryPathInfo(i);
|
2008-01-29 20:17:36 +02:00
|
|
|
|
2010-11-16 19:11:46 +02:00
|
|
|
if (showHash) {
|
2020-03-29 01:22:10 +02:00
|
|
|
s += info->narHash.to_string(Base::Base16, false) + "\n";
|
2016-04-19 19:50:15 +03:00
|
|
|
s += (format("%1%\n") % info->narSize).str();
|
2010-11-16 19:11:46 +02:00
|
|
|
}
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
auto deriver = showDerivers && info->deriver ? printStorePath(*info->deriver) : "";
|
2008-01-29 20:17:36 +02:00
|
|
|
s += deriver + "\n";
|
|
|
|
|
2016-04-19 19:50:15 +03:00
|
|
|
s += (format("%1%\n") % info->references.size()).str();
|
2008-01-29 20:17:36 +02:00
|
|
|
|
2016-04-19 19:50:15 +03:00
|
|
|
for (auto & j : info->references)
|
2019-12-05 20:11:09 +02:00
|
|
|
s += printStorePath(j) + "\n";
|
2008-01-29 20:17:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
void Store::pathInfoToJSON(JSONPlaceholder & jsonOut, const StorePathSet & storePaths,
|
2020-02-12 00:50:16 +02:00
|
|
|
bool includeImpureInfo, bool showClosureSize,
|
|
|
|
Base hashBase,
|
|
|
|
AllowInvalidFlag allowInvalid)
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
{
|
|
|
|
auto jsonList = jsonOut.list();
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
for (auto & storePath : storePaths) {
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
auto jsonPath = jsonList.object();
|
2019-12-05 20:11:09 +02:00
|
|
|
jsonPath.attr("path", printStorePath(storePath));
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
|
2017-07-14 16:27:21 +03:00
|
|
|
try {
|
|
|
|
auto info = queryPathInfo(storePath);
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
|
2017-07-14 16:27:21 +03:00
|
|
|
jsonPath
|
2020-02-12 00:50:16 +02:00
|
|
|
.attr("narHash", info->narHash.to_string(hashBase))
|
2017-07-14 16:27:21 +03:00
|
|
|
.attr("narSize", info->narSize);
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
|
2017-07-14 16:27:21 +03:00
|
|
|
{
|
|
|
|
auto jsonRefs = jsonPath.list("references");
|
|
|
|
for (auto & ref : info->references)
|
2019-12-05 20:11:09 +02:00
|
|
|
jsonRefs.elem(printStorePath(ref));
|
2017-07-14 16:27:21 +03:00
|
|
|
}
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
|
2017-07-14 16:27:21 +03:00
|
|
|
if (info->ca != "")
|
|
|
|
jsonPath.attr("ca", info->ca);
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
|
2017-07-14 18:36:49 +03:00
|
|
|
std::pair<uint64_t, uint64_t> closureSizes;
|
|
|
|
|
|
|
|
if (showClosureSize) {
|
2019-12-05 20:11:09 +02:00
|
|
|
closureSizes = getClosureSize(info->path);
|
2017-07-14 18:36:49 +03:00
|
|
|
jsonPath.attr("closureSize", closureSizes.first);
|
|
|
|
}
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
|
2017-07-14 16:27:21 +03:00
|
|
|
if (includeImpureInfo) {
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
if (info->deriver)
|
|
|
|
jsonPath.attr("deriver", printStorePath(*info->deriver));
|
2017-07-14 16:27:21 +03:00
|
|
|
|
|
|
|
if (info->registrationTime)
|
|
|
|
jsonPath.attr("registrationTime", info->registrationTime);
|
|
|
|
|
|
|
|
if (info->ultimate)
|
|
|
|
jsonPath.attr("ultimate", info->ultimate);
|
|
|
|
|
|
|
|
if (!info->sigs.empty()) {
|
|
|
|
auto jsonSigs = jsonPath.list("signatures");
|
|
|
|
for (auto & sig : info->sigs)
|
|
|
|
jsonSigs.elem(sig);
|
|
|
|
}
|
2017-05-08 14:36:23 +03:00
|
|
|
|
2017-07-14 18:36:49 +03:00
|
|
|
auto narInfo = std::dynamic_pointer_cast<const NarInfo>(
|
|
|
|
std::shared_ptr<const ValidPathInfo>(info));
|
|
|
|
|
|
|
|
if (narInfo) {
|
2017-11-24 19:08:50 +02:00
|
|
|
if (!narInfo->url.empty())
|
|
|
|
jsonPath.attr("url", narInfo->url);
|
2017-07-14 18:36:49 +03:00
|
|
|
if (narInfo->fileHash)
|
|
|
|
jsonPath.attr("downloadHash", narInfo->fileHash.to_string());
|
|
|
|
if (narInfo->fileSize)
|
|
|
|
jsonPath.attr("downloadSize", narInfo->fileSize);
|
|
|
|
if (showClosureSize)
|
|
|
|
jsonPath.attr("closureDownloadSize", closureSizes.second);
|
|
|
|
}
|
2017-05-08 14:36:23 +03:00
|
|
|
}
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
|
2017-07-14 16:27:21 +03:00
|
|
|
} catch (InvalidPath &) {
|
|
|
|
jsonPath.attr("valid", false);
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
std::pair<uint64_t, uint64_t> Store::getClosureSize(const StorePath & storePath)
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
{
|
2017-07-14 18:36:49 +03:00
|
|
|
uint64_t totalNarSize = 0, totalDownloadSize = 0;
|
2019-12-05 20:11:09 +02:00
|
|
|
StorePathSet closure;
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
computeFSClosure(storePath, closure, false, false);
|
2017-07-14 18:36:49 +03:00
|
|
|
for (auto & p : closure) {
|
|
|
|
auto info = queryPathInfo(p);
|
|
|
|
totalNarSize += info->narSize;
|
|
|
|
auto narInfo = std::dynamic_pointer_cast<const NarInfo>(
|
|
|
|
std::shared_ptr<const ValidPathInfo>(info));
|
|
|
|
if (narInfo)
|
|
|
|
totalDownloadSize += narInfo->fileSize;
|
|
|
|
}
|
|
|
|
return {totalNarSize, totalDownloadSize};
|
exportReferencesGraph: Export more complete info in JSON format
This writes info about every path in the closure in the same format as
‘nix path-info --json’. Thus it also includes NAR hashes and sizes.
Example:
[
{
"path": "/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"narHash": "sha256:0ckdc4z20kkmpqdilx0wl6cricxv90lh85xpv2qljppcmz6vzcxl",
"narSize": 197648,
"references": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20939776
},
{
"path": "/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24",
"narHash": "sha256:1nfn3m3p98y1c0kd0brp80dn9n5mycwgrk183j17rajya0h7gax3",
"narSize": 20742128,
"references": [
"/nix/store/27binbdy296qvjycdgr1535v8872vz3z-glibc-2.24"
],
"closureSize": 20742128
}
]
Fixes #1134.
2017-01-26 21:36:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-19 19:50:15 +03:00
|
|
|
const Store::Stats & Store::getStats()
|
|
|
|
{
|
2016-04-20 15:12:38 +03:00
|
|
|
{
|
|
|
|
auto state_(state.lock());
|
|
|
|
stats.pathInfoCacheSize = state_->pathInfoCache.size();
|
|
|
|
}
|
2016-04-19 19:50:15 +03:00
|
|
|
return stats;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
void Store::buildPaths(const std::vector<StorePathWithOutputs> & paths, BuildMode buildMode)
|
2017-05-01 14:43:34 +03:00
|
|
|
{
|
2019-12-05 20:11:09 +02:00
|
|
|
StorePathSet paths2;
|
|
|
|
|
|
|
|
for (auto & path : paths) {
|
|
|
|
if (path.path.isDerivation())
|
2019-01-18 14:34:23 +02:00
|
|
|
unsupported("buildPaths");
|
2019-12-05 20:11:09 +02:00
|
|
|
paths2.insert(path.path.clone());
|
|
|
|
}
|
2017-05-01 14:43:34 +03:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
if (queryValidPaths(paths2).size() != paths2.size())
|
2019-01-18 14:34:23 +02:00
|
|
|
unsupported("buildPaths");
|
2017-05-01 14:43:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-03 15:45:50 +03:00
|
|
|
void copyStorePath(ref<Store> srcStore, ref<Store> dstStore,
|
2019-12-05 20:11:09 +02:00
|
|
|
const StorePath & storePath, RepairFlag repair, CheckSigsFlag checkSigs)
|
2016-05-03 15:45:50 +03:00
|
|
|
{
|
2019-07-10 20:46:15 +03:00
|
|
|
auto srcUri = srcStore->getUri();
|
|
|
|
auto dstUri = dstStore->getUri();
|
|
|
|
|
2020-03-29 01:22:10 +02:00
|
|
|
Activity act(*logger, Verbosity::Info, ActivityType::CopyPath,
|
2019-07-10 20:46:15 +03:00
|
|
|
srcUri == "local" || srcUri == "daemon"
|
2019-12-05 20:11:09 +02:00
|
|
|
? fmt("copying path '%s' to '%s'", srcStore->printStorePath(storePath), dstUri)
|
2019-07-10 20:46:15 +03:00
|
|
|
: dstUri == "local" || dstUri == "daemon"
|
2019-12-05 20:11:09 +02:00
|
|
|
? fmt("copying path '%s' from '%s'", srcStore->printStorePath(storePath), srcUri)
|
|
|
|
: fmt("copying path '%s' from '%s' to '%s'", srcStore->printStorePath(storePath), srcUri, dstUri),
|
|
|
|
{srcStore->printStorePath(storePath), srcUri, dstUri});
|
2019-07-10 20:46:15 +03:00
|
|
|
PushActivity pact(act.id);
|
|
|
|
|
|
|
|
auto info = srcStore->queryPathInfo(storePath);
|
|
|
|
|
|
|
|
uint64_t total = 0;
|
|
|
|
|
|
|
|
if (!info->narHash) {
|
|
|
|
StringSink sink;
|
|
|
|
srcStore->narFromPath({storePath}, sink);
|
|
|
|
auto info2 = make_ref<ValidPathInfo>(*info);
|
2020-03-29 01:22:10 +02:00
|
|
|
info2->narHash = hashString(HashType::SHA256, *sink.s);
|
2019-07-10 20:46:15 +03:00
|
|
|
if (!info->narSize) info2->narSize = sink.s->size();
|
|
|
|
if (info->ultimate) info2->ultimate = false;
|
|
|
|
info = info2;
|
|
|
|
|
|
|
|
StringSource source(*sink.s);
|
|
|
|
dstStore->addToStore(*info, source, repair, checkSigs);
|
|
|
|
return;
|
|
|
|
}
|
2017-02-07 20:23:16 +02:00
|
|
|
|
2019-07-10 20:46:15 +03:00
|
|
|
if (info->ultimate) {
|
|
|
|
auto info2 = make_ref<ValidPathInfo>(*info);
|
|
|
|
info2->ultimate = false;
|
|
|
|
info = info2;
|
|
|
|
}
|
2017-05-01 21:03:25 +03:00
|
|
|
|
2019-07-10 20:46:15 +03:00
|
|
|
auto source = sinkToSource([&](Sink & sink) {
|
|
|
|
LambdaSink wrapperSink([&](const unsigned char * data, size_t len) {
|
|
|
|
sink(data, len);
|
|
|
|
total += len;
|
|
|
|
act.progress(total, info->narSize);
|
2018-03-16 21:22:34 +02:00
|
|
|
});
|
2019-12-05 20:11:09 +02:00
|
|
|
srcStore->narFromPath(storePath, wrapperSink);
|
2019-07-10 20:46:15 +03:00
|
|
|
}, [&]() {
|
2019-12-05 20:11:09 +02:00
|
|
|
throw EndOfFile("NAR for '%s' fetched from '%s' is incomplete", srcStore->printStorePath(storePath), srcStore->getUri());
|
2019-06-24 22:48:52 +03:00
|
|
|
});
|
2019-07-10 20:46:15 +03:00
|
|
|
|
|
|
|
dstStore->addToStore(*info, *source, repair, checkSigs);
|
2016-05-03 15:45:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
void copyPaths(ref<Store> srcStore, ref<Store> dstStore, const StorePathSet & storePaths,
|
2017-06-28 19:11:01 +03:00
|
|
|
RepairFlag repair, CheckSigsFlag checkSigs, SubstituteFlag substitute)
|
2016-10-07 20:15:25 +03:00
|
|
|
{
|
2019-12-05 20:11:09 +02:00
|
|
|
auto valid = dstStore->queryValidPaths(storePaths, substitute);
|
2017-06-28 19:11:01 +03:00
|
|
|
|
|
|
|
PathSet missing;
|
2016-10-07 20:15:25 +03:00
|
|
|
for (auto & path : storePaths)
|
2019-12-05 20:11:09 +02:00
|
|
|
if (!valid.count(path)) missing.insert(srcStore->printStorePath(path));
|
2016-10-07 20:15:25 +03:00
|
|
|
|
2017-10-18 16:02:58 +03:00
|
|
|
if (missing.empty()) return;
|
|
|
|
|
2020-03-29 01:22:10 +02:00
|
|
|
Activity act(*logger, Verbosity::Info, ActivityType::CopyPaths, fmt("copying %d paths", missing.size()));
|
2017-08-14 16:28:16 +03:00
|
|
|
|
2017-08-14 23:12:36 +03:00
|
|
|
std::atomic<size_t> nrDone{0};
|
2018-07-24 18:03:54 +03:00
|
|
|
std::atomic<size_t> nrFailed{0};
|
2017-08-14 20:00:03 +03:00
|
|
|
std::atomic<uint64_t> bytesExpected{0};
|
2017-08-14 23:42:17 +03:00
|
|
|
std::atomic<uint64_t> nrRunning{0};
|
2017-08-14 16:28:16 +03:00
|
|
|
|
|
|
|
auto showProgress = [&]() {
|
2018-07-24 18:03:54 +03:00
|
|
|
act.progress(nrDone, missing.size(), nrRunning, nrFailed);
|
2017-08-14 16:28:16 +03:00
|
|
|
};
|
|
|
|
|
2017-06-28 19:11:01 +03:00
|
|
|
ThreadPool pool;
|
2017-03-16 14:50:01 +02:00
|
|
|
|
2017-06-28 19:11:01 +03:00
|
|
|
processGraph<Path>(pool,
|
|
|
|
PathSet(missing.begin(), missing.end()),
|
2016-10-07 20:15:25 +03:00
|
|
|
|
2017-06-28 19:11:01 +03:00
|
|
|
[&](const Path & storePath) {
|
2019-12-05 20:11:09 +02:00
|
|
|
if (dstStore->isValidPath(dstStore->parseStorePath(storePath))) {
|
2017-08-14 16:28:16 +03:00
|
|
|
nrDone++;
|
|
|
|
showProgress();
|
|
|
|
return PathSet();
|
|
|
|
}
|
2017-08-14 20:00:03 +03:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
auto info = srcStore->queryPathInfo(srcStore->parseStorePath(storePath));
|
2017-08-14 20:00:03 +03:00
|
|
|
|
|
|
|
bytesExpected += info->narSize;
|
2020-03-29 01:22:10 +02:00
|
|
|
act.setExpected(ActivityType::CopyPath, bytesExpected);
|
2017-08-14 20:00:03 +03:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
return srcStore->printStorePathSet(info->references);
|
2017-06-28 19:11:01 +03:00
|
|
|
},
|
2016-10-07 20:15:25 +03:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
[&](const Path & storePathS) {
|
2017-06-28 19:11:01 +03:00
|
|
|
checkInterrupt();
|
2016-10-07 20:15:25 +03:00
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
auto storePath = dstStore->parseStorePath(storePathS);
|
|
|
|
|
2017-06-28 19:11:01 +03:00
|
|
|
if (!dstStore->isValidPath(storePath)) {
|
2017-08-14 23:42:17 +03:00
|
|
|
MaintainCount<decltype(nrRunning)> mc(nrRunning);
|
|
|
|
showProgress();
|
2018-07-24 18:03:54 +03:00
|
|
|
try {
|
|
|
|
copyStorePath(srcStore, dstStore, storePath, repair, checkSigs);
|
|
|
|
} catch (Error &e) {
|
|
|
|
nrFailed++;
|
|
|
|
if (!settings.keepGoing)
|
|
|
|
throw e;
|
2020-03-29 01:22:10 +02:00
|
|
|
logger->log(Verbosity::Error, fmt("could not copy %s: %s", storePathS, e.what()));
|
2018-07-24 18:03:54 +03:00
|
|
|
showProgress();
|
|
|
|
return;
|
|
|
|
}
|
2017-06-28 19:11:01 +03:00
|
|
|
}
|
2017-08-14 16:28:16 +03:00
|
|
|
|
|
|
|
nrDone++;
|
|
|
|
showProgress();
|
2017-06-28 19:11:01 +03:00
|
|
|
});
|
|
|
|
}
|
2016-10-07 20:15:25 +03:00
|
|
|
|
|
|
|
|
2017-06-28 19:11:01 +03:00
|
|
|
void copyClosure(ref<Store> srcStore, ref<Store> dstStore,
|
2019-12-05 20:11:09 +02:00
|
|
|
const StorePathSet & storePaths, RepairFlag repair, CheckSigsFlag checkSigs,
|
2017-06-28 19:11:01 +03:00
|
|
|
SubstituteFlag substitute)
|
|
|
|
{
|
2019-12-05 20:11:09 +02:00
|
|
|
StorePathSet closure;
|
|
|
|
srcStore->computeFSClosure(storePaths, closure);
|
2017-06-28 19:11:01 +03:00
|
|
|
copyPaths(srcStore, dstStore, closure, repair, checkSigs, substitute);
|
2016-10-07 20:15:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
std::optional<ValidPathInfo> decodeValidPathInfo(const Store & store, std::istream & str, bool hashGiven)
|
2007-08-12 03:29:28 +03:00
|
|
|
{
|
2019-12-05 20:11:09 +02:00
|
|
|
std::string path;
|
|
|
|
getline(str, path);
|
|
|
|
if (str.eof()) { return {}; }
|
|
|
|
ValidPathInfo info(store.parseStorePath(path));
|
2008-01-29 20:17:36 +02:00
|
|
|
if (hashGiven) {
|
|
|
|
string s;
|
|
|
|
getline(str, s);
|
2020-03-29 01:22:10 +02:00
|
|
|
info.narHash = Hash(s, HashType::SHA256);
|
2010-11-16 19:11:46 +02:00
|
|
|
getline(str, s);
|
|
|
|
if (!string2Int(s, info.narSize)) throw Error("number expected");
|
2008-01-29 20:17:36 +02:00
|
|
|
}
|
2019-12-05 20:11:09 +02:00
|
|
|
std::string deriver;
|
|
|
|
getline(str, deriver);
|
|
|
|
if (deriver != "") info.deriver = store.parseStorePath(deriver);
|
2007-08-12 03:29:28 +03:00
|
|
|
string s; int n;
|
|
|
|
getline(str, s);
|
|
|
|
if (!string2Int(s, n)) throw Error("number expected");
|
|
|
|
while (n--) {
|
|
|
|
getline(str, s);
|
2019-12-05 20:11:09 +02:00
|
|
|
info.references.insert(store.parseStorePath(s));
|
2007-08-12 03:29:28 +03:00
|
|
|
}
|
|
|
|
if (!str || str.eof()) throw Error("missing input");
|
2019-12-05 20:11:09 +02:00
|
|
|
return std::optional<ValidPathInfo>(std::move(info));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::string Store::showPaths(const StorePathSet & paths)
|
|
|
|
{
|
|
|
|
std::string s;
|
|
|
|
for (auto & i : paths) {
|
|
|
|
if (s.size() != 0) s += ", ";
|
|
|
|
s += "'" + printStorePath(i) + "'";
|
|
|
|
}
|
|
|
|
return s;
|
2007-08-12 03:29:28 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-06-09 16:52:45 +03:00
|
|
|
string showPaths(const PathSet & paths)
|
|
|
|
{
|
2019-05-02 22:09:52 +03:00
|
|
|
return concatStringsSep(", ", quoteStrings(paths));
|
2008-06-09 16:52:45 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
std::string ValidPathInfo::fingerprint(const Store & store) const
|
2016-03-24 12:41:00 +02:00
|
|
|
{
|
2016-04-20 15:12:38 +03:00
|
|
|
if (narSize == 0 || !narHash)
|
2019-12-05 20:11:09 +02:00
|
|
|
throw Error("cannot calculate fingerprint of path '%s' because its size/hash is not known",
|
|
|
|
store.printStorePath(path));
|
2016-03-24 12:41:00 +02:00
|
|
|
return
|
2019-12-05 20:11:09 +02:00
|
|
|
"1;" + store.printStorePath(path) + ";"
|
2020-03-29 01:22:10 +02:00
|
|
|
+ narHash.to_string(Base::Base32) + ";"
|
2016-03-24 12:41:00 +02:00
|
|
|
+ std::to_string(narSize) + ";"
|
2019-12-05 20:11:09 +02:00
|
|
|
+ concatStringsSep(",", store.printStorePathSet(references));
|
2016-03-24 12:41:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
void ValidPathInfo::sign(const Store & store, const SecretKey & secretKey)
|
2016-03-24 12:41:00 +02:00
|
|
|
{
|
2019-12-05 20:11:09 +02:00
|
|
|
sigs.insert(secretKey.signDetached(fingerprint(store)));
|
2016-03-24 12:41:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-03 14:17:11 +03:00
|
|
|
bool ValidPathInfo::isContentAddressed(const Store & store) const
|
|
|
|
{
|
|
|
|
auto warn = [&]() {
|
2019-12-05 20:11:09 +02:00
|
|
|
printError("warning: path '%s' claims to be content-addressed but isn't", store.printStorePath(path));
|
2016-08-03 14:17:11 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
if (hasPrefix(ca, "text:")) {
|
2017-07-04 15:47:59 +03:00
|
|
|
Hash hash(std::string(ca, 5));
|
2019-12-05 20:11:09 +02:00
|
|
|
if (store.makeTextPath(path.name(), hash, references) == path)
|
2016-08-03 14:17:11 +03:00
|
|
|
return true;
|
|
|
|
else
|
|
|
|
warn();
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (hasPrefix(ca, "fixed:")) {
|
2020-03-29 08:04:55 +03:00
|
|
|
FileIngestionMethod recursive { ca.compare(6, 2, "r:") == 0 };
|
2020-03-31 01:31:51 +03:00
|
|
|
Hash hash(std::string(ca, recursive == FileIngestionMethod::Recursive ? 8 : 6));
|
2019-12-05 20:11:09 +02:00
|
|
|
auto refs = cloneStorePathSet(references);
|
|
|
|
bool hasSelfReference = false;
|
|
|
|
if (refs.count(path)) {
|
|
|
|
hasSelfReference = true;
|
|
|
|
refs.erase(path);
|
|
|
|
}
|
|
|
|
if (store.makeFixedOutputPath(recursive, hash, path.name(), refs, hasSelfReference) == path)
|
2016-08-03 14:17:11 +03:00
|
|
|
return true;
|
|
|
|
else
|
|
|
|
warn();
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
size_t ValidPathInfo::checkSignatures(const Store & store, const PublicKeys & publicKeys) const
|
2016-03-24 12:41:00 +02:00
|
|
|
{
|
2016-08-03 14:17:11 +03:00
|
|
|
if (isContentAddressed(store)) return maxSigs;
|
|
|
|
|
|
|
|
size_t good = 0;
|
2016-03-24 12:41:00 +02:00
|
|
|
for (auto & sig : sigs)
|
2019-12-05 20:11:09 +02:00
|
|
|
if (checkSignature(store, publicKeys, sig))
|
2016-03-24 12:41:00 +02:00
|
|
|
good++;
|
|
|
|
return good;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-05 20:11:09 +02:00
|
|
|
bool ValidPathInfo::checkSignature(const Store & store, const PublicKeys & publicKeys, const std::string & sig) const
|
2016-04-07 16:14:12 +03:00
|
|
|
{
|
2019-12-05 20:11:09 +02:00
|
|
|
return verifyDetached(fingerprint(store), sig, publicKeys);
|
2016-04-07 16:14:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-20 15:12:38 +03:00
|
|
|
Strings ValidPathInfo::shortRefs() const
|
|
|
|
{
|
|
|
|
Strings refs;
|
|
|
|
for (auto & r : references)
|
2019-12-05 20:11:09 +02:00
|
|
|
refs.push_back(std::string(r.to_string()));
|
2016-04-20 15:12:38 +03:00
|
|
|
return refs;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-16 21:22:34 +02:00
|
|
|
void Store::addToStore(const ValidPathInfo & info, Source & narSource,
|
|
|
|
RepairFlag repair, CheckSigsFlag checkSigs,
|
|
|
|
std::shared_ptr<FSAccessor> accessor)
|
|
|
|
{
|
|
|
|
addToStore(info, make_ref<std::string>(narSource.drain()), repair, checkSigs, accessor);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Store::addToStore(const ValidPathInfo & info, const ref<std::string> & nar,
|
|
|
|
RepairFlag repair, CheckSigsFlag checkSigs,
|
|
|
|
std::shared_ptr<FSAccessor> accessor)
|
|
|
|
{
|
|
|
|
StringSource source(*nar);
|
|
|
|
addToStore(info, source, repair, checkSigs, accessor);
|
|
|
|
}
|
|
|
|
|
2006-11-30 19:43:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#include "local-store.hh"
|
2006-11-30 20:35:36 +02:00
|
|
|
#include "remote-store.hh"
|
2006-11-30 19:43:04 +02:00
|
|
|
|
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
2016-02-29 17:11:11 +02:00
|
|
|
RegisterStoreImplementation::Implementations * RegisterStoreImplementation::implementations = 0;
|
|
|
|
|
2019-02-21 12:44:25 +02:00
|
|
|
/* Split URI into protocol+hierarchy part and its parameter set. */
|
|
|
|
std::pair<std::string, Store::Params> splitUriAndParams(const std::string & uri_)
|
2006-11-30 19:43:04 +02:00
|
|
|
{
|
2016-04-29 17:26:16 +03:00
|
|
|
auto uri(uri_);
|
2019-02-21 12:44:25 +02:00
|
|
|
Store::Params params;
|
2016-04-29 17:26:16 +03:00
|
|
|
auto q = uri.find('?');
|
|
|
|
if (q != std::string::npos) {
|
2020-03-30 17:04:18 +03:00
|
|
|
params = decodeQuery(uri.substr(q + 1));
|
2016-04-29 17:26:16 +03:00
|
|
|
uri = uri_.substr(0, q);
|
|
|
|
}
|
2019-02-21 12:44:25 +02:00
|
|
|
return {uri, params};
|
|
|
|
}
|
|
|
|
|
|
|
|
ref<Store> openStore(const std::string & uri_,
|
|
|
|
const Store::Params & extraParams)
|
|
|
|
{
|
|
|
|
auto [uri, uriParams] = splitUriAndParams(uri_);
|
|
|
|
auto params = extraParams;
|
|
|
|
params.insert(uriParams.begin(), uriParams.end());
|
2016-04-29 17:26:16 +03:00
|
|
|
|
2016-02-29 17:11:11 +02:00
|
|
|
for (auto fun : *RegisterStoreImplementation::implementations) {
|
2016-04-29 17:26:16 +03:00
|
|
|
auto store = fun(uri, params);
|
2017-04-13 16:55:38 +03:00
|
|
|
if (store) {
|
2018-03-27 19:41:31 +03:00
|
|
|
store->warnUnknownSettings();
|
2017-04-13 16:55:38 +03:00
|
|
|
return ref<Store>(store);
|
|
|
|
}
|
2016-02-24 15:48:16 +02:00
|
|
|
}
|
|
|
|
|
2017-07-30 14:27:57 +03:00
|
|
|
throw Error("don't know how to open Nix store '%s'", uri);
|
2016-02-29 17:11:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-02 13:39:29 +03:00
|
|
|
StoreType getStoreType(const std::string & uri, const std::string & stateDir)
|
2016-02-29 17:11:11 +02:00
|
|
|
{
|
2016-09-02 13:39:29 +03:00
|
|
|
if (uri == "daemon") {
|
|
|
|
return tDaemon;
|
2017-10-24 16:16:18 +03:00
|
|
|
} else if (uri == "local" || hasPrefix(uri, "/")) {
|
2016-09-02 13:39:29 +03:00
|
|
|
return tLocal;
|
2017-04-06 20:01:05 +03:00
|
|
|
} else if (uri == "" || uri == "auto") {
|
2016-06-02 14:33:49 +03:00
|
|
|
if (access(stateDir.c_str(), R_OK | W_OK) == 0)
|
2016-09-02 13:39:29 +03:00
|
|
|
return tLocal;
|
2016-01-31 11:19:14 +02:00
|
|
|
else if (pathExists(settings.nixDaemonSocketFile))
|
2016-09-02 13:39:29 +03:00
|
|
|
return tDaemon;
|
2016-01-31 11:19:14 +02:00
|
|
|
else
|
2016-09-02 13:39:29 +03:00
|
|
|
return tLocal;
|
|
|
|
} else {
|
|
|
|
return tOther;
|
2016-01-31 11:19:14 +02:00
|
|
|
}
|
2016-02-29 17:11:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-29 17:26:16 +03:00
|
|
|
static RegisterStoreImplementation regStore([](
|
2016-06-01 15:49:12 +03:00
|
|
|
const std::string & uri, const Store::Params & params)
|
2016-04-29 17:26:16 +03:00
|
|
|
-> std::shared_ptr<Store>
|
|
|
|
{
|
2019-12-05 20:11:09 +02:00
|
|
|
switch (getStoreType(uri, get(params, "state").value_or(settings.nixStateDir))) {
|
2016-09-02 13:39:29 +03:00
|
|
|
case tDaemon:
|
2016-09-02 21:15:04 +03:00
|
|
|
return std::shared_ptr<Store>(std::make_shared<UDSRemoteStore>(params));
|
2017-10-24 16:16:18 +03:00
|
|
|
case tLocal: {
|
|
|
|
Store::Params params2 = params;
|
|
|
|
if (hasPrefix(uri, "/"))
|
|
|
|
params2["root"] = uri;
|
|
|
|
return std::shared_ptr<Store>(std::make_shared<LocalStore>(params2));
|
|
|
|
}
|
2016-09-02 13:39:29 +03:00
|
|
|
default:
|
|
|
|
return nullptr;
|
2016-01-31 11:19:14 +02:00
|
|
|
}
|
2016-02-29 17:11:11 +02:00
|
|
|
});
|
2016-02-24 15:48:16 +02:00
|
|
|
|
|
|
|
|
2016-04-29 14:57:08 +03:00
|
|
|
std::list<ref<Store>> getDefaultSubstituters()
|
|
|
|
{
|
2017-07-04 17:26:48 +03:00
|
|
|
static auto stores([]() {
|
2016-04-29 14:57:08 +03:00
|
|
|
std::list<ref<Store>> stores;
|
|
|
|
|
2017-07-04 17:26:48 +03:00
|
|
|
StringSet done;
|
2016-04-29 14:57:08 +03:00
|
|
|
|
2017-07-04 17:26:48 +03:00
|
|
|
auto addStore = [&](const std::string & uri) {
|
2019-10-09 16:51:52 +03:00
|
|
|
if (!done.insert(uri).second) return;
|
2018-02-09 15:36:38 +02:00
|
|
|
try {
|
|
|
|
stores.push_back(openStore(uri));
|
|
|
|
} catch (Error & e) {
|
|
|
|
printError("warning: %s", e.what());
|
|
|
|
}
|
2017-07-04 17:26:48 +03:00
|
|
|
};
|
2016-04-29 14:57:08 +03:00
|
|
|
|
2017-07-04 17:26:48 +03:00
|
|
|
for (auto uri : settings.substituters.get())
|
|
|
|
addStore(uri);
|
2017-03-21 18:59:18 +02:00
|
|
|
|
2017-07-04 17:26:48 +03:00
|
|
|
for (auto uri : settings.extraSubstituters.get())
|
|
|
|
addStore(uri);
|
2016-04-29 14:57:08 +03:00
|
|
|
|
2017-07-04 17:34:53 +03:00
|
|
|
stores.sort([](ref<Store> & a, ref<Store> & b) {
|
2019-12-17 18:17:53 +02:00
|
|
|
return a->priority < b->priority;
|
2017-07-04 17:34:53 +03:00
|
|
|
});
|
|
|
|
|
2017-07-04 17:26:48 +03:00
|
|
|
return stores;
|
|
|
|
} ());
|
2016-04-29 14:57:08 +03:00
|
|
|
|
2017-07-04 17:26:48 +03:00
|
|
|
return stores;
|
2016-04-29 14:57:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-30 19:43:04 +02:00
|
|
|
}
|