aboutsummaryrefslogtreecommitdiff
path: root/src/libexpr/primops/flake.cc
diff options
context:
space:
mode:
authorNick Van den Broeck <nick.van.den.broeck666@gmail.com>2019-02-21 06:53:01 +0100
committerNick Van den Broeck <nick.van.den.broeck666@gmail.com>2019-03-22 11:16:20 +0100
commitd4ee8afd59cd7935f59b730c432cf58460af8a84 (patch)
tree42f09e66ed56a03508b1b1b55fc03f2e8d6abd88 /src/libexpr/primops/flake.cc
parent6542de98c298b6dc268b358166bd2f5bea2cc230 (diff)
Implemented --flake flag for nix build
Also fixed Eelco's PR comments
Diffstat (limited to 'src/libexpr/primops/flake.cc')
-rw-r--r--src/libexpr/primops/flake.cc64
1 files changed, 58 insertions, 6 deletions
diff --git a/src/libexpr/primops/flake.cc b/src/libexpr/primops/flake.cc
index 9d1da84f1..df0845c24 100644
--- a/src/libexpr/primops/flake.cc
+++ b/src/libexpr/primops/flake.cc
@@ -3,7 +3,9 @@
#include "eval-inline.hh"
#include "fetchGit.hh"
#include "download.hh"
+#include "args.hh"
+#include <iostream>
#include <queue>
#include <regex>
#include <nlohmann/json.hpp>
@@ -32,10 +34,10 @@ static std::unique_ptr<FlakeRegistry> readRegistry(const Path & path)
}
/* Write the registry or lock file to a file. */
-static void writeRegistry(FlakeRegistry registry, Path path = "./flake.lock")
+void writeRegistry(FlakeRegistry registry, Path path)
{
nlohmann::json json = {};
- json["value"] = 0; // Not sure whether this should be 0.
+ json["version"] = 1;
json["flakes"] = {};
for (auto elem : registry.entries) {
json["flakes"][elem.first] = elem.second.ref.to_string();
@@ -107,9 +109,21 @@ struct FlakeSourceInfo
static FlakeSourceInfo fetchFlake(EvalState & state, const FlakeRef & flakeRef)
{
- assert(flakeRef.isDirect());
+ FlakeRef directFlakeRef = FlakeRef(flakeRef);
+ if (!flakeRef.isDirect())
+ {
+ std::vector<const FlakeRegistry *> registries;
+ // 'pureEval' is a setting which cannot be changed in `nix flake`,
+ // but without flagging it off, we can't use any FlakeIds.
+ // if (!evalSettings.pureEval) {
+ registries.push_back(&state.getFlakeRegistry());
+ // }
+ directFlakeRef = lookupFlake(state, flakeRef, registries);
+ }
+ assert(directFlakeRef.isDirect());
+ // NOTE FROM NICK: I don't see why one wouldn't fetch FlakeId flakes..
- if (auto refData = std::get_if<FlakeRef::IsGitHub>(&flakeRef.data)) {
+ if (auto refData = std::get_if<FlakeRef::IsGitHub>(&directFlakeRef.data)) {
// FIXME: require hash in pure mode.
// FIXME: use regular /archive URLs instead? api.github.com
@@ -141,7 +155,7 @@ static FlakeSourceInfo fetchFlake(EvalState & state, const FlakeRef & flakeRef)
return info;
}
- else if (auto refData = std::get_if<FlakeRef::IsGit>(&flakeRef.data)) {
+ else if (auto refData = std::get_if<FlakeRef::IsGit>(&directFlakeRef.data)) {
auto gitInfo = exportGit(state.store, refData->uri, refData->ref,
refData->rev ? refData->rev->to_string(Base16, false) : "", "source");
FlakeSourceInfo info;
@@ -165,7 +179,16 @@ Flake getFlake(EvalState & state, const FlakeRef & flakeRef)
if (state.allowedPaths)
state.allowedPaths->insert(flakePath);
- Flake flake;
+ FlakeRef newFlakeRef(flakeRef);
+ if (std::get_if<FlakeRef::IsGitHub>(&newFlakeRef.data)) {
+ FlakeSourceInfo srcInfo = fetchFlake(state, newFlakeRef);
+ if (srcInfo.rev) {
+ std::string uri = flakeRef.to_string();
+ newFlakeRef = FlakeRef(uri + "/" + srcInfo.rev->to_string());
+ }
+ }
+
+ Flake flake(newFlakeRef);
Value vInfo;
state.evalFile(flakePath + "/flake.nix", vInfo); // FIXME: symlink attack
@@ -258,6 +281,35 @@ static std::tuple<FlakeId, std::map<FlakeId, Flake>> resolveFlake(EvalState & st
return {*topFlakeId, std::move(done)};
}
+FlakeRegistry updateLockFile(EvalState & evalState, FlakeRef & flakeRef)
+{
+ FlakeRegistry newLockFile;
+ std::map<FlakeId, Flake> myDependencyMap = get<1>(resolveFlake(evalState, flakeRef, false));
+ // Nick assumed that "topRefPure" means that the Flake for flakeRef can be
+ // fetched purely.
+ for (auto const& require : myDependencyMap) {
+ FlakeRegistry::Entry entry = FlakeRegistry::Entry(require.second.ref);
+ // The FlakeRefs are immutable because they come out of the Flake objects,
+ // not from the requires.
+ newLockFile.entries.insert(std::pair<FlakeId, FlakeRegistry::Entry>(require.first, entry));
+ }
+ return newLockFile;
+}
+
+void updateLockFile(EvalState & state, std::string path)
+{
+ // 'path' is the path to the local flake repo.
+ FlakeRef flakeRef = FlakeRef(path);
+ if (std::get_if<FlakeRef::IsGit>(&flakeRef.data)) {
+ FlakeRegistry newLockFile = updateLockFile(state, flakeRef);
+ writeRegistry(newLockFile, path + "/flake.lock");
+ } else if (std::get_if<FlakeRef::IsGitHub>(&flakeRef.data)) {
+ throw UsageError("You can only update local flakes, not flakes on GitHub.");
+ } else {
+ throw UsageError("You can only update local flakes, not flakes through their FlakeId.");
+ }
+}
+
Value * makeFlakeValue(EvalState & state, std::string flakeUri, Value & v)
{
// FIXME: temporary hack to make the default installation source