lix/src/nix/search.cc

177 lines
6.3 KiB
C++
Raw Normal View History

2017-07-17 17:02:56 +00:00
#include "command.hh"
#include "globals.hh"
#include "eval.hh"
#include "eval-inline.hh"
#include "names.hh"
#include "get-drvs.hh"
2017-07-18 15:30:09 +00:00
#include "common-args.hh"
#include "json.hh"
#include "shared.hh"
#include "eval-cache.hh"
#include "attr-path.hh"
#include "hilite.hh"
2017-07-17 17:02:56 +00:00
#include <regex>
#include <fstream>
2017-07-17 17:02:56 +00:00
using namespace nix;
std::string wrap(std::string prefix, std::string s)
{
return prefix + s + ANSI_NORMAL;
}
2020-04-16 13:36:15 +00:00
struct CmdSearch : InstallableCommand, MixJSON
2017-07-17 17:02:56 +00:00
{
std::vector<std::string> res;
2017-07-17 17:02:56 +00:00
CmdSearch()
{
expectArgs("regex", &res);
2017-07-17 17:02:56 +00:00
}
std::string description() override
{
2020-12-08 17:09:30 +00:00
return "search for packages";
2017-07-17 17:02:56 +00:00
}
2020-12-08 17:09:30 +00:00
std::string doc() override
2017-09-07 18:42:11 +00:00
{
2020-12-08 17:09:30 +00:00
return
#include "search.md"
;
2017-09-07 18:42:11 +00:00
}
2020-04-16 13:36:15 +00:00
Strings getDefaultFlakeAttrPaths() override
{
return {
"packages." + settings.thisSystem.get() + ".",
"legacyPackages." + settings.thisSystem.get() + "."
};
2020-04-16 13:36:15 +00:00
}
2017-07-17 17:02:56 +00:00
void run(ref<Store> store) override
{
settings.readOnlyMode = true;
evalSettings.enableImportFromDerivation.setDefault(false);
2017-07-17 17:02:56 +00:00
// Empty search string should match all packages
// Use "^" here instead of ".*" due to differences in resulting highlighting
// (see #1893 -- libc++ claims empty search string is not in POSIX grammar)
if (res.empty())
res.push_back("^");
std::vector<std::regex> regexes;
regexes.reserve(res.size());
2020-04-16 13:36:15 +00:00
for (auto & re : res)
regexes.push_back(std::regex(re, std::regex::extended | std::regex::icase));
2017-07-17 17:02:56 +00:00
auto state = getEvalState();
2017-11-14 13:27:01 +00:00
auto jsonOut = json ? std::make_unique<JSONObject>(std::cout) : nullptr;
2017-07-18 15:30:09 +00:00
uint64_t results = 0;
std::function<void(eval_cache::AttrCursor & cursor, const std::vector<Symbol> & attrPath, bool initialRecurse)> visit;
2017-07-17 17:02:56 +00:00
visit = [&](eval_cache::AttrCursor & cursor, const std::vector<Symbol> & attrPath, bool initialRecurse)
{
Activity act(*logger, lvlInfo, actUnknown,
fmt("evaluating '%s'", concatStringsSep(".", attrPath)));
2017-07-17 17:02:56 +00:00
try {
auto recurse = [&]()
{
for (const auto & attr : cursor.getAttrs()) {
auto cursor2 = cursor.getAttr(attr);
auto attrPath2(attrPath);
attrPath2.push_back(attr);
visit(*cursor2, attrPath2, false);
}
};
2017-07-17 17:02:56 +00:00
if (cursor.isDerivation()) {
DrvName name(cursor.getAttr("name")->getString());
2017-07-17 17:02:56 +00:00
auto aMeta = cursor.maybeGetAttr("meta");
auto aDescription = aMeta ? aMeta->maybeGetAttr("description") : nullptr;
auto description = aDescription ? aDescription->getString() : "";
std::replace(description.begin(), description.end(), '\n', ' ');
auto attrPath2 = concatStringsSep(".", attrPath);
2017-07-17 17:02:56 +00:00
std::vector<std::smatch> attrPathMatches;
std::vector<std::smatch> descriptionMatches;
std::vector<std::smatch> nameMatches;
bool found = false;
2017-07-17 17:02:56 +00:00
for (auto & regex : regexes) {
found = false;
auto addAll = [&found](std::sregex_iterator it, std::vector<std::smatch> & vec) {
const auto end = std::sregex_iterator();
while (it != end) {
vec.push_back(*it++);
found = true;
}
};
addAll(std::sregex_iterator(attrPath2.begin(), attrPath2.end(), regex), attrPathMatches);
addAll(std::sregex_iterator(name.name.begin(), name.name.end(), regex), nameMatches);
addAll(std::sregex_iterator(description.begin(), description.end(), regex), descriptionMatches);
if (!found)
break;
}
2017-07-17 17:02:56 +00:00
if (found)
{
results++;
2017-07-18 15:30:09 +00:00
if (json) {
auto jsonElem = jsonOut->object(attrPath2);
2020-04-24 12:42:17 +00:00
jsonElem.attr("pname", name.name);
jsonElem.attr("version", name.version);
2017-07-18 15:30:09 +00:00
jsonElem.attr("description", description);
} else {
auto name2 = hiliteMatches(name.name, std::move(nameMatches), ANSI_GREEN, "\e[0;2m");
if (results > 1) logger->cout("");
logger->cout(
2020-04-24 12:42:17 +00:00
"* %s%s",
wrap("\e[0;1m", hiliteMatches(attrPath2, std::move(attrPathMatches), ANSI_GREEN, "\e[0;1m")),
2020-04-24 12:42:17 +00:00
name.version != "" ? " (" + name.version + ")" : "");
if (description != "")
logger->cout(
" %s", hiliteMatches(description, std::move(descriptionMatches), ANSI_GREEN, ANSI_NORMAL));
}
}
2017-07-17 17:02:56 +00:00
}
else if (
attrPath.size() == 0
|| (attrPath[0] == "legacyPackages" && attrPath.size() <= 2)
|| (attrPath[0] == "packages" && attrPath.size() <= 2))
recurse();
2017-07-17 17:02:56 +00:00
else if (initialRecurse)
recurse();
else if (attrPath[0] == "legacyPackages" && attrPath.size() > 2) {
auto attr = cursor.maybeGetAttr("recurseForDerivations");
if (attr && attr->getBool())
recurse();
2017-07-17 17:02:56 +00:00
}
} catch (EvalError & e) {
if (!(attrPath.size() > 0 && attrPath[0] == "legacyPackages"))
throw;
2017-07-17 17:02:56 +00:00
}
};
for (auto & cursor : installable->getCursors(*state))
visit(*cursor, cursor->getAttrPath(), true);
if (!json && !results)
throw Error("no results for the given search term(s)!");
2017-07-17 17:02:56 +00:00
}
};
static auto rCmdSearch = registerCommand<CmdSearch>("search");