From 091d8370a215b384b6cdb5448e5c7110ada9d32b Mon Sep 17 00:00:00 2001 From: Robert Hensing Date: Sat, 29 Jun 2024 16:25:25 +0200 Subject: [PATCH 1/3] devShellTools: init (empty) --- doc/build-helpers.md | 1 + doc/build-helpers/dev-shell-tools.chapter.md | 14 ++++++++++++++ pkgs/build-support/dev-shell-tools/README.md | 13 +++++++++++++ pkgs/build-support/dev-shell-tools/default.nix | 4 ++++ .../dev-shell-tools/tests/default.nix | 5 +++++ pkgs/test/default.nix | 2 ++ pkgs/top-level/all-packages.nix | 2 ++ 7 files changed, 41 insertions(+) create mode 100644 doc/build-helpers/dev-shell-tools.chapter.md create mode 100644 pkgs/build-support/dev-shell-tools/README.md create mode 100644 pkgs/build-support/dev-shell-tools/default.nix create mode 100644 pkgs/build-support/dev-shell-tools/tests/default.nix diff --git a/doc/build-helpers.md b/doc/build-helpers.md index 06737e166760..010665484cfd 100644 --- a/doc/build-helpers.md +++ b/doc/build-helpers.md @@ -20,6 +20,7 @@ There is no uniform interface for build helpers. build-helpers/fetchers.chapter.md build-helpers/trivial-build-helpers.chapter.md build-helpers/testers.chapter.md +build-helpers/dev-shell-tools.chapter.md build-helpers/special.md build-helpers/images.md hooks/index.md diff --git a/doc/build-helpers/dev-shell-tools.chapter.md b/doc/build-helpers/dev-shell-tools.chapter.md new file mode 100644 index 000000000000..09b9893daff7 --- /dev/null +++ b/doc/build-helpers/dev-shell-tools.chapter.md @@ -0,0 +1,14 @@ +# Development Shell helpers {#chap-devShellTools} + +The `nix-shell` command has popularized the concept of transient shell environments for development or testing purposes. + +However, `nix-shell` is not the only way to create such environments, and even `nix-shell` itself can indirectly benefit from this library. + +This library provides a set of functions that help create such environments. + + diff --git a/pkgs/build-support/dev-shell-tools/README.md b/pkgs/build-support/dev-shell-tools/README.md new file mode 100644 index 000000000000..d6d9a8c8ad0d --- /dev/null +++ b/pkgs/build-support/dev-shell-tools/README.md @@ -0,0 +1,13 @@ + +# `devShellTools` + +This directory implements the `pkgs.devShellTools` library. + +# Contributing to `devShellTools` + +- Documentation should be contributed to the Nixpkgs manual, not here. + +- Tests are available in the `tests` directory. + You may run them with `nix-build -A tests.devShellTools`. + +- See [../../README.md](../../README.md) for more information on contributing to Nixpkgs. diff --git a/pkgs/build-support/dev-shell-tools/default.nix b/pkgs/build-support/dev-shell-tools/default.nix new file mode 100644 index 000000000000..b61813bb4eb6 --- /dev/null +++ b/pkgs/build-support/dev-shell-tools/default.nix @@ -0,0 +1,4 @@ +{ }: + +{ +} diff --git a/pkgs/build-support/dev-shell-tools/tests/default.nix b/pkgs/build-support/dev-shell-tools/tests/default.nix new file mode 100644 index 000000000000..d8147ea85118 --- /dev/null +++ b/pkgs/build-support/dev-shell-tools/tests/default.nix @@ -0,0 +1,5 @@ +{ ... }: + +{ + +} diff --git a/pkgs/test/default.nix b/pkgs/test/default.nix index 2b978ff3e1bc..1b7882bcbed3 100644 --- a/pkgs/test/default.nix +++ b/pkgs/test/default.nix @@ -83,6 +83,8 @@ with pkgs; inherit gccTests; }; + devShellTools = callPackage ../build-support/dev-shell-tools/tests { }; + stdenv-inputs = callPackage ./stdenv-inputs { }; stdenv = callPackage ./stdenv { }; diff --git a/pkgs/top-level/all-packages.nix b/pkgs/top-level/all-packages.nix index c202166f41a8..f5520bfe3631 100644 --- a/pkgs/top-level/all-packages.nix +++ b/pkgs/top-level/all-packages.nix @@ -837,6 +837,8 @@ with pkgs; grsync = callPackage ../applications/misc/grsync { }; + devShellTools = callPackage ../build-support/dev-shell-tools { }; + dockerTools = callPackage ../build-support/docker { writePython3 = buildPackages.writers.writePython3; } // { __attrsFailEvaluation = true; }; From 469039098bfb7f3c9b1c6229139f22e4ef89d704 Mon Sep 17 00:00:00 2001 From: Robert Hensing Date: Sat, 29 Jun 2024 17:07:17 +0200 Subject: [PATCH 2/3] devShellTools.stringValue: init --- doc/build-helpers/dev-shell-tools.chapter.md | 17 ++++++- .../build-support/dev-shell-tools/default.nix | 18 ++++++-- .../dev-shell-tools/tests/default.nix | 44 ++++++++++++++++++- pkgs/build-support/docker/default.nix | 16 +++---- 4 files changed, 78 insertions(+), 17 deletions(-) diff --git a/doc/build-helpers/dev-shell-tools.chapter.md b/doc/build-helpers/dev-shell-tools.chapter.md index 09b9893daff7..4f6980143210 100644 --- a/doc/build-helpers/dev-shell-tools.chapter.md +++ b/doc/build-helpers/dev-shell-tools.chapter.md @@ -11,4 +11,19 @@ However, `nix-shell` is not the only way to create such environments, and even ` This library provides a set of functions that help create such environments. - +## `devShellTools.stringValue` {#sec-devShellTools-stringValue} + +Converts Nix values to strings in the way the [`derivation` built-in function](https://nix.dev/manual/nix/2.23/language/derivations) does. + +:::{.example} +## `stringValue` usage examples + +```nix +devShellTools.stringValue (builtins.toFile "foo" "bar") +=> "/nix/store/...-foo" +``` + +```nix +devShellTools.stringValue false +=> "" +``` diff --git a/pkgs/build-support/dev-shell-tools/default.nix b/pkgs/build-support/dev-shell-tools/default.nix index b61813bb4eb6..f0f180dc1905 100644 --- a/pkgs/build-support/dev-shell-tools/default.nix +++ b/pkgs/build-support/dev-shell-tools/default.nix @@ -1,4 +1,16 @@ -{ }: - -{ +{ lib }: +let + inherit (builtins) typeOf; +in +rec { + # This function closely mirrors what this Nix code does: + # https://github.com/NixOS/nix/blob/2.8.0/src/libexpr/primops.cc#L1102 + # https://github.com/NixOS/nix/blob/2.8.0/src/libexpr/eval.cc#L1981-L2036 + stringValue = value: + # We can't just use `toString` on all derivation attributes because that + # would not put path literals in the closure. So we explicitly copy + # those into the store here + if typeOf value == "path" then "${value}" + else if typeOf value == "list" then toString (map stringValue value) + else toString value; } diff --git a/pkgs/build-support/dev-shell-tools/tests/default.nix b/pkgs/build-support/dev-shell-tools/tests/default.nix index d8147ea85118..d4b0872dd6b8 100644 --- a/pkgs/build-support/dev-shell-tools/tests/default.nix +++ b/pkgs/build-support/dev-shell-tools/tests/default.nix @@ -1,5 +1,45 @@ -{ ... }: - { + devShellTools, + emptyFile, + lib, + stdenv, + hello, +}: +let + inherit (lib) escapeShellArg; +in +{ + # nix-build -A tests.devShellTools.stringValue + stringValue = + let inherit (devShellTools) stringValue; in + stdenv.mkDerivation { + name = "devShellTools-stringValue-built-tests"; + + # Test inputs + inherit emptyFile hello; + one = 1; + boolTrue = true; + boolFalse = false; + foo = "foo"; + list = [ 1 2 3 ]; + pathDefaultNix = ./default.nix; + packages = [ hello emptyFile ]; + # TODO: nested lists + + buildCommand = '' + touch $out + ( set -x + [[ "$one" = ${escapeShellArg (stringValue 1)} ]] + [[ "$boolTrue" = ${escapeShellArg (stringValue true)} ]] + [[ "$boolFalse" = ${escapeShellArg (stringValue false)} ]] + [[ "$foo" = ${escapeShellArg (stringValue "foo")} ]] + [[ "$hello" = ${escapeShellArg (stringValue hello)} ]] + [[ "$list" = ${escapeShellArg (stringValue [ 1 2 3 ])} ]] + [[ "$packages" = ${escapeShellArg (stringValue [ hello emptyFile ])} ]] + [[ "$pathDefaultNix" = ${escapeShellArg (stringValue ./default.nix)} ]] + [[ "$emptyFile" = ${escapeShellArg (stringValue emptyFile)} ]] + ) >log 2>&1 || { cat log; exit 1; } + ''; + }; } diff --git a/pkgs/build-support/docker/default.nix b/pkgs/build-support/docker/default.nix index 1d1989d27fbb..033969a7fc88 100644 --- a/pkgs/build-support/docker/default.nix +++ b/pkgs/build-support/docker/default.nix @@ -4,6 +4,7 @@ , callPackage , closureInfo , coreutils +, devShellTools , e2fsprogs , proot , fakeNss @@ -49,6 +50,10 @@ let toList ; + inherit (devShellTools) + stringValue + ; + mkDbExtraCommand = contents: let contentsList = if builtins.isList contents then contents else [ contents ]; @@ -1173,17 +1178,6 @@ rec { # https://github.com/NixOS/nix/blob/2.8.0/src/libstore/globals.hh#L464-L465 sandboxBuildDir = "/build"; - # This function closely mirrors what this Nix code does: - # https://github.com/NixOS/nix/blob/2.8.0/src/libexpr/primops.cc#L1102 - # https://github.com/NixOS/nix/blob/2.8.0/src/libexpr/eval.cc#L1981-L2036 - stringValue = value: - # We can't just use `toString` on all derivation attributes because that - # would not put path literals in the closure. So we explicitly copy - # those into the store here - if builtins.typeOf value == "path" then "${value}" - else if builtins.typeOf value == "list" then toString (map stringValue value) - else toString value; - # https://github.com/NixOS/nix/blob/2.8.0/src/libstore/build/local-derivation-goal.cc#L992-L1004 drvEnv = lib.mapAttrs' (name: value: let str = stringValue value; From 8398e087cdf4b41d5495a9b623a1fbf026471c78 Mon Sep 17 00:00:00 2001 From: Robert Hensing Date: Sat, 29 Jun 2024 17:22:57 +0200 Subject: [PATCH 3/3] devShellTools.{stringValue -> valueToString} --- doc/build-helpers/dev-shell-tools.chapter.md | 8 +++--- .../build-support/dev-shell-tools/default.nix | 4 +-- .../dev-shell-tools/tests/default.nix | 26 +++++++++---------- pkgs/build-support/docker/default.nix | 6 ++--- 4 files changed, 22 insertions(+), 22 deletions(-) diff --git a/doc/build-helpers/dev-shell-tools.chapter.md b/doc/build-helpers/dev-shell-tools.chapter.md index 4f6980143210..21636df8017b 100644 --- a/doc/build-helpers/dev-shell-tools.chapter.md +++ b/doc/build-helpers/dev-shell-tools.chapter.md @@ -11,19 +11,19 @@ However, `nix-shell` is not the only way to create such environments, and even ` This library provides a set of functions that help create such environments. -## `devShellTools.stringValue` {#sec-devShellTools-stringValue} +## `devShellTools.valueToString` {#sec-devShellTools-valueToString} Converts Nix values to strings in the way the [`derivation` built-in function](https://nix.dev/manual/nix/2.23/language/derivations) does. :::{.example} -## `stringValue` usage examples +## `valueToString` usage examples ```nix -devShellTools.stringValue (builtins.toFile "foo" "bar") +devShellTools.valueToString (builtins.toFile "foo" "bar") => "/nix/store/...-foo" ``` ```nix -devShellTools.stringValue false +devShellTools.valueToString false => "" ``` diff --git a/pkgs/build-support/dev-shell-tools/default.nix b/pkgs/build-support/dev-shell-tools/default.nix index f0f180dc1905..cd5fa5f5937e 100644 --- a/pkgs/build-support/dev-shell-tools/default.nix +++ b/pkgs/build-support/dev-shell-tools/default.nix @@ -6,11 +6,11 @@ rec { # This function closely mirrors what this Nix code does: # https://github.com/NixOS/nix/blob/2.8.0/src/libexpr/primops.cc#L1102 # https://github.com/NixOS/nix/blob/2.8.0/src/libexpr/eval.cc#L1981-L2036 - stringValue = value: + valueToString = value: # We can't just use `toString` on all derivation attributes because that # would not put path literals in the closure. So we explicitly copy # those into the store here if typeOf value == "path" then "${value}" - else if typeOf value == "list" then toString (map stringValue value) + else if typeOf value == "list" then toString (map valueToString value) else toString value; } diff --git a/pkgs/build-support/dev-shell-tools/tests/default.nix b/pkgs/build-support/dev-shell-tools/tests/default.nix index d4b0872dd6b8..bfedc04409a9 100644 --- a/pkgs/build-support/dev-shell-tools/tests/default.nix +++ b/pkgs/build-support/dev-shell-tools/tests/default.nix @@ -9,12 +9,12 @@ let inherit (lib) escapeShellArg; in { - # nix-build -A tests.devShellTools.stringValue - stringValue = - let inherit (devShellTools) stringValue; in + # nix-build -A tests.devShellTools.valueToString + valueToString = + let inherit (devShellTools) valueToString; in stdenv.mkDerivation { - name = "devShellTools-stringValue-built-tests"; + name = "devShellTools-valueToString-built-tests"; # Test inputs inherit emptyFile hello; @@ -30,15 +30,15 @@ in buildCommand = '' touch $out ( set -x - [[ "$one" = ${escapeShellArg (stringValue 1)} ]] - [[ "$boolTrue" = ${escapeShellArg (stringValue true)} ]] - [[ "$boolFalse" = ${escapeShellArg (stringValue false)} ]] - [[ "$foo" = ${escapeShellArg (stringValue "foo")} ]] - [[ "$hello" = ${escapeShellArg (stringValue hello)} ]] - [[ "$list" = ${escapeShellArg (stringValue [ 1 2 3 ])} ]] - [[ "$packages" = ${escapeShellArg (stringValue [ hello emptyFile ])} ]] - [[ "$pathDefaultNix" = ${escapeShellArg (stringValue ./default.nix)} ]] - [[ "$emptyFile" = ${escapeShellArg (stringValue emptyFile)} ]] + [[ "$one" = ${escapeShellArg (valueToString 1)} ]] + [[ "$boolTrue" = ${escapeShellArg (valueToString true)} ]] + [[ "$boolFalse" = ${escapeShellArg (valueToString false)} ]] + [[ "$foo" = ${escapeShellArg (valueToString "foo")} ]] + [[ "$hello" = ${escapeShellArg (valueToString hello)} ]] + [[ "$list" = ${escapeShellArg (valueToString [ 1 2 3 ])} ]] + [[ "$packages" = ${escapeShellArg (valueToString [ hello emptyFile ])} ]] + [[ "$pathDefaultNix" = ${escapeShellArg (valueToString ./default.nix)} ]] + [[ "$emptyFile" = ${escapeShellArg (valueToString emptyFile)} ]] ) >log 2>&1 || { cat log; exit 1; } ''; }; diff --git a/pkgs/build-support/docker/default.nix b/pkgs/build-support/docker/default.nix index 033969a7fc88..ea461ccffa07 100644 --- a/pkgs/build-support/docker/default.nix +++ b/pkgs/build-support/docker/default.nix @@ -51,7 +51,7 @@ let ; inherit (devShellTools) - stringValue + valueToString ; mkDbExtraCommand = contents: @@ -1146,7 +1146,7 @@ rec { # A binary that calls the command to build the derivation builder = writeShellScriptBin "buildDerivation" '' - exec ${lib.escapeShellArg (stringValue drv.drvAttrs.builder)} ${lib.escapeShellArgs (map stringValue drv.drvAttrs.args)} + exec ${lib.escapeShellArg (valueToString drv.drvAttrs.builder)} ${lib.escapeShellArgs (map valueToString drv.drvAttrs.args)} ''; staticPath = "${dirOf shell}:${lib.makeBinPath [ builder ]}"; @@ -1180,7 +1180,7 @@ rec { # https://github.com/NixOS/nix/blob/2.8.0/src/libstore/build/local-derivation-goal.cc#L992-L1004 drvEnv = lib.mapAttrs' (name: value: - let str = stringValue value; + let str = valueToString value; in if lib.elem name (drv.drvAttrs.passAsFile or []) then lib.nameValuePair "${name}Path" (writeText "pass-as-text-${name}" str) else lib.nameValuePair name str