From ee65c7afd60a930057afd1544f9d6675f44703ed Mon Sep 17 00:00:00 2001 From: Silvan Mosberger Date: Mon, 9 Jul 2018 23:45:35 +0200 Subject: [PATCH] Separate NUR nixpkgs from repos nixpkgs, avoid callPackages This is essential to get modules in NUR to work. By taking a separate argument for NUR's nixpkgs (for fetchgit, fetchzip and lib), we don't need to evaluate the nixpkgs used for repos. This also implies that you won't be able to `callPackage` NUR anymore, and instead you'll have to use `import (builtins.fetchGit ".../NUR") { inherit pkgs; }` instead. Doing this also prevents the evaluation of pkgs. In case of NixOS, this pkgs depends on your whole config, which is the source of the recursion. Evaluating this at the last possible moment is key. This also means that you won't be able to take package arguments in a repo definition, you instead get just `pkgs`, also to avoid evaluation of it. An error will be thrown when pkgs was required for evaluation but wasn't passed to the NUR import The old callPackage syntax will still be supported albeit with a warning Also repos receive a lib argument, Using this lib instead of pkgs.lib makes it possible to define library functions that use other library functions without depending on pkgs -> should prevent some infinite recursion cases for NixOS module usage. --- default.nix | 13 +++++++++---- lib/evalRepo.nix | 34 ++++++++++++++++++++++++++++++++++ 2 files changed, 43 insertions(+), 4 deletions(-) create mode 100644 lib/evalRepo.nix diff --git a/default.nix b/default.nix index 65526ab38..1703a7999 100644 --- a/default.nix +++ b/default.nix @@ -1,7 +1,9 @@ +{ nurpkgs ? import {} # For nixpkgs dependencies used by NUR itself + # Dependencies to call NUR repos with +, pkgs ? null }: -{ pkgs ? import {} }: let - inherit (pkgs) fetchgit fetchzip callPackages lib; + inherit (nurpkgs) fetchgit fetchzip lib; manifest = (builtins.fromJSON (builtins.readFile ./repos.json)).repos; lockedRevisions = (builtins.fromJSON (builtins.readFile ./repos.json.lock)).repos; @@ -40,9 +42,12 @@ let fetchSubmodules = submodules; }; - expressionPath = name: attr: (repoSource name attr) + "/" + (attr.file or ""); + createRepo = name: attr: import ./lib/evalRepo.nix { + inherit name pkgs lib; + inherit (attr) url; + src = repoSource name attr + "/" + (attr.file or ""); + }; - createRepo = (name: attr: callPackages (expressionPath name attr) {}); in { repos = lib.mapAttrs createRepo manifest; repo-sources = lib.mapAttrs repoSource manifest; diff --git a/lib/evalRepo.nix b/lib/evalRepo.nix new file mode 100644 index 000000000..b7383c249 --- /dev/null +++ b/lib/evalRepo.nix @@ -0,0 +1,34 @@ +{ name +, url +, src +, pkgs # Do not use this for anything other than passing it along as an argument to the repository +, lib +}: +let + + prettyName = "${name}"; + + # Arguments passed to each repositories default.nix + passedArgs = { + pkgs = if pkgs != null then pkgs else throw '' + NUR import call didn't receive a pkgs argument, but the evaluation of NUR's ${prettyName} repository requires it. + + This is either because + - You're trying to use a package from that repository, but didn't pass a `pkgs` argument to the NUR import. + In that case, refer to the installation instructions at https://github.com/nix-community/nur#installation on how to properly import NUR + + - You're trying to use a module/overlay from that repository, but it didn't properly declare their module. + In that case, inform the maintainer of the repository: ${url} + ''; + }; + + expr = import src; + args = builtins.functionArgs expr; + # True if not all arguments are either passed by default (e.g. pkgs) or defaulted (e.g. foo ? 10) + usesCallPackage = ! lib.all (arg: lib.elem arg (lib.attrNames passedArgs) || args.${arg}) (lib.attrNames args); + +in if usesCallPackage then lib.warn '' + NUR repository ${prettyName} is using the deprecated callPackage syntax which + might result in infinite recursion when used with NixOS modules. + '' (passedArgs.pkgs.callPackages src {}) + else expr (builtins.intersectAttrs args passedArgs)