00b2c05749
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.
123 lines
3.5 KiB
C++
123 lines
3.5 KiB
C++
#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>());
|