nixpkgs mirror (for testing) github.com/NixOS/nixpkgs
nix
at python-updates 130 lines 5.1 kB view raw
1diff --git a/src/client/opamInitDefaults.ml b/src/client/opamInitDefaults.ml 2index eca13a7c..1fd66f43 100644 3--- a/src/client/opamInitDefaults.ml 4+++ b/src/client/opamInitDefaults.ml 5@@ -38,6 +38,9 @@ let eval_variables = 6 let os_filter os = 7 FOp (FIdent ([], OpamVariable.of_string "os", None), `Eq, FString os) 8 9+let os_distribution_filter distro = 10+ FOp (FIdent ([], OpamVariable.of_string "os-distribution", None), `Eq, FString distro) 11+ 12 let linux_filter = os_filter "linux" 13 let macos_filter = os_filter "macos" 14 let openbsd_filter = os_filter "openbsd" 15@@ -51,6 +54,7 @@ let win32_filter = os_filter "win32" 16 let not_win32_filter = 17 FOp (FIdent ([], OpamVariable.of_string "os", None), `Neq, FString "win32") 18 let sandbox_filter = FOr (linux_filter, macos_filter) 19+let nixos_filter = os_distribution_filter "nixos" 20 21 let gtar_filter = openbsd_filter 22 let tar_filter = FNot gtar_filter 23@@ -69,6 +73,11 @@ let sandbox_wrappers = 24 25 let wrappers ~sandboxing () = 26 let w = OpamFile.Wrappers.empty in 27+ let w = { w with 28+ OpamFile.Wrappers. 29+ pre_build = [[CString "%{hooks}%/shebangs.sh", None], Some nixos_filter]; 30+ } 31+ in 32 if sandboxing then 33 List.fold_left OpamFile.Wrappers.(fun w -> function 34 | `build wrap_build -> { w with wrap_build } 35@@ -147,6 +156,7 @@ let required_packages_for_cygwin = 36 let init_scripts () = [ 37 ("sandbox.sh", OpamScript.bwrap), Some bwrap_filter; 38 ("sandbox.sh", OpamScript.sandbox_exec), Some macos_filter; 39+ ("shebangs.sh", OpamScript.patch_shebangs), Some nixos_filter; 40 ] 41 42 module I = OpamFile.InitConfig 43diff --git a/src/state/opamScript.mli b/src/state/opamScript.mli 44index 03449970..83de0b53 100644 45--- a/src/state/opamScript.mli 46+++ b/src/state/opamScript.mli 47@@ -20,3 +20,4 @@ val env_hook : string 48 val env_hook_zsh : string 49 val env_hook_csh : string 50 val env_hook_fish : string 51+val patch_shebangs : string 52diff --git a/src/state/shellscripts/patch_shebangs.sh b/src/state/shellscripts/patch_shebangs.sh 53new file mode 100755 54index 00000000..3ea84e2d 55--- /dev/null 56+++ b/src/state/shellscripts/patch_shebangs.sh 57@@ -0,0 +1,73 @@ 58+#!/usr/bin/env bash 59+# This setup hook causes the fixup phase to rewrite all script 60+# interpreter file names (`#! /path') to paths found in $PATH. E.g., 61+# /bin/sh will be rewritten to /nix/store/<hash>-some-bash/bin/sh. 62+# /usr/bin/env gets special treatment so that ".../bin/env python" is 63+# rewritten to /nix/store/<hash>/bin/python. Interpreters that are 64+# already in the store are left untouched. 65+ 66+header() { echo "$1"; } 67+stopNest() { true; } 68+ 69+fixupOutputHooks+=('if [ -z "${dontPatchShebangs-}" -a -e "$prefix" ]; then patchShebangs "$prefix"; fi') 70+ 71+patchShebangs() { 72+ local dir="$1" 73+ header "patching script interpreter paths in $dir" 74+ local f 75+ local oldPath 76+ local newPath 77+ local arg0 78+ local args 79+ local oldInterpreterLine 80+ local newInterpreterLine 81+ 82+ find "$dir" -type f -perm -0100 | while read f; do 83+ if [ "$(head -1 "$f" | head -c+2)" != '#!' ]; then 84+ # missing shebang => not a script 85+ continue 86+ fi 87+ 88+ oldInterpreterLine=$(head -1 "$f" | tail -c+3) 89+ read -r oldPath arg0 args <<< "$oldInterpreterLine" 90+ 91+ if $(echo "$oldPath" | grep -q "/bin/env$"); then 92+ # Check for unsupported 'env' functionality: 93+ # - options: something starting with a '-' 94+ # - environment variables: foo=bar 95+ if $(echo "$arg0" | grep -q -- "^-.*\|.*=.*"); then 96+ echo "unsupported interpreter directive \"$oldInterpreterLine\" (set dontPatchShebangs=1 and handle shebang patching yourself)" 97+ exit 1 98+ fi 99+ newPath="$(command -v "$arg0" || true)" 100+ else 101+ if [ "$oldPath" = "" ]; then 102+ # If no interpreter is specified linux will use /bin/sh. Set 103+ # oldpath="/bin/sh" so that we get /nix/store/.../sh. 104+ oldPath="/bin/sh" 105+ fi 106+ newPath="$(command -v "$(basename "$oldPath")" || true)" 107+ args="$arg0 $args" 108+ fi 109+ 110+ # Strip trailing whitespace introduced when no arguments are present 111+ newInterpreterLine="$(echo "$newPath $args" | sed 's/[[:space:]]*$//')" 112+ 113+ if [ -n "$oldPath" -a "${oldPath:0:${#NIX_STORE}}" != "$NIX_STORE" ]; then 114+ if [ -n "$newPath" -a "$newPath" != "$oldPath" ]; then 115+ echo "$f: interpreter directive changed from \"$oldInterpreterLine\" to \"$newInterpreterLine\"" 116+ # escape the escape chars so that sed doesn't interpret them 117+ escapedInterpreterLine=$(echo "$newInterpreterLine" | sed 's|\\|\\\\|g') 118+ # Preserve times, see: https://github.com/NixOS/nixpkgs/pull/33281 119+ touch -r "$f" "$f.timestamp" 120+ sed -i -e "1 s|.*|#\!$escapedInterpreterLine|" "$f" 121+ touch -r "$f.timestamp" "$f" 122+ rm "$f.timestamp" 123+ fi 124+ fi 125+ done 126+ 127+ stopNest 128+} 129+ 130+patchShebangs .