2020-06-22 18:08:27 +00:00
|
|
|
#include "args.hh"
|
2020-06-02 19:44:58 +00:00
|
|
|
#include "content-address.hh"
|
2020-07-20 17:42:34 +00:00
|
|
|
#include "split.hh"
|
2020-06-01 21:32:27 +00:00
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
2020-09-22 09:40:19 +00:00
|
|
|
std::string FixedOutputHash::printMethodAlgo() const
|
|
|
|
{
|
2020-06-23 17:03:10 +00:00
|
|
|
return makeFileIngestionPrefix(method) + printHashType(hash.type);
|
2020-06-01 21:32:27 +00:00
|
|
|
}
|
|
|
|
|
2020-10-07 13:52:20 +00:00
|
|
|
|
2020-10-12 23:51:23 +00:00
|
|
|
std::string makeFileIngestionPrefix(FileIngestionMethod m)
|
2020-09-22 09:40:19 +00:00
|
|
|
{
|
2020-06-01 21:32:27 +00:00
|
|
|
switch (m) {
|
|
|
|
case FileIngestionMethod::Flat:
|
|
|
|
return "";
|
|
|
|
case FileIngestionMethod::Recursive:
|
|
|
|
return "r:";
|
|
|
|
}
|
2020-10-07 13:52:20 +00:00
|
|
|
assert(false);
|
2020-06-01 21:32:27 +00:00
|
|
|
}
|
|
|
|
|
2020-10-12 23:51:23 +00:00
|
|
|
std::string makeContentAddressingPrefix(ContentAddressMethod m) {
|
|
|
|
return std::visit(overloaded {
|
|
|
|
[](TextHashMethod _) -> std::string { return "text:"; },
|
|
|
|
[](FileIngestionMethod m2) {
|
|
|
|
/* Not prefixed for back compat with things that couldn't produce text before. */
|
|
|
|
return makeFileIngestionPrefix(m2);
|
|
|
|
},
|
|
|
|
}, m);
|
|
|
|
}
|
|
|
|
|
|
|
|
ContentAddressMethod parseContentAddressingPrefix(std::string_view & m)
|
|
|
|
{
|
|
|
|
ContentAddressMethod method = FileIngestionMethod::Flat;
|
|
|
|
if (splitPrefix(m, "r:"))
|
|
|
|
method = FileIngestionMethod::Recursive;
|
|
|
|
else if (splitPrefix(m, "text:"))
|
|
|
|
method = TextHashMethod {};
|
|
|
|
return method;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-01 21:32:27 +00:00
|
|
|
std::string makeFixedOutputCA(FileIngestionMethod method, const Hash & hash)
|
|
|
|
{
|
|
|
|
return "fixed:"
|
|
|
|
+ makeFileIngestionPrefix(method)
|
2020-06-18 23:01:58 +00:00
|
|
|
+ hash.to_string(Base32, true);
|
2020-06-01 21:32:27 +00:00
|
|
|
}
|
|
|
|
|
2020-09-22 09:40:19 +00:00
|
|
|
std::string renderContentAddress(ContentAddress ca)
|
|
|
|
{
|
2020-06-01 23:26:40 +00:00
|
|
|
return std::visit(overloaded {
|
|
|
|
[](TextHash th) {
|
2020-10-07 13:52:20 +00:00
|
|
|
return "text:"
|
|
|
|
+ th.hash.to_string(Base32, true);
|
2020-06-01 23:26:40 +00:00
|
|
|
},
|
2020-06-19 15:18:19 +00:00
|
|
|
[](FixedOutputHash fsh) {
|
2020-10-07 13:52:20 +00:00
|
|
|
return "fixed:"
|
|
|
|
+ makeFileIngestionPrefix(fsh.method)
|
|
|
|
+ fsh.hash.to_string(Base32, true);
|
2020-06-01 23:26:40 +00:00
|
|
|
}
|
|
|
|
}, ca);
|
|
|
|
}
|
|
|
|
|
2020-10-12 23:51:23 +00:00
|
|
|
std::string renderContentAddressMethodAndHash(ContentAddressMethod cam, HashType ht)
|
2020-09-22 09:40:19 +00:00
|
|
|
{
|
2020-09-17 15:15:05 +00:00
|
|
|
return std::visit(overloaded {
|
2020-10-12 23:51:23 +00:00
|
|
|
[&](TextHashMethod & th) {
|
|
|
|
return std::string{"text:"} + printHashType(ht);
|
2020-09-17 15:15:05 +00:00
|
|
|
},
|
2020-10-12 23:51:23 +00:00
|
|
|
[&](FileIngestionMethod & fim) {
|
|
|
|
return "fixed:" + makeFileIngestionPrefix(fim) + printHashType(ht);
|
2020-09-17 15:15:05 +00:00
|
|
|
}
|
|
|
|
}, cam);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Parses content address strings up to the hash.
|
|
|
|
*/
|
2020-10-12 23:51:23 +00:00
|
|
|
static std::pair<ContentAddressMethod, HashType> parseContentAddressMethodPrefix(std::string_view & rest)
|
2020-09-22 09:40:19 +00:00
|
|
|
{
|
2020-09-17 20:04:05 +00:00
|
|
|
std::string_view wholeInput { rest };
|
2020-06-22 18:08:27 +00:00
|
|
|
|
2020-06-30 15:57:46 +00:00
|
|
|
std::string_view prefix;
|
|
|
|
{
|
2020-07-02 23:16:57 +00:00
|
|
|
auto optPrefix = splitPrefixTo(rest, ':');
|
2020-06-30 15:57:46 +00:00
|
|
|
if (!optPrefix)
|
2020-09-17 15:15:05 +00:00
|
|
|
throw UsageError("not a content address because it is not in the form '<prefix>:<rest>': %s", wholeInput);
|
2020-06-30 15:57:46 +00:00
|
|
|
prefix = *optPrefix;
|
|
|
|
}
|
2020-06-22 18:08:27 +00:00
|
|
|
|
|
|
|
auto parseHashType_ = [&](){
|
2020-07-02 23:16:57 +00:00
|
|
|
auto hashTypeRaw = splitPrefixTo(rest, ':');
|
2020-06-30 15:57:46 +00:00
|
|
|
if (!hashTypeRaw)
|
2020-09-17 15:15:05 +00:00
|
|
|
throw UsageError("content address hash must be in form '<algo>:<hash>', but found: %s", wholeInput);
|
2020-06-30 15:57:46 +00:00
|
|
|
HashType hashType = parseHashType(*hashTypeRaw);
|
2020-06-22 18:08:27 +00:00
|
|
|
return std::move(hashType);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Switch on prefix
|
|
|
|
if (prefix == "text") {
|
2020-09-17 15:15:05 +00:00
|
|
|
// No parsing of the ingestion method, "text" only support flat.
|
2020-06-22 18:08:27 +00:00
|
|
|
HashType hashType = parseHashType_();
|
2020-10-12 23:51:23 +00:00
|
|
|
return {
|
|
|
|
TextHashMethod {},
|
|
|
|
std::move(hashType),
|
|
|
|
};
|
2020-06-22 18:08:27 +00:00
|
|
|
} else if (prefix == "fixed") {
|
|
|
|
// Parse method
|
|
|
|
auto method = FileIngestionMethod::Flat;
|
2020-07-02 23:16:57 +00:00
|
|
|
if (splitPrefix(rest, "r:"))
|
2020-06-22 18:08:27 +00:00
|
|
|
method = FileIngestionMethod::Recursive;
|
|
|
|
HashType hashType = parseHashType_();
|
2020-10-12 23:51:23 +00:00
|
|
|
return {
|
|
|
|
std::move(method),
|
|
|
|
std::move(hashType),
|
2020-06-22 18:08:27 +00:00
|
|
|
};
|
|
|
|
} else
|
2020-08-05 14:47:48 +00:00
|
|
|
throw UsageError("content address prefix '%s' is unrecognized. Recogonized prefixes are 'text' or 'fixed'", prefix);
|
2020-09-17 15:15:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ContentAddress parseContentAddress(std::string_view rawCa) {
|
|
|
|
auto rest = rawCa;
|
|
|
|
|
2020-10-13 04:11:25 +00:00
|
|
|
auto [caMethod, hashType_] = parseContentAddressMethodPrefix(rest);
|
|
|
|
auto hashType = hashType_; // work around clang bug
|
2020-10-12 23:51:23 +00:00
|
|
|
|
|
|
|
return std::visit(overloaded {
|
|
|
|
[&](TextHashMethod _) {
|
|
|
|
return ContentAddress(TextHash {
|
|
|
|
.hash = Hash::parseNonSRIUnprefixed(rest, hashType)
|
|
|
|
});
|
|
|
|
},
|
|
|
|
[&](FileIngestionMethod fim) {
|
|
|
|
return ContentAddress(FixedOutputHash {
|
|
|
|
.method = fim,
|
|
|
|
.hash = Hash::parseNonSRIUnprefixed(rest, hashType),
|
|
|
|
});
|
|
|
|
},
|
|
|
|
}, caMethod);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<ContentAddressMethod, HashType> parseContentAddressMethod(std::string_view caMethod)
|
2020-09-22 09:40:19 +00:00
|
|
|
{
|
2020-09-17 15:15:05 +00:00
|
|
|
std::string_view asPrefix {std::string{caMethod} + ":"};
|
|
|
|
return parseContentAddressMethodPrefix(asPrefix);
|
|
|
|
}
|
2020-06-02 00:37:43 +00:00
|
|
|
|
2020-09-22 09:40:19 +00:00
|
|
|
std::optional<ContentAddress> parseContentAddressOpt(std::string_view rawCaOpt)
|
|
|
|
{
|
|
|
|
return rawCaOpt == "" ? std::optional<ContentAddress>() : parseContentAddress(rawCaOpt);
|
2020-06-02 00:37:43 +00:00
|
|
|
};
|
|
|
|
|
2020-09-22 09:40:19 +00:00
|
|
|
std::string renderContentAddress(std::optional<ContentAddress> ca)
|
|
|
|
{
|
2020-06-02 00:37:43 +00:00
|
|
|
return ca ? renderContentAddress(*ca) : "";
|
2020-06-01 23:26:40 +00:00
|
|
|
}
|
|
|
|
|
2020-10-12 23:51:23 +00:00
|
|
|
ContentAddressWithReferences contentAddressFromMethodHashAndRefs(
|
|
|
|
ContentAddressMethod method, Hash && hash, PathReferences<StorePath> && refs)
|
|
|
|
{
|
|
|
|
return std::visit(overloaded {
|
|
|
|
[&](TextHashMethod _) -> ContentAddressWithReferences {
|
|
|
|
if (refs.hasSelfReference)
|
|
|
|
throw UsageError("Cannot have a self reference with text hashing scheme");
|
|
|
|
return TextInfo {
|
|
|
|
{ .hash = std::move(hash) },
|
|
|
|
std::move(refs.references),
|
|
|
|
};
|
|
|
|
},
|
|
|
|
[&](FileIngestionMethod m2) -> ContentAddressWithReferences {
|
|
|
|
return FixedOutputInfo {
|
|
|
|
{
|
|
|
|
.method = m2,
|
|
|
|
.hash = std::move(hash),
|
|
|
|
},
|
|
|
|
std::move(refs),
|
|
|
|
};
|
|
|
|
},
|
|
|
|
}, method);
|
|
|
|
}
|
|
|
|
|
|
|
|
ContentAddressMethod getContentAddressMethod(const ContentAddressWithReferences & ca)
|
|
|
|
{
|
|
|
|
return std::visit(overloaded {
|
|
|
|
[](TextInfo th) -> ContentAddressMethod {
|
|
|
|
return TextHashMethod {};
|
|
|
|
},
|
|
|
|
[](FixedOutputInfo fsh) -> ContentAddressMethod {
|
|
|
|
return fsh.method;
|
|
|
|
},
|
|
|
|
}, ca);
|
|
|
|
}
|
|
|
|
|
2020-07-10 11:21:37 +00:00
|
|
|
Hash getContentAddressHash(const ContentAddress & ca)
|
|
|
|
{
|
|
|
|
return std::visit(overloaded {
|
|
|
|
[](TextHash th) {
|
|
|
|
return th.hash;
|
|
|
|
},
|
|
|
|
[](FixedOutputHash fsh) {
|
|
|
|
return fsh.hash;
|
2020-10-07 13:52:20 +00:00
|
|
|
},
|
|
|
|
}, ca);
|
|
|
|
}
|
|
|
|
|
|
|
|
ContentAddressWithReferences caWithoutRefs(const ContentAddress & ca) {
|
|
|
|
return std::visit(overloaded {
|
|
|
|
[&](TextHash h) -> ContentAddressWithReferences {
|
|
|
|
return TextInfo { h, {}};
|
|
|
|
},
|
|
|
|
[&](FixedOutputHash h) -> ContentAddressWithReferences {
|
|
|
|
return FixedOutputInfo { h, {}};
|
|
|
|
},
|
2020-07-10 11:21:37 +00:00
|
|
|
}, ca);
|
|
|
|
}
|
|
|
|
|
2020-10-12 23:51:23 +00:00
|
|
|
Hash getContentAddressHash(const ContentAddressWithReferences & ca)
|
|
|
|
{
|
|
|
|
return std::visit(overloaded {
|
|
|
|
[](TextInfo th) {
|
|
|
|
return th.hash;
|
|
|
|
},
|
|
|
|
[](FixedOutputInfo fsh) {
|
|
|
|
return fsh.hash;
|
|
|
|
},
|
|
|
|
}, ca);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string printMethodAlgo(const ContentAddressWithReferences & ca) {
|
|
|
|
return makeContentAddressingPrefix(getContentAddressMethod(ca))
|
|
|
|
+ printHashType(getContentAddressHash(ca).type);
|
|
|
|
}
|
|
|
|
|
2020-06-01 21:32:27 +00:00
|
|
|
}
|