#include "fetchers.hh" #include "store-api.hh" #include "archive.hh" namespace nix::fetchers { struct PathInputScheme : InputScheme { std::optional inputFromURL(const ParsedURL & url, bool requireTree) const override { if (url.scheme != "path") return {}; if (url.authority && *url.authority != "") throw Error("path URL '%s' should not have an authority ('%s')", url.url, *url.authority); Input input; input.attrs.insert_or_assign("type", "path"); input.attrs.insert_or_assign("path", url.path); for (auto & [name, value] : url.query) if (name == "rev" || name == "narHash") input.attrs.insert_or_assign(name, value); else if (name == "revCount" || name == "lastModified") { if (auto n = string2Int(value)) input.attrs.insert_or_assign(name, *n); else throw Error("path URL '%s' has invalid parameter '%s'", url.to_string(), name); } else throw Error("path URL '%s' has unsupported parameter '%s'", url.to_string(), name); return input; } std::optional inputFromAttrs(const Attrs & attrs) const override { if (maybeGetStrAttr(attrs, "type") != "path") return {}; getStrAttr(attrs, "path"); for (auto & [name, value] : attrs) /* Allow the user to pass in "fake" tree info attributes. This is useful for making a pinned tree work the same as the repository from which is exported (e.g. path:/nix/store/...-source?lastModified=1585388205&rev=b0c285...). */ if (name == "type" || name == "rev" || name == "revCount" || name == "lastModified" || name == "narHash" || name == "path") // checked in Input::fromAttrs ; else throw Error("unsupported path input attribute '%s'", name); Input input; input.attrs = attrs; return input; } ParsedURL toURL(const Input & input) const override { auto query = attrsToQuery(input.attrs); query.erase("path"); query.erase("type"); return ParsedURL { .scheme = "path", .path = getStrAttr(input.attrs, "path"), .query = query, }; } bool hasAllInfo(const Input & input) const override { return true; } std::optional getSourcePath(const Input & input) const override { return getStrAttr(input.attrs, "path"); } void putFile( const Input & input, const CanonPath & path, std::string_view contents, std::optional commitMsg) const override { writeFile((CanonPath(getAbsPath(input)) + path).abs(), contents); } CanonPath getAbsPath(const Input & input) const { auto path = getStrAttr(input.attrs, "path"); if (path[0] == '/') return CanonPath(path); throw Error("cannot fetch input '%s' because it uses a relative path", input.to_string()); } std::pair fetch(ref store, const Input & _input) override { Input input(_input); std::string absPath; auto path = getStrAttr(input.attrs, "path"); if (path[0] != '/') { if (!input.parent) throw Error("cannot fetch input '%s' because it uses a relative path", input.to_string()); auto parent = canonPath(*input.parent); // the path isn't relative, prefix it absPath = nix::absPath(path, parent); // for security, ensure that if the parent is a store path, it's inside it if (store->isInStore(parent)) { auto storePath = store->printStorePath(store->toStorePath(parent).first); if (!isDirOrInDir(absPath, storePath)) throw BadStorePath("relative path '%s' points outside of its parent's store path '%s'", path, storePath); } } else absPath = path; Activity act(*logger, lvlTalkative, actUnknown, fmt("copying '%s'", absPath)); // FIXME: check whether access to 'path' is allowed. auto storePath = store->maybeParseStorePath(absPath); if (storePath) store->addTempRoot(*storePath); time_t mtime = 0; if (!storePath || storePath->name() != "source" || !store->isValidPath(*storePath)) { // FIXME: try to substitute storePath. auto src = GeneratorSource{dumpPathAndGetMtime(absPath, mtime, defaultPathFilter)}; storePath = store->addToStoreFromDump(src, "source"); } input.attrs.insert_or_assign("lastModified", uint64_t(mtime)); return {std::move(*storePath), input}; } }; static auto rPathInputScheme = OnStartup([] { registerInputScheme(std::make_unique()); }); }