lix/src/nix/flake.cc

316 lines
7.7 KiB
C++
Raw Normal View History

2018-11-29 18:18:36 +00:00
#include "command.hh"
#include "common-args.hh"
#include "shared.hh"
#include "progress-bar.hh"
#include "eval.hh"
2019-04-16 12:10:05 +00:00
#include "primops/flake.hh"
#include <nlohmann/json.hpp>
2019-03-29 15:18:25 +00:00
#include <queue>
2018-11-29 18:18:36 +00:00
using namespace nix;
struct CmdFlakeList : StoreCommand, MixEvalArgs
{
std::string name() override
{
return "list";
}
std::string description() override
{
return "list available Nix flakes";
}
void run(nix::ref<nix::Store> store) override
{
auto evalState = std::make_shared<EvalState>(searchPath, store);
2019-03-21 08:30:16 +00:00
auto registries = evalState->getFlakeRegistries();
2018-11-29 18:18:36 +00:00
stopProgressBar();
2019-04-08 17:03:00 +00:00
for (auto & registry : registries)
for (auto & entry : registry->entries)
std::cout << entry.first.to_string() << " " << entry.second.to_string() << "\n";
2018-11-29 18:18:36 +00:00
}
};
void printFlakeInfo(Flake & flake, bool json) {
if (json) {
nlohmann::json j;
2019-03-21 08:30:16 +00:00
j["id"] = flake.id;
j["location"] = flake.path;
j["description"] = flake.description;
std::cout << j.dump(4) << std::endl;
} else {
2019-03-21 08:30:16 +00:00
std::cout << "ID: " << flake.id << "\n";
std::cout << "Description: " << flake.description << "\n";
std::cout << "Location: " << flake.path << "\n";
}
}
void printNonFlakeInfo(NonFlake & nonFlake, bool json) {
if (json) {
nlohmann::json j;
2019-03-21 08:30:16 +00:00
j["name"] = nonFlake.alias;
j["location"] = nonFlake.path;
std::cout << j.dump(4) << std::endl;
} else {
2019-03-21 08:30:16 +00:00
std::cout << "name: " << nonFlake.alias << "\n";
std::cout << "Location: " << nonFlake.path << "\n";
}
}
struct CmdFlakeDeps : FlakeCommand, MixJSON, StoreCommand, MixEvalArgs
{
std::string name() override
{
return "deps";
}
std::string description() override
{
return "list informaton about dependencies";
}
void run(nix::ref<nix::Store> store) override
{
auto evalState = std::make_shared<EvalState>(searchPath, store);
FlakeRef flakeRef(flakeUri);
Dependencies deps = resolveFlake(*evalState, flakeRef, AllowRegistryAtTop);
2019-03-29 15:18:25 +00:00
std::queue<Dependencies> todo;
todo.push(deps);
2019-03-29 15:18:25 +00:00
while (!todo.empty()) {
deps = todo.front();
todo.pop();
for (auto & nonFlake : deps.nonFlakeDeps)
printNonFlakeInfo(nonFlake, json);
for (auto & newDeps : deps.flakeDeps)
todo.push(newDeps);
}
}
};
2019-02-21 05:53:01 +00:00
struct CmdFlakeUpdate : StoreCommand, GitRepoCommand, MixEvalArgs
{
std::string name() override
{
return "update";
}
std::string description() override
{
return "update flake lock file";
}
void run(nix::ref<nix::Store> store) override
{
auto evalState = std::make_shared<EvalState>(searchPath, store);
2019-03-10 06:05:05 +00:00
if (gitPath == "") gitPath = absPath(".");
2019-03-21 08:30:16 +00:00
updateLockFile(*evalState, gitPath);
}
};
2019-03-10 06:05:05 +00:00
struct CmdFlakeInfo : FlakeCommand, MixJSON, MixEvalArgs, StoreCommand
2019-02-21 05:53:01 +00:00
{
std::string name() override
{
return "info";
}
std::string description() override
{
return "list info about a given flake";
}
2019-04-16 13:06:40 +00:00
CmdFlakeInfo () { }
2019-04-16 06:21:52 +00:00
2019-02-21 05:53:01 +00:00
void run(nix::ref<nix::Store> store) override
{
auto evalState = std::make_shared<EvalState>(searchPath, store);
2019-04-08 17:03:00 +00:00
nix::Flake flake = nix::getFlake(*evalState, FlakeRef(flakeUri), true);
printFlakeInfo(flake, json);
2019-02-21 05:53:01 +00:00
}
};
2019-03-10 06:05:05 +00:00
struct CmdFlakeAdd : MixEvalArgs, Command
{
2019-04-08 17:03:00 +00:00
FlakeUri alias;
FlakeUri uri;
2019-03-10 06:05:05 +00:00
std::string name() override
{
return "add";
}
std::string description() override
{
return "upsert flake in user flake registry";
}
CmdFlakeAdd()
{
2019-04-08 17:03:00 +00:00
expectArg("alias", &alias);
expectArg("flake-uri", &uri);
2019-03-10 06:05:05 +00:00
}
void run() override
{
2019-04-08 17:03:00 +00:00
FlakeRef aliasRef(alias);
2019-03-10 06:05:05 +00:00
Path userRegistryPath = getUserRegistryPath();
auto userRegistry = readRegistry(userRegistryPath);
2019-04-08 17:03:00 +00:00
userRegistry->entries.erase(aliasRef);
userRegistry->entries.insert_or_assign(aliasRef, FlakeRef(uri));
2019-03-10 06:05:05 +00:00
writeRegistry(*userRegistry, userRegistryPath);
}
};
struct CmdFlakeRemove : virtual Args, MixEvalArgs, Command
{
2019-04-08 17:03:00 +00:00
FlakeUri alias;
2019-03-10 06:05:05 +00:00
std::string name() override
{
return "remove";
}
std::string description() override
{
return "remove flake from user flake registry";
}
CmdFlakeRemove()
{
2019-04-08 17:03:00 +00:00
expectArg("alias", &alias);
2019-03-10 06:05:05 +00:00
}
void run() override
{
Path userRegistryPath = getUserRegistryPath();
auto userRegistry = readRegistry(userRegistryPath);
2019-04-08 17:03:00 +00:00
userRegistry->entries.erase(FlakeRef(alias));
2019-03-10 06:05:05 +00:00
writeRegistry(*userRegistry, userRegistryPath);
}
};
struct CmdFlakePin : virtual Args, StoreCommand, MixEvalArgs
{
2019-04-08 17:03:00 +00:00
FlakeUri alias;
2019-03-10 06:05:05 +00:00
std::string name() override
{
return "pin";
}
std::string description() override
{
return "pin flake require in user flake registry";
}
CmdFlakePin()
{
2019-04-08 17:03:00 +00:00
expectArg("alias", &alias);
2019-03-10 06:05:05 +00:00
}
void run(nix::ref<nix::Store> store) override
{
auto evalState = std::make_shared<EvalState>(searchPath, store);
Path userRegistryPath = getUserRegistryPath();
FlakeRegistry userRegistry = *readRegistry(userRegistryPath);
2019-04-08 17:03:00 +00:00
auto it = userRegistry.entries.find(FlakeRef(alias));
2019-03-10 06:05:05 +00:00
if (it != userRegistry.entries.end()) {
2019-04-08 17:03:00 +00:00
it->second = getFlake(*evalState, it->second, true).ref;
2019-03-10 06:05:05 +00:00
// The 'ref' in 'flake' is immutable.
writeRegistry(userRegistry, userRegistryPath);
2019-04-16 06:21:52 +00:00
} else {
std::shared_ptr<FlakeRegistry> globalReg = getGlobalRegistry();
it = globalReg->entries.find(FlakeRef(alias));
if (it != globalReg->entries.end()) {
FlakeRef newRef = getFlake(*evalState, it->second, true).ref;
userRegistry.entries.insert_or_assign(alias, newRef);
writeRegistry(userRegistry, userRegistryPath);
} else
throw Error("the flake alias '%s' does not exist in the user or global registry", alias);
}
2019-03-10 06:05:05 +00:00
}
};
struct CmdFlakeInit : virtual Args, Command
{
std::string name() override
{
return "init";
}
std::string description() override
{
return "create a skeleton 'flake.nix' file in the current directory";
}
void run() override
{
Path flakeDir = absPath(".");
if (!pathExists(flakeDir + "/.git"))
throw Error("the directory '%s' is not a Git repository", flakeDir);
Path flakePath = flakeDir + "/flake.nix";
if (pathExists(flakePath))
throw Error("file '%s' already exists", flakePath);
writeFile(flakePath,
#include "flake-template.nix.gen.hh"
);
}
};
2018-11-29 18:18:36 +00:00
struct CmdFlake : virtual MultiCommand, virtual Command
{
CmdFlake()
: MultiCommand({make_ref<CmdFlakeList>()
2019-03-10 06:05:05 +00:00
, make_ref<CmdFlakeUpdate>()
, make_ref<CmdFlakeInfo>()
, make_ref<CmdFlakeDeps>()
2019-03-10 06:05:05 +00:00
, make_ref<CmdFlakeAdd>()
, make_ref<CmdFlakeRemove>()
, make_ref<CmdFlakePin>()
, make_ref<CmdFlakeInit>()
})
2018-11-29 18:18:36 +00:00
{
}
std::string name() override
{
return "flake";
}
std::string description() override
{
return "manage Nix flakes";
}
void run() override
{
if (!command)
throw UsageError("'nix flake' requires a sub-command.");
command->run();
}
void printHelp(const string & programName, std::ostream & out) override
{
MultiCommand::printHelp(programName, out);
}
};
static RegisterCommand r1(make_ref<CmdFlake>());