Fixes <https://issues.guix.gnu.org/74450>.

* guix/build/utils.scm (patch-shebang): Match an additional second potential
argument with regexp to be checked for the "env -S CMD" form.

Change-Id: Ib6f8528a83721dd3ceef34e7fceacf6274b857e5
---
 guix/build/utils.scm | 84 ++++++++++++++++++++++++++++++--------------
 1 file changed, 57 insertions(+), 27 deletions(-)

diff --git a/guix/build/utils.scm b/guix/build/utils.scm
index 94714bf397..451c6322cb 100644
--- a/guix/build/utils.scm
+++ b/guix/build/utils.scm
@@ -10,6 +10,7 @@
 ;;; Copyright © 2021, 2022 Maxime Devos <maximede...@telenet.be>
 ;;; Copyright © 2021 Brendan Tildesley <m...@brendan.scot>
 ;;; Copyright © 2023 Carlo Zancanaro <ca...@zancanaro.id.au>
+;;; Copyright © 2025 aurtzy <aur...@gmail.com>
 ;;;
 ;;; This file is part of GNU Guix.
 ;;;
@@ -1114,7 +1115,18 @@ (define (get-char* p)
     (x x)))
 
 (define patch-shebang
-  (let ((shebang-rx (make-regexp 
"^[[:blank:]]*(/[[:graph:]]+)[[:blank:]]*([[:graph:]]*)(.*)$")))
+  ;; This procedure assumes that the operating system only splits on whitespace
+  ;; immediately after the interpreter, so the rest of the string is considered
+  ;; a singular argument.  Sometimes when a package desires more than one
+  ;; argument in a shebang, "#!.../env -S CMD ..." is commonly used, so we also
+  ;; patch CMD as a special case.
+
+  ;; Whitespace-split just the first two substrings (if any) from the optional
+  ;; singular argument so we can check if the form "env -S CMD" is present and
+  ;; patch it.
+  (let ((shebang-rx
+         (make-regexp
+          
"^[[:blank:]]*(/[[:graph:]]+)[[:blank:]]*([[:graph:]]*)[[:blank:]]*([[:graph:]]*)[[:blank:]]*(.*)$")))
     (lambda* (file
               #:optional
               (path (search-path-as-string->list (getenv "PATH")))
@@ -1123,7 +1135,7 @@ (define patch-shebang
 PATH, when FILE actually starts with a shebang.  Return #t when FILE was
 patched, #f otherwise.  When KEEP-MTIME? is true, the atime/mtime of
 FILE are kept unchanged."
-      (define (patch p interpreter rest-of-line)
+      (define (patch p interpreter . arguments)
         (let* ((template (string-append file ".XXXXXX"))
                (out      (mkstemp! template))
                (st       (stat file))
@@ -1131,7 +1143,9 @@ (define patch-shebang
           (with-throw-handler #t
             (lambda ()
               (format out "#!~a~a~%"
-                      interpreter rest-of-line)
+                      interpreter
+                      (string-join (filter (negate string-null?) arguments)
+                                   " " 'prefix))
               (dump-port p out)
               (close out)
               (chmod template mode)
@@ -1146,6 +1160,9 @@ (define patch-shebang
               (false-if-exception (delete-file template))
               #f))))
 
+      (define (bin-from-path cmd-or-filepath)
+        (search-path path (basename cmd-or-filepath)))
+
       (call-with-ascii-input-file file
         (lambda (p)
           (and (eq? #\# (get-char* p))
@@ -1153,34 +1170,47 @@ (define patch-shebang
                (let ((line (false-if-exception (read-line p))))
                  (and=> (and line (regexp-exec shebang-rx line))
                         (lambda (m)
-                          (let* ((interp (match:substring m 1))
+                          (let* ((arg0 (match:substring m 1))
                                  (arg1 (match:substring m 2))
-                                 (rest (match:substring m 3))
-                                 (has-env (string-suffix? "/env" interp))
-                                 (cmd (if has-env arg1 (basename interp)))
-                                 (bin (search-path path cmd)))
-                            (if bin
-                                (if (string=? bin interp)
-                                    #f            ; nothing to do
-                                    (if has-env
+                                 (arg2 (match:substring m 3))
+                                 (rest (match:substring m 4))
+                                 (has-env? (string-suffix? "/env" arg0)))
+                            (catch 'binary-not-found
+                              (lambda ()
+                                (cond
+                                 ((and has-env? (string=? arg1 "-S"))
+                                  (let ((arg0-bin (bin-from-path "env"))
+                                        (arg2-bin (bin-from-path arg2)))
+                                    (unless arg0-bin (throw 'binary-not-found 
arg0))
+                                    (unless arg2-bin (throw 'binary-not-found 
arg2))
+                                    (format (current-error-port)
+                                            "patch-shebang: ~a: changing `~a' 
to `~a'~%"
+                                            file
+                                            (string-join (list arg0 arg1 arg2))
+                                            (string-join (list arg0-bin arg1 
arg2-bin)))
+                                    (patch p arg0-bin arg1 arg2-bin rest)))
+                                 (has-env?
+                                  (let ((arg1-bin (bin-from-path arg1)))
+                                    (unless arg1-bin (throw 'binary-not-found 
arg1))
+                                    (format (current-error-port)
+                                            "patch-shebang: ~a: changing `~a' 
to `~a'~%"
+                                            file (string-append arg0 " " arg1) 
arg1-bin)
+                                    (patch p arg1-bin arg2 rest)))
+                                 (else
+                                  (let ((arg0-bin (bin-from-path arg0)))
+                                    (unless arg0-bin (throw 'binary-not-found 
arg0))
+                                    (if (string=? arg0-bin arg0)
+                                        #f ; nothing to do
                                         (begin
                                           (format (current-error-port)
                                                   "patch-shebang: ~a: changing 
`~a' to `~a'~%"
-                                                  file (string-append interp " 
" arg1) bin)
-                                          (patch p bin rest))
-                                      (begin 
-                                        (format (current-error-port)
-                                                "patch-shebang: ~a: changing 
`~a' to `~a'~%"
-                                                file interp bin)
-                                        (patch p bin
-                                               (if (string-null? arg1)
-                                                   ""
-                                                   (string-append " " arg1 
rest))))))
-                                (begin
-                                  (format (current-error-port)
-                                          "patch-shebang: ~a: warning: no 
binary for interpreter `~a' found in $PATH~%"
-                                          file (basename cmd))
-                                  #f))))))))))))
+                                                  file arg0 arg0-bin)
+                                          (patch p arg0-bin arg1 arg2 
rest)))))))
+                              (lambda (kind cmd)
+                                (format (current-error-port)
+                                        "patch-shebang: ~a: warning: no binary 
for interpreter `~a' found in $PATH~%"
+                                        file (basename cmd))
+                                #f))))))))))))
 
 (define* (patch-makefile-SHELL file #:key (keep-mtime? #t))
   "Patch the `SHELL' variable in FILE, which is supposedly a makefile.
-- 
2.49.0




Reply via email to