nix nixlang: неопределенные переменные pkgs в default.nix через nix-build -A привет, но работает в nix repl - PullRequest
1 голос
/ 09 февраля 2020

Я написал очень простой файл default.nix, с помощью которого я смог бы собрать пакет gnu hello (аналогично nix-pills).

Но теперь я столкнулся с ошибкой:

[jane@nixos: ~ / graphviz] $ nix-build -A привет

ошибка: неопределенная переменная 'pkgs' в /home/jane/graphviz/default.nix:3: 47

это исходный код:

[jane@nixos:~/graphviz]$ cat default.nix 
{
    pkgs = import <nixpkgs> {}; 
    mkDerivation = import ./autotools.nix pkgs;
    hello = import ./hello.nix { inherit mkDerivation ;};

}

, который не имеет абсолютно никакого смысла (для меня), как правая строка выше, я определил pkgs.

Как я не мог понять, что не так, я открыл nix repl и ввел строки.

nix-repl> pkgs = import <nixpkgs> {} 

nix-repl> mkDerivation = import ./autotools.nix pkgs

nix-repl> hello = import ./hello.nix { inherit mkDerivation ;}  

nix-repl> hello
«derivation /nix/store/g2y6sf5q236icvv2gwyg0lnij3mkr36j-hellooo.drv»

И вуаля, там все работает. Так что я не понимаю, почему это не удается с default.nix. Я мог только предположить, что default.nix является чем-то особенным, но с синтаксисом все должно быть хорошо, иначе nix repl также не будет работать.

Может кто-нибудь объяснить, почему я получаю эту неопределенную переменную ошибку?

Редактировать: Сразу после задания вопроса я нашел один способ решения ошибки неопределенной переменной, если я поставлю это так:

let pkgs = import <nixpkgs> {}; mkDerivation = import ./autotools.nix pkgs;
in
{
    hello = import ./hello.nix { inherit mkDerivation ;};

}

это работает.

Но мой оригинальный вопрос все еще остается.

1 Ответ

2 голосов
/ 09 февраля 2020

Синтаксис { } определяет только значение. Это не приносит атрибуты в это в область. Вы можете использовать синтаксис rec { }, который делает и то, и другое.

rec {
  pkgs = import <nixpkgs> {}; 
  mkDerivation = import ./autotools.nix pkgs;
  hello = import ./hello.nix { inherit mkDerivation ;};
}

То, что делает nix repl, по сути, создает let привязку каждый раз, когда что-то называют. Ваш сеанс repl может рассматриваться как интерактивное построение этого выражения:

let pkgs = import <nixpkgs> {};
in /* nixlang scope on 2nd prompt */
  let mkDerivation = import ./autotools.nix pkgs
  in /* nixlang scope on 3rd prompt */
    let hello = import ./hello.nix { inherit mkDerivation ;}
      /* nixlang scope on 4th prompt */
    in hello

Чтобы проиллюстрировать различие между созданием набора атрибутов и привязкой имени, вы могли бы назвать набор атрибутов и использовать его в своем собственном определении

let
  myScope = {
    # myScope is like any other attribute set
    pkgs = import <nixpkgs> {};

    # laziness permits the attributes inside to reference myScope
    mkDerivation = import ./autotools.nix myScope.pkgs;
    hello = import ./hello.nix { inherit (myScope) mkDerivation ;};
  };
in
  # the value of this file is the myScope attribute set
  myScope

Это эквивалентно примеру rec, но в область применения входит только одно имя: myScope.

...