forked from lix-project/lix
Jade Lovelace
1fa6a3e335
* some things that can throw are marked noexcept
yet the linter seems to think not. Maybe they can't throw in practice.
I would rather not have the UB possibility in pretty obvious cold
paths.
* various default-case-missing complaints
* a fair pile of casts from integer to character, which are in fact
deliberate.
* an instance of <https://clang.llvm.org/extra/clang-tidy/checks/bugprone/move-forwarding-reference.html>
* bugprone-not-null-terminated-result on handing a string to curl in
chunks of bytes. our usage is fine.
* reassigning a unique_ptr by CRIMES instead of using release(), then
using release() and ignoring the result. wild. let's use release() for
its intended purpose.
Change-Id: Ic3e7affef12383576213a8a7c8145c27e662513d
229 lines
6.5 KiB
C++
229 lines
6.5 KiB
C++
#include "args.hh"
|
|
#include "content-address.hh"
|
|
#include "split.hh"
|
|
|
|
namespace nix {
|
|
|
|
std::string makeFileIngestionPrefix(FileIngestionMethod m)
|
|
{
|
|
switch (m) {
|
|
case FileIngestionMethod::Flat:
|
|
return "";
|
|
case FileIngestionMethod::Recursive:
|
|
return "r:";
|
|
default:
|
|
throw Error("impossible, caught both cases");
|
|
}
|
|
}
|
|
|
|
std::string ContentAddressMethod::renderPrefix() const
|
|
{
|
|
return std::visit(overloaded {
|
|
[](TextIngestionMethod) -> std::string { return "text:"; },
|
|
[](FileIngestionMethod m2) {
|
|
/* Not prefixed for back compat with things that couldn't produce text before. */
|
|
return makeFileIngestionPrefix(m2);
|
|
},
|
|
}, raw);
|
|
}
|
|
|
|
ContentAddressMethod ContentAddressMethod::parsePrefix(std::string_view & m)
|
|
{
|
|
if (splitPrefix(m, "r:")) {
|
|
return FileIngestionMethod::Recursive;
|
|
}
|
|
else if (splitPrefix(m, "text:")) {
|
|
return TextIngestionMethod {};
|
|
}
|
|
return FileIngestionMethod::Flat;
|
|
}
|
|
|
|
std::string ContentAddressMethod::render(HashType ht) const
|
|
{
|
|
return std::visit(overloaded {
|
|
[&](const TextIngestionMethod & th) {
|
|
return std::string{"text:"} + printHashType(ht);
|
|
},
|
|
[&](const FileIngestionMethod & fim) {
|
|
return "fixed:" + makeFileIngestionPrefix(fim) + printHashType(ht);
|
|
}
|
|
}, raw);
|
|
}
|
|
|
|
std::string ContentAddress::render() const
|
|
{
|
|
return std::visit(overloaded {
|
|
[](const TextIngestionMethod &) -> std::string {
|
|
return "text:";
|
|
},
|
|
[](const FileIngestionMethod & method) {
|
|
return "fixed:"
|
|
+ makeFileIngestionPrefix(method);
|
|
},
|
|
}, method.raw)
|
|
+ this->hash.to_string(Base32, true);
|
|
}
|
|
|
|
/**
|
|
* Parses content address strings up to the hash.
|
|
*/
|
|
static std::pair<ContentAddressMethod, HashType> parseContentAddressMethodPrefix(std::string_view & rest)
|
|
{
|
|
std::string_view wholeInput { rest };
|
|
|
|
std::string_view prefix;
|
|
{
|
|
auto optPrefix = splitPrefixTo(rest, ':');
|
|
if (!optPrefix)
|
|
throw UsageError("not a content address because it is not in the form '<prefix>:<rest>': %s", wholeInput);
|
|
prefix = *optPrefix;
|
|
}
|
|
|
|
auto parseHashType_ = [&](){
|
|
auto hashTypeRaw = splitPrefixTo(rest, ':');
|
|
if (!hashTypeRaw)
|
|
throw UsageError("content address hash must be in form '<algo>:<hash>', but found: %s", wholeInput);
|
|
HashType hashType = parseHashType(*hashTypeRaw);
|
|
return hashType;
|
|
};
|
|
|
|
// Switch on prefix
|
|
if (prefix == "text") {
|
|
// No parsing of the ingestion method, "text" only support flat.
|
|
HashType hashType = parseHashType_();
|
|
return {
|
|
TextIngestionMethod {},
|
|
std::move(hashType),
|
|
};
|
|
} else if (prefix == "fixed") {
|
|
// Parse method
|
|
auto method = FileIngestionMethod::Flat;
|
|
if (splitPrefix(rest, "r:"))
|
|
method = FileIngestionMethod::Recursive;
|
|
HashType hashType = parseHashType_();
|
|
return {
|
|
std::move(method),
|
|
std::move(hashType),
|
|
};
|
|
} else
|
|
throw UsageError("content address prefix '%s' is unrecognized. Recogonized prefixes are 'text' or 'fixed'", prefix);
|
|
}
|
|
|
|
ContentAddress ContentAddress::parse(std::string_view rawCa)
|
|
{
|
|
auto rest = rawCa;
|
|
|
|
auto [caMethod, hashType] = parseContentAddressMethodPrefix(rest);
|
|
|
|
return ContentAddress {
|
|
.method = std::move(caMethod),
|
|
.hash = Hash::parseNonSRIUnprefixed(rest, hashType),
|
|
};
|
|
}
|
|
|
|
std::pair<ContentAddressMethod, HashType> ContentAddressMethod::parse(std::string_view caMethod)
|
|
{
|
|
std::string asPrefix = std::string{caMethod} + ":";
|
|
// parseContentAddressMethodPrefix takes its argument by reference
|
|
std::string_view asPrefixView = asPrefix;
|
|
return parseContentAddressMethodPrefix(asPrefixView);
|
|
}
|
|
|
|
std::optional<ContentAddress> ContentAddress::parseOpt(std::string_view rawCaOpt)
|
|
{
|
|
return rawCaOpt == ""
|
|
? std::nullopt
|
|
: std::optional { ContentAddress::parse(rawCaOpt) };
|
|
};
|
|
|
|
std::string renderContentAddress(std::optional<ContentAddress> ca)
|
|
{
|
|
return ca ? ca->render() : "";
|
|
}
|
|
|
|
std::string ContentAddress::printMethodAlgo() const
|
|
{
|
|
return method.renderPrefix()
|
|
+ printHashType(hash.type);
|
|
}
|
|
|
|
bool StoreReferences::empty() const
|
|
{
|
|
return !self && others.empty();
|
|
}
|
|
|
|
size_t StoreReferences::size() const
|
|
{
|
|
return (self ? 1 : 0) + others.size();
|
|
}
|
|
|
|
ContentAddressWithReferences ContentAddressWithReferences::withoutRefs(const ContentAddress & ca)
|
|
{
|
|
return std::visit(overloaded {
|
|
[&](const TextIngestionMethod &) -> ContentAddressWithReferences {
|
|
return TextInfo {
|
|
.hash = ca.hash,
|
|
.references = {},
|
|
};
|
|
},
|
|
[&](const FileIngestionMethod & method) -> ContentAddressWithReferences {
|
|
return FixedOutputInfo {
|
|
.method = method,
|
|
.hash = ca.hash,
|
|
.references = {},
|
|
};
|
|
},
|
|
}, ca.method.raw);
|
|
}
|
|
|
|
std::optional<ContentAddressWithReferences> ContentAddressWithReferences::fromPartsOpt(
|
|
ContentAddressMethod method, Hash hash, StoreReferences refs)
|
|
{
|
|
return std::visit(overloaded {
|
|
[&](TextIngestionMethod _) -> std::optional<ContentAddressWithReferences> {
|
|
if (refs.self)
|
|
return std::nullopt;
|
|
return ContentAddressWithReferences {
|
|
TextInfo {
|
|
.hash = std::move(hash),
|
|
.references = std::move(refs.others),
|
|
}
|
|
};
|
|
},
|
|
[&](FileIngestionMethod m2) -> std::optional<ContentAddressWithReferences> {
|
|
return ContentAddressWithReferences {
|
|
FixedOutputInfo {
|
|
.method = m2,
|
|
.hash = std::move(hash),
|
|
.references = std::move(refs),
|
|
}
|
|
};
|
|
},
|
|
}, method.raw);
|
|
}
|
|
|
|
ContentAddressMethod ContentAddressWithReferences::getMethod() const
|
|
{
|
|
return std::visit(overloaded {
|
|
[](const TextInfo & th) -> ContentAddressMethod {
|
|
return TextIngestionMethod {};
|
|
},
|
|
[](const FixedOutputInfo & fsh) -> ContentAddressMethod {
|
|
return fsh.method;
|
|
},
|
|
}, raw);
|
|
}
|
|
|
|
Hash ContentAddressWithReferences::getHash() const
|
|
{
|
|
return std::visit(overloaded {
|
|
[](const TextInfo & th) {
|
|
return th.hash;
|
|
},
|
|
[](const FixedOutputInfo & fsh) {
|
|
return fsh.hash;
|
|
},
|
|
}, raw);
|
|
}
|
|
|
|
}
|