#include "args.hh" #include "content-address.hh" #include "split.hh" #include "strings.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(Base::Base32, true); } /** * Parses content address strings up to the hash. */ static std::pair 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 ':': %s", wholeInput); prefix = *optPrefix; } auto parseHashType_ = [&](){ auto hashTypeRaw = splitPrefixTo(rest, ':'); if (!hashTypeRaw) throw UsageError("content address hash must be in form ':', 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::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::parseOpt(std::string_view rawCaOpt) { return rawCaOpt == "" ? std::nullopt : std::optional { ContentAddress::parse(rawCaOpt) }; }; std::string renderContentAddress(std::optional 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::fromPartsOpt( ContentAddressMethod method, Hash hash, StoreReferences refs) { return std::visit(overloaded { [&](TextIngestionMethod _) -> std::optional { if (refs.self) return std::nullopt; return ContentAddressWithReferences { TextInfo { .hash = std::move(hash), .references = std::move(refs.others), } }; }, [&](FileIngestionMethod m2) -> std::optional { 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); } }