aboutsummaryrefslogtreecommitdiff
path: root/src/libexpr
diff options
context:
space:
mode:
authorTom Bereknyei <tomberek@gmail.com>2021-12-03 10:53:41 -0500
committerTom Bereknyei <tomberek@gmail.com>2021-12-03 10:55:30 -0500
commitb6cc0a704d8c1432e230ff65d4b74ea7114a730b (patch)
tree4e7ac9e3b2fd049fa8a605fd4495ea4de56c4332 /src/libexpr
parent2e606e87c44a8dc42664f8938eac1d4b63047dd6 (diff)
flakes: search up to git or filesystem boundary
While parsing a flakeref, upon not finding a flake.nix, search upwards until git or filesystem boundary.
Diffstat (limited to 'src/libexpr')
-rw-r--r--src/libexpr/flake/flakeref.cc23
1 files changed, 21 insertions, 2 deletions
diff --git a/src/libexpr/flake/flakeref.cc b/src/libexpr/flake/flakeref.cc
index 29128d789..074727f06 100644
--- a/src/libexpr/flake/flakeref.cc
+++ b/src/libexpr/flake/flakeref.cc
@@ -117,8 +117,27 @@ std::pair<FlakeRef, std::string> parseFlakeRefWithFragment(
if (!S_ISDIR(lstat(path).st_mode))
throw BadURL("path '%s' is not a flake (because it's not a directory)", path);
- if (!allowMissing && !pathExists(path + "/flake.nix"))
- throw BadURL("path '%s' is not a flake (because it doesn't contain a 'flake.nix' file)", path);
+ if (!allowMissing && !pathExists(path + "/flake.nix")){
+ notice("path '%s' does not contain a 'flake.nix', searching up",path);
+
+ // Save device to detect filesystem boundary
+ dev_t device = lstat(path).st_dev;
+ bool found = false;
+ while (path != "/") {
+ if (pathExists(path + "/flake.nix")) {
+ found = true;
+ break;
+ } else if (pathExists(path + "/.git"))
+ throw Error("unable to find a flake before encountering git boundary at '%s'", path);
+ else {
+ if (lstat(path).st_dev != device)
+ throw Error("unable to find a flake before encountering filesystem boundary at '%s'", path);
+ }
+ path = dirOf(path);
+ }
+ if (!found)
+ throw BadURL("could not find a flake.nix file");
+ }
auto flakeRoot = path;
std::string subdir;