3
0
Fork 0
forked from mirrors/nixpkgs

mkShell: make it buildable (#153194)

When I designed `mkShell`, I didn't have a good idea of what the output
should look like and so decided to make the build fail. In practice,
this causes quite a bit of confusion and complications because now the
shell cannot be part of a normal package set without failing the CI as
well.

This commit changes that build phase to record all the build inputs in a
file. That way it becomes possible to build it, makes sure that all the
build inputs get built as well, and also can be used as a GC root.
(by applying the same trick as #95536).

The documentation has also been improved to better describe what mkShell
does and how to use it.
This commit is contained in:
Jonas Chevalier 2022-01-08 01:54:04 +01:00 committed by GitHub
parent cacab72b75
commit 1e910209ae
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 42 additions and 17 deletions

View file

@ -1,17 +1,37 @@
# pkgs.mkShell {#sec-pkgs-mkShell} # pkgs.mkShell {#sec-pkgs-mkShell}
`pkgs.mkShell` is a special kind of derivation that is only useful when using `pkgs.mkShell` is a specialized `stdenv.mkDerivation` that removes some
it combined with `nix-shell`. It will in fact fail to instantiate when invoked repetition when using it with `nix-shell` (or `nix develop`).
with `nix-build`.
## Usage {#sec-pkgs-mkShell-usage} ## Usage {#sec-pkgs-mkShell-usage}
Here is a common usage example:
```nix ```nix
{ pkgs ? import <nixpkgs> {} }: { pkgs ? import <nixpkgs> {} }:
pkgs.mkShell { pkgs.mkShell {
# specify which packages to add to the shell environment
packages = [ pkgs.gnumake ]; packages = [ pkgs.gnumake ];
# add all the dependencies, of the given packages, to the shell environment
inputsFrom = with pkgs; [ hello gnutar ]; inputsFrom = [ pkgs.hello pkgs.gnutar ];
shellHook = ''
export DEBUG=1
'';
} }
``` ```
## Attributes
* `name` (default: `nix-shell`). Set the name of the derivation.
* `packages` (default: `[]`). Add executable packages to the `nix-shell` environment.
* `inputsFrom` (default: `[]`). Add build dependencies of the listed derivations to the `nix-shell` environment.
* `shellHook` (default: `""`). Bash statements that are executed by `nix-shell`.
... all the attributes of `stdenv.mkDerivation`.
## Building the shell
This derivation output will contain a text file that contains a reference to
all the build inputs. This is useful in CI where we want to make sure that
every derivation, and its dependencies, build properly. Or when creating a GC
root so that the build dependencies don't get garbage-collected.

View file

@ -1,9 +1,9 @@
{ lib, stdenv }: { lib, stdenv, buildEnv }:
# A special kind of derivation that is only meant to be consumed by the # A special kind of derivation that is only meant to be consumed by the
# nix-shell. # nix-shell.
{ { name ? "nix-shell"
# a list of packages to add to the shell environment , # a list of packages to add to the shell environment
packages ? [ ] packages ? [ ]
, # propagate all the inputs from the given derivations , # propagate all the inputs from the given derivations
inputsFrom ? [ ] inputsFrom ? [ ]
@ -15,10 +15,11 @@
}@attrs: }@attrs:
let let
mergeInputs = name: mergeInputs = name:
(attrs.${name} or []) ++ (attrs.${name} or [ ]) ++
(lib.subtractLists inputsFrom (lib.flatten (lib.catAttrs name inputsFrom))); (lib.subtractLists inputsFrom (lib.flatten (lib.catAttrs name inputsFrom)));
rest = builtins.removeAttrs attrs [ rest = builtins.removeAttrs attrs [
"name"
"packages" "packages"
"inputsFrom" "inputsFrom"
"buildInputs" "buildInputs"
@ -30,8 +31,7 @@ let
in in
stdenv.mkDerivation ({ stdenv.mkDerivation ({
name = "nix-shell"; inherit name;
phases = [ "nobuildPhase" ];
buildInputs = mergeInputs "buildInputs"; buildInputs = mergeInputs "buildInputs";
nativeBuildInputs = packages ++ (mergeInputs "nativeBuildInputs"); nativeBuildInputs = packages ++ (mergeInputs "nativeBuildInputs");
@ -41,10 +41,15 @@ stdenv.mkDerivation ({
shellHook = lib.concatStringsSep "\n" (lib.catAttrs "shellHook" shellHook = lib.concatStringsSep "\n" (lib.catAttrs "shellHook"
(lib.reverseList inputsFrom ++ [ attrs ])); (lib.reverseList inputsFrom ++ [ attrs ]));
nobuildPhase = '' phases = [ "buildPhase" ];
echo
echo "This derivation is not meant to be built, aborting"; buildPhase = ''
echo echo "------------------------------------------------------------" >>$out
exit 1 echo " WARNING: the existence of this path is not guaranteed." >>$out
echo " It is an internal implementation detail for pkgs.mkShell." >>$out
echo "------------------------------------------------------------" >>$out
echo >> $out
# Record all build inputs as runtime dependencies
export >> $out
''; '';
} // rest) } // rest)