nix: Add commands to query contents of NARs / binary caches
For example, $ NIX_REMOTE=file:///my-cache nix ls-store -lR /nix/store/f4kbgl8shhyy76rkk3nbxr0lz8d2ip7q-binutils-2.23.1 dr-xr-xr-x 0 ./bin -r-xr-xr-x 30748 ./bin/addr2line -r-xr-xr-x 66973 ./bin/ar ... Similarly, "nix ls-nar" lists the contents of a NAR file, "nix cat-nar" extracts a file from a NAR file, and "nix cat-store" extract a file from a Nix store.
This commit is contained in:
parent
24a8f9e27b
commit
00b2c05749
2 changed files with 197 additions and 0 deletions
74
src/nix/cat.cc
Normal file
74
src/nix/cat.cc
Normal file
|
@ -0,0 +1,74 @@
|
|||
#include "command.hh"
|
||||
#include "store-api.hh"
|
||||
#include "fs-accessor.hh"
|
||||
#include "nar-accessor.hh"
|
||||
|
||||
using namespace nix;
|
||||
|
||||
struct MixCat : virtual Args
|
||||
{
|
||||
std::string path;
|
||||
|
||||
void cat(ref<FSAccessor> accessor)
|
||||
{
|
||||
auto st = accessor->stat(path);
|
||||
if (st.type == FSAccessor::Type::tMissing)
|
||||
throw Error(format("path ‘%1%’ does not exist") % path);
|
||||
if (st.type != FSAccessor::Type::tRegular)
|
||||
throw Error(format("path ‘%1%’ is not a regular file") % path);
|
||||
|
||||
std::cout << accessor->readFile(path);
|
||||
}
|
||||
};
|
||||
|
||||
struct CmdCatStore : StoreCommand, MixCat
|
||||
{
|
||||
CmdCatStore()
|
||||
{
|
||||
expectArg("path", &path);
|
||||
}
|
||||
|
||||
std::string name() override
|
||||
{
|
||||
return "cat-store";
|
||||
}
|
||||
|
||||
std::string description() override
|
||||
{
|
||||
return "print the contents of a store file on stdout";
|
||||
}
|
||||
|
||||
void run(ref<Store> store) override
|
||||
{
|
||||
cat(store->getFSAccessor());
|
||||
}
|
||||
};
|
||||
|
||||
struct CmdCatNar : StoreCommand, MixCat
|
||||
{
|
||||
Path narPath;
|
||||
|
||||
CmdCatNar()
|
||||
{
|
||||
expectArg("nar", &narPath);
|
||||
expectArg("path", &path);
|
||||
}
|
||||
|
||||
std::string name() override
|
||||
{
|
||||
return "cat-nar";
|
||||
}
|
||||
|
||||
std::string description() override
|
||||
{
|
||||
return "print the contents of a file inside a NAR file";
|
||||
}
|
||||
|
||||
void run(ref<Store> store) override
|
||||
{
|
||||
cat(makeNarAccessor(make_ref<std::string>(readFile(narPath))));
|
||||
}
|
||||
};
|
||||
|
||||
static RegisterCommand r1(make_ref<CmdCatStore>());
|
||||
static RegisterCommand r2(make_ref<CmdCatNar>());
|
123
src/nix/ls.cc
Normal file
123
src/nix/ls.cc
Normal file
|
@ -0,0 +1,123 @@
|
|||
#include "command.hh"
|
||||
#include "store-api.hh"
|
||||
#include "fs-accessor.hh"
|
||||
#include "nar-accessor.hh"
|
||||
|
||||
using namespace nix;
|
||||
|
||||
struct MixLs : virtual Args
|
||||
{
|
||||
std::string path;
|
||||
|
||||
bool recursive = false;
|
||||
bool verbose = false;
|
||||
bool showDirectory = false;
|
||||
|
||||
MixLs()
|
||||
{
|
||||
mkFlag('R', "recursive", "list subdirectories recursively", &recursive);
|
||||
mkFlag('l', "long", "show more file information", &verbose);
|
||||
mkFlag('d', "directory", "show directories rather than their contents", &showDirectory);
|
||||
}
|
||||
|
||||
void list(ref<FSAccessor> accessor)
|
||||
{
|
||||
std::function<void(const FSAccessor::Stat &, const Path &, const std::string &, bool)> doPath;
|
||||
|
||||
auto showFile = [&](const Path & curPath, const std::string & relPath) {
|
||||
if (verbose) {
|
||||
auto st = accessor->stat(curPath);
|
||||
std::string tp =
|
||||
st.type == FSAccessor::Type::tRegular ?
|
||||
(st.isExecutable ? "-r-xr-xr-x" : "-r--r--r--") :
|
||||
st.type == FSAccessor::Type::tSymlink ? "lrwxrwxrwx" :
|
||||
"dr-xr-xr-x";
|
||||
std::cout <<
|
||||
(format("%s %20d %s") % tp % st.fileSize % relPath);
|
||||
if (st.type == FSAccessor::Type::tSymlink)
|
||||
std::cout << " -> " << accessor->readLink(curPath)
|
||||
;
|
||||
std::cout << "\n";
|
||||
if (recursive && st.type == FSAccessor::Type::tDirectory)
|
||||
doPath(st, curPath, relPath, false);
|
||||
} else {
|
||||
std::cout << relPath << "\n";
|
||||
if (recursive) {
|
||||
auto st = accessor->stat(curPath);
|
||||
if (st.type == FSAccessor::Type::tDirectory)
|
||||
doPath(st, curPath, relPath, false);
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
doPath = [&](const FSAccessor::Stat & st , const Path & curPath,
|
||||
const std::string & relPath, bool showDirectory)
|
||||
{
|
||||
if (st.type == FSAccessor::Type::tDirectory && !showDirectory) {
|
||||
auto names = accessor->readDirectory(curPath);
|
||||
for (auto & name : names)
|
||||
showFile(curPath + "/" + name, relPath + "/" + name);
|
||||
} else
|
||||
showFile(curPath, relPath);
|
||||
};
|
||||
|
||||
auto st = accessor->stat(path);
|
||||
if (st.type == FSAccessor::Type::tMissing)
|
||||
throw Error(format("path ‘%1%’ does not exist") % path);
|
||||
doPath(st, path,
|
||||
st.type == FSAccessor::Type::tDirectory ? "." : baseNameOf(path),
|
||||
showDirectory);
|
||||
}
|
||||
};
|
||||
|
||||
struct CmdLsStore : StoreCommand, MixLs
|
||||
{
|
||||
CmdLsStore()
|
||||
{
|
||||
expectArg("path", &path);
|
||||
}
|
||||
|
||||
std::string name() override
|
||||
{
|
||||
return "ls-store";
|
||||
}
|
||||
|
||||
std::string description() override
|
||||
{
|
||||
return "show information about a store path";
|
||||
}
|
||||
|
||||
void run(ref<Store> store) override
|
||||
{
|
||||
list(store->getFSAccessor());
|
||||
}
|
||||
};
|
||||
|
||||
struct CmdLsNar : Command, MixLs
|
||||
{
|
||||
Path narPath;
|
||||
|
||||
CmdLsNar()
|
||||
{
|
||||
expectArg("nar", &narPath);
|
||||
expectArg("path", &path);
|
||||
}
|
||||
|
||||
std::string name() override
|
||||
{
|
||||
return "ls-nar";
|
||||
}
|
||||
|
||||
std::string description() override
|
||||
{
|
||||
return "show information about the contents of a NAR file";
|
||||
}
|
||||
|
||||
void run() override
|
||||
{
|
||||
list(makeNarAccessor(make_ref<std::string>(readFile(narPath))));
|
||||
}
|
||||
};
|
||||
|
||||
static RegisterCommand r1(make_ref<CmdLsStore>());
|
||||
static RegisterCommand r2(make_ref<CmdLsNar>());
|
Loading…
Reference in a new issue