2013-01-05 16:08:07 +01:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
2019-01-29 09:49:33 +01:00
|
|
|
|
;;; Copyright © 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019 Ludovic Courtès <ludo@gnu.org>
|
2013-02-22 23:00:41 +01:00
|
|
|
|
;;; Copyright © 2013 Andreas Enge <andreas@enge.fr>
|
2013-03-04 00:20:28 +01:00
|
|
|
|
;;; Copyright © 2013 Nikita Karetnikov <nikita@karetnikov.org>
|
2018-03-16 23:29:31 +01:00
|
|
|
|
;;; Copyright © 2015, 2018 Mark H Weaver <mhw@netris.org>
|
2018-07-11 09:33:33 +02:00
|
|
|
|
;;; Copyright © 2018 Arun Isaac <arunisaac@systemreboot.net>
|
2018-01-02 21:43:07 +01:00
|
|
|
|
;;; Copyright © 2018, 2019 Ricardo Wurmus <rekado@elephly.net>
|
2012-06-13 17:03:34 +02:00
|
|
|
|
;;;
|
2013-01-05 16:08:07 +01:00
|
|
|
|
;;; This file is part of GNU Guix.
|
2012-06-13 17:03:34 +02:00
|
|
|
|
;;;
|
2013-01-05 16:08:07 +01:00
|
|
|
|
;;; GNU Guix is free software; you can redistribute it and/or modify it
|
2012-06-13 17:03:34 +02:00
|
|
|
|
;;; under the terms of the GNU General Public License as published by
|
|
|
|
|
;;; the Free Software Foundation; either version 3 of the License, or (at
|
|
|
|
|
;;; your option) any later version.
|
|
|
|
|
;;;
|
2013-01-05 16:08:07 +01:00
|
|
|
|
;;; GNU Guix is distributed in the hope that it will be useful, but
|
2012-06-13 17:03:34 +02:00
|
|
|
|
;;; WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
;;; GNU General Public License for more details.
|
|
|
|
|
;;;
|
|
|
|
|
;;; You should have received a copy of the GNU General Public License
|
2013-01-05 16:08:07 +01:00
|
|
|
|
;;; along with GNU Guix. If not, see <http://www.gnu.org/licenses/>.
|
2012-06-13 17:03:34 +02:00
|
|
|
|
|
|
|
|
|
(define-module (guix build utils)
|
|
|
|
|
#:use-module (srfi srfi-1)
|
2012-07-01 17:32:03 +02:00
|
|
|
|
#:use-module (srfi srfi-11)
|
2015-08-30 14:59:31 +02:00
|
|
|
|
#:use-module (srfi srfi-26)
|
2018-03-16 23:29:31 +01:00
|
|
|
|
#:use-module (srfi srfi-34)
|
|
|
|
|
#:use-module (srfi srfi-35)
|
2015-02-27 14:54:00 +01:00
|
|
|
|
#:use-module (srfi srfi-60)
|
2012-10-17 23:06:17 +02:00
|
|
|
|
#:use-module (ice-9 ftw)
|
2012-07-01 17:32:03 +02:00
|
|
|
|
#:use-module (ice-9 match)
|
|
|
|
|
#:use-module (ice-9 regex)
|
|
|
|
|
#:use-module (ice-9 rdelim)
|
2014-09-14 17:54:25 +02:00
|
|
|
|
#:use-module (ice-9 format)
|
2017-09-05 19:41:31 +02:00
|
|
|
|
#:use-module (ice-9 threads)
|
2012-08-19 16:44:08 +02:00
|
|
|
|
#:use-module (rnrs bytevectors)
|
|
|
|
|
#:use-module (rnrs io ports)
|
2013-07-03 23:08:41 +02:00
|
|
|
|
#:re-export (alist-cons
|
2017-05-13 23:08:27 +02:00
|
|
|
|
alist-delete
|
|
|
|
|
|
|
|
|
|
;; Note: Re-export 'delete' to allow for proper syntax matching
|
|
|
|
|
;; in 'modify-phases' forms. See
|
|
|
|
|
;; <https://debbugs.gnu.org/cgi/bugreport.cgi?bug=26805#16>.
|
|
|
|
|
delete)
|
2014-03-10 23:51:31 +01:00
|
|
|
|
#:export (%store-directory
|
2015-04-05 15:31:55 +02:00
|
|
|
|
store-file-name?
|
2015-08-30 14:38:10 +02:00
|
|
|
|
strip-store-file-name
|
2015-08-30 14:59:31 +02:00
|
|
|
|
package-name->name+version
|
2014-11-22 12:32:32 +01:00
|
|
|
|
parallel-job-count
|
|
|
|
|
|
2014-03-10 23:51:31 +01:00
|
|
|
|
directory-exists?
|
2012-12-15 16:35:26 +01:00
|
|
|
|
executable-file?
|
2014-12-01 15:46:10 +01:00
|
|
|
|
symbolic-link?
|
2012-12-21 22:31:25 +01:00
|
|
|
|
call-with-ascii-input-file
|
2014-11-22 12:44:56 +01:00
|
|
|
|
elf-file?
|
2014-11-22 21:44:59 +01:00
|
|
|
|
ar-file?
|
2017-01-26 21:58:37 +01:00
|
|
|
|
gzip-file?
|
|
|
|
|
reset-gzip-timestamp
|
2012-07-01 17:32:03 +02:00
|
|
|
|
with-directory-excursion
|
2012-10-17 22:51:08 +02:00
|
|
|
|
mkdir-p
|
2015-08-28 22:07:05 +02:00
|
|
|
|
install-file
|
2017-01-25 16:28:03 +01:00
|
|
|
|
make-file-writable
|
2012-10-17 23:06:17 +02:00
|
|
|
|
copy-recursively
|
2013-03-05 18:53:53 +01:00
|
|
|
|
delete-file-recursively
|
2015-03-31 22:55:41 +02:00
|
|
|
|
file-name-predicate
|
2012-10-17 23:17:15 +02:00
|
|
|
|
find-files
|
2018-03-11 22:04:12 +01:00
|
|
|
|
false-if-file-not-found
|
2012-10-17 23:17:15 +02:00
|
|
|
|
|
2014-12-27 23:24:35 +01:00
|
|
|
|
search-path-as-list
|
2012-07-01 17:32:03 +02:00
|
|
|
|
set-path-environment-variable
|
2012-08-19 16:44:08 +02:00
|
|
|
|
search-path-as-string->list
|
|
|
|
|
list->search-path-as-string
|
2013-01-05 16:02:32 +01:00
|
|
|
|
which
|
|
|
|
|
|
2015-10-03 12:28:58 +02:00
|
|
|
|
every*
|
2012-07-01 17:32:03 +02:00
|
|
|
|
alist-cons-before
|
|
|
|
|
alist-cons-after
|
|
|
|
|
alist-replace
|
2015-02-26 22:48:14 +01:00
|
|
|
|
modify-phases
|
2015-10-03 12:28:58 +02:00
|
|
|
|
|
2012-10-16 17:28:11 +02:00
|
|
|
|
with-atomic-file-replacement
|
2012-07-07 16:25:10 +02:00
|
|
|
|
substitute
|
2012-08-19 16:44:08 +02:00
|
|
|
|
substitute*
|
|
|
|
|
dump-port
|
2012-12-31 01:17:43 +01:00
|
|
|
|
set-file-time
|
2012-10-16 23:01:01 +02:00
|
|
|
|
patch-shebang
|
2012-12-21 22:31:25 +01:00
|
|
|
|
patch-makefile-SHELL
|
2015-01-09 22:35:33 +01:00
|
|
|
|
patch-/usr/bin/file
|
2012-10-16 23:01:01 +02:00
|
|
|
|
fold-port-matches
|
2013-03-04 00:20:28 +01:00
|
|
|
|
remove-store-references
|
2018-07-11 09:33:33 +02:00
|
|
|
|
wrapper?
|
2015-02-27 14:54:00 +01:00
|
|
|
|
wrap-program
|
2018-01-02 21:43:07 +01:00
|
|
|
|
wrap-script
|
|
|
|
|
|
|
|
|
|
wrap-error?
|
|
|
|
|
wrap-error-program
|
|
|
|
|
wrap-error-type
|
2018-03-16 23:29:31 +01:00
|
|
|
|
|
2017-06-01 19:04:10 +02:00
|
|
|
|
invoke
|
2018-03-16 23:29:31 +01:00
|
|
|
|
invoke-error?
|
|
|
|
|
invoke-error-program
|
|
|
|
|
invoke-error-arguments
|
|
|
|
|
invoke-error-exit-status
|
|
|
|
|
invoke-error-term-signal
|
|
|
|
|
invoke-error-stop-signal
|
2019-01-29 11:00:42 +01:00
|
|
|
|
report-invoke-error
|
2015-02-27 14:54:00 +01:00
|
|
|
|
|
2019-06-17 16:06:27 +02:00
|
|
|
|
invoke/quiet
|
|
|
|
|
|
2020-01-30 16:53:44 +01:00
|
|
|
|
make-desktop-entry-file
|
|
|
|
|
|
2015-02-27 14:54:00 +01:00
|
|
|
|
locale-category->string))
|
2012-07-01 17:32:03 +02:00
|
|
|
|
|
2019-01-29 09:49:33 +01:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Guile 2.0 compatibility later.
|
|
|
|
|
;;;
|
2013-02-22 23:00:41 +01:00
|
|
|
|
|
2019-01-29 09:49:33 +01:00
|
|
|
|
;; The bootstrap Guile is Guile 2.0, so provide a compatibility layer.
|
|
|
|
|
(cond-expand
|
|
|
|
|
((and guile-2 (not guile-2.2))
|
|
|
|
|
(define (setvbuf port mode . rest)
|
|
|
|
|
(apply (@ (guile) setvbuf) port
|
|
|
|
|
(match mode
|
|
|
|
|
('line _IOLBF)
|
|
|
|
|
('block _IOFBF)
|
|
|
|
|
('none _IONBF)
|
|
|
|
|
(_ mode)) ;an _IO* integer
|
|
|
|
|
rest))
|
|
|
|
|
|
|
|
|
|
(module-replace! (current-module) '(setvbuf)))
|
|
|
|
|
(else #f))
|
|
|
|
|
|
|
|
|
|
|
2012-07-01 17:32:03 +02:00
|
|
|
|
;;;
|
|
|
|
|
;;; Directories.
|
|
|
|
|
;;;
|
2012-06-13 17:03:34 +02:00
|
|
|
|
|
2014-03-10 23:51:31 +01:00
|
|
|
|
(define (%store-directory)
|
|
|
|
|
"Return the directory name of the store."
|
|
|
|
|
(or (getenv "NIX_STORE")
|
|
|
|
|
"/gnu/store"))
|
|
|
|
|
|
2015-04-05 15:31:55 +02:00
|
|
|
|
(define (store-file-name? file)
|
|
|
|
|
"Return true if FILE is in the store."
|
|
|
|
|
(string-prefix? (%store-directory) file))
|
|
|
|
|
|
2015-08-30 14:38:10 +02:00
|
|
|
|
(define (strip-store-file-name file)
|
|
|
|
|
"Strip the '/gnu/store' and hash from FILE, a store file name. The result
|
|
|
|
|
is typically a \"PACKAGE-VERSION\" string."
|
|
|
|
|
(string-drop file
|
|
|
|
|
(+ 34 (string-length (%store-directory)))))
|
|
|
|
|
|
2015-08-30 14:59:31 +02:00
|
|
|
|
(define (package-name->name+version name)
|
|
|
|
|
"Given NAME, a package name like \"foo-0.9.1b\", return two values:
|
|
|
|
|
\"foo\" and \"0.9.1b\". When the version part is unavailable, NAME and
|
|
|
|
|
#f are returned. The first hyphen followed by a digit is considered to
|
|
|
|
|
introduce the version part."
|
|
|
|
|
;; See also `DrvName' in Nix.
|
|
|
|
|
|
|
|
|
|
(define number?
|
|
|
|
|
(cut char-set-contains? char-set:digit <>))
|
|
|
|
|
|
|
|
|
|
(let loop ((chars (string->list name))
|
|
|
|
|
(prefix '()))
|
|
|
|
|
(match chars
|
|
|
|
|
(()
|
|
|
|
|
(values name #f))
|
|
|
|
|
((#\- (? number? n) rest ...)
|
|
|
|
|
(values (list->string (reverse prefix))
|
|
|
|
|
(list->string (cons n rest))))
|
|
|
|
|
((head tail ...)
|
|
|
|
|
(loop tail (cons head prefix))))))
|
|
|
|
|
|
2014-11-22 21:57:05 +01:00
|
|
|
|
(define parallel-job-count
|
|
|
|
|
;; Number of processes to be passed next to GNU Make's `-j' argument.
|
|
|
|
|
(make-parameter
|
|
|
|
|
(match (getenv "NIX_BUILD_CORES") ;set by the daemon
|
|
|
|
|
(#f 1)
|
|
|
|
|
("0" (current-processor-count))
|
|
|
|
|
(x (or (string->number x) 1)))))
|
2014-11-22 12:32:32 +01:00
|
|
|
|
|
2012-06-13 17:03:34 +02:00
|
|
|
|
(define (directory-exists? dir)
|
|
|
|
|
"Return #t if DIR exists and is a directory."
|
2012-06-16 16:16:16 +02:00
|
|
|
|
(let ((s (stat dir #f)))
|
|
|
|
|
(and s
|
|
|
|
|
(eq? 'directory (stat:type s)))))
|
2012-06-13 17:03:34 +02:00
|
|
|
|
|
2012-12-15 16:35:26 +01:00
|
|
|
|
(define (executable-file? file)
|
|
|
|
|
"Return #t if FILE exists and is executable."
|
|
|
|
|
(let ((s (stat file #f)))
|
|
|
|
|
(and s
|
|
|
|
|
(not (zero? (logand (stat:mode s) #o100))))))
|
|
|
|
|
|
2014-12-01 15:46:10 +01:00
|
|
|
|
(define (symbolic-link? file)
|
|
|
|
|
"Return #t if FILE is a symbolic link (aka. \"symlink\".)"
|
|
|
|
|
(eq? (stat:type (lstat file)) 'symlink))
|
|
|
|
|
|
2012-12-21 22:31:25 +01:00
|
|
|
|
(define (call-with-ascii-input-file file proc)
|
|
|
|
|
"Open FILE as an ASCII or binary file, and pass the resulting port to
|
|
|
|
|
PROC. FILE is closed when PROC's dynamic extent is left. Return the
|
|
|
|
|
return values of applying PROC to the port."
|
|
|
|
|
(let ((port (with-fluids ((%default-port-encoding #f))
|
|
|
|
|
;; Use "b" so that `open-file' ignores `coding:' cookies.
|
|
|
|
|
(open-file file "rb"))))
|
|
|
|
|
(dynamic-wind
|
|
|
|
|
(lambda ()
|
|
|
|
|
#t)
|
|
|
|
|
(lambda ()
|
|
|
|
|
(proc port))
|
|
|
|
|
(lambda ()
|
|
|
|
|
(close-input-port port)))))
|
|
|
|
|
|
2014-11-22 21:52:57 +01:00
|
|
|
|
(define (file-header-match header)
|
|
|
|
|
"Return a procedure that returns true when its argument is a file starting
|
|
|
|
|
with the bytes in HEADER, a bytevector."
|
|
|
|
|
(define len
|
|
|
|
|
(bytevector-length header))
|
2014-11-22 12:44:56 +01:00
|
|
|
|
|
2014-11-22 21:52:57 +01:00
|
|
|
|
(lambda (file)
|
|
|
|
|
"Return true if FILE starts with the right magic bytes."
|
|
|
|
|
(define (get-header)
|
|
|
|
|
(call-with-input-file file
|
|
|
|
|
(lambda (port)
|
|
|
|
|
(get-bytevector-n port len))
|
|
|
|
|
#:binary #t #:guess-encoding #f))
|
|
|
|
|
|
2014-11-23 19:15:21 +01:00
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(equal? (get-header) header))
|
|
|
|
|
(lambda args
|
|
|
|
|
(if (= EISDIR (system-error-errno args))
|
|
|
|
|
#f ;FILE is a directory
|
|
|
|
|
(apply throw args))))))
|
2014-11-22 21:52:57 +01:00
|
|
|
|
|
|
|
|
|
(define %elf-magic-bytes
|
|
|
|
|
;; Magic bytes of ELF files. See <elf.h>.
|
|
|
|
|
(u8-list->bytevector (map char->integer (string->list "\x7FELF"))))
|
|
|
|
|
|
|
|
|
|
(define elf-file?
|
|
|
|
|
(file-header-match %elf-magic-bytes))
|
2014-11-22 12:44:56 +01:00
|
|
|
|
|
2014-11-22 21:44:59 +01:00
|
|
|
|
(define %ar-magic-bytes
|
|
|
|
|
;; Magic bytes of archives created by 'ar'. See <ar.h>.
|
|
|
|
|
(u8-list->bytevector (map char->integer (string->list "!<arch>\n"))))
|
|
|
|
|
|
2014-11-22 21:52:57 +01:00
|
|
|
|
(define ar-file?
|
|
|
|
|
(file-header-match %ar-magic-bytes))
|
2014-11-22 21:44:59 +01:00
|
|
|
|
|
2017-01-26 21:58:37 +01:00
|
|
|
|
(define %gzip-magic-bytes
|
|
|
|
|
;; Magic bytes of gzip file. Beware, it's a small header so there could be
|
|
|
|
|
;; false positives.
|
|
|
|
|
#vu8(#x1f #x8b))
|
|
|
|
|
|
|
|
|
|
(define gzip-file?
|
|
|
|
|
(file-header-match %gzip-magic-bytes))
|
|
|
|
|
|
|
|
|
|
(define* (reset-gzip-timestamp file #:key (keep-mtime? #t))
|
|
|
|
|
"If FILE is a gzip file, reset its embedded timestamp (as with 'gzip
|
|
|
|
|
--no-name') and return true. Otherwise return #f. When KEEP-MTIME? is true,
|
|
|
|
|
preserve FILE's modification time."
|
|
|
|
|
(let ((stat (stat file))
|
|
|
|
|
(port (open file O_RDWR)))
|
|
|
|
|
(dynamic-wind
|
|
|
|
|
(const #t)
|
|
|
|
|
(lambda ()
|
|
|
|
|
(and (= 4 (seek port 4 SEEK_SET))
|
|
|
|
|
(put-bytevector port #vu8(0 0 0 0))))
|
|
|
|
|
(lambda ()
|
|
|
|
|
(close-port port)
|
|
|
|
|
(set-file-time file stat)))))
|
|
|
|
|
|
2012-07-01 17:32:03 +02:00
|
|
|
|
(define-syntax-rule (with-directory-excursion dir body ...)
|
|
|
|
|
"Run BODY with DIR as the process's current directory."
|
|
|
|
|
(let ((init (getcwd)))
|
|
|
|
|
(dynamic-wind
|
|
|
|
|
(lambda ()
|
|
|
|
|
(chdir dir))
|
|
|
|
|
(lambda ()
|
|
|
|
|
body ...)
|
|
|
|
|
(lambda ()
|
|
|
|
|
(chdir init)))))
|
|
|
|
|
|
2012-10-17 22:51:08 +02:00
|
|
|
|
(define (mkdir-p dir)
|
|
|
|
|
"Create directory DIR and all its ancestors."
|
|
|
|
|
(define absolute?
|
|
|
|
|
(string-prefix? "/" dir))
|
|
|
|
|
|
|
|
|
|
(define not-slash
|
|
|
|
|
(char-set-complement (char-set #\/)))
|
|
|
|
|
|
|
|
|
|
(let loop ((components (string-tokenize dir not-slash))
|
|
|
|
|
(root (if absolute?
|
|
|
|
|
""
|
|
|
|
|
".")))
|
|
|
|
|
(match components
|
|
|
|
|
((head tail ...)
|
|
|
|
|
(let ((path (string-append root "/" head)))
|
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(mkdir path)
|
|
|
|
|
(loop tail path))
|
|
|
|
|
(lambda args
|
|
|
|
|
(if (= EEXIST (system-error-errno args))
|
|
|
|
|
(loop tail path)
|
|
|
|
|
(apply throw args))))))
|
|
|
|
|
(() #t))))
|
|
|
|
|
|
2015-08-28 22:07:05 +02:00
|
|
|
|
(define (install-file file directory)
|
|
|
|
|
"Create DIRECTORY if it does not exist and copy FILE in there under the same
|
|
|
|
|
name."
|
|
|
|
|
(mkdir-p directory)
|
|
|
|
|
(copy-file file (string-append directory "/" (basename file))))
|
|
|
|
|
|
2017-01-25 16:28:03 +01:00
|
|
|
|
(define (make-file-writable file)
|
|
|
|
|
"Make FILE writable for its owner."
|
|
|
|
|
(let ((stat (lstat file))) ;XXX: symlinks
|
|
|
|
|
(chmod file (logior #o600 (stat:perms stat)))))
|
|
|
|
|
|
2012-10-17 23:06:17 +02:00
|
|
|
|
(define* (copy-recursively source destination
|
2013-03-05 19:03:39 +01:00
|
|
|
|
#:key
|
|
|
|
|
(log (current-output-port))
|
2014-04-14 00:08:54 +02:00
|
|
|
|
(follow-symlinks? #f)
|
|
|
|
|
keep-mtime?)
|
2013-03-05 19:03:39 +01:00
|
|
|
|
"Copy SOURCE directory to DESTINATION. Follow symlinks if FOLLOW-SYMLINKS?
|
2014-04-14 00:08:54 +02:00
|
|
|
|
is true; otherwise, just preserve them. When KEEP-MTIME? is true, keep the
|
|
|
|
|
modification time of the files in SOURCE on those of DESTINATION. Write
|
|
|
|
|
verbose output to the LOG port."
|
2012-10-17 23:06:17 +02:00
|
|
|
|
(define strip-source
|
|
|
|
|
(let ((len (string-length source)))
|
|
|
|
|
(lambda (file)
|
|
|
|
|
(substring file len))))
|
|
|
|
|
|
|
|
|
|
(file-system-fold (const #t) ; enter?
|
|
|
|
|
(lambda (file stat result) ; leaf
|
|
|
|
|
(let ((dest (string-append destination
|
|
|
|
|
(strip-source file))))
|
|
|
|
|
(format log "`~a' -> `~a'~%" file dest)
|
2013-03-05 19:03:39 +01:00
|
|
|
|
(case (stat:type stat)
|
|
|
|
|
((symlink)
|
|
|
|
|
(let ((target (readlink file)))
|
|
|
|
|
(symlink target dest)))
|
|
|
|
|
(else
|
2014-04-14 00:08:54 +02:00
|
|
|
|
(copy-file file dest)
|
|
|
|
|
(when keep-mtime?
|
|
|
|
|
(set-file-time dest stat))))))
|
2012-10-17 23:06:17 +02:00
|
|
|
|
(lambda (dir stat result) ; down
|
2014-04-14 00:08:54 +02:00
|
|
|
|
(let ((target (string-append destination
|
|
|
|
|
(strip-source dir))))
|
|
|
|
|
(mkdir-p target)
|
|
|
|
|
(when keep-mtime?
|
|
|
|
|
(set-file-time target stat))))
|
2012-10-17 23:06:17 +02:00
|
|
|
|
(lambda (dir stat result) ; up
|
|
|
|
|
result)
|
|
|
|
|
(const #t) ; skip
|
|
|
|
|
(lambda (file stat errno result)
|
|
|
|
|
(format (current-error-port) "i/o error: ~a: ~a~%"
|
|
|
|
|
file (strerror errno))
|
|
|
|
|
#f)
|
|
|
|
|
#t
|
2013-03-05 19:03:39 +01:00
|
|
|
|
source
|
|
|
|
|
|
|
|
|
|
(if follow-symlinks?
|
|
|
|
|
stat
|
|
|
|
|
lstat)))
|
2012-10-17 23:06:17 +02:00
|
|
|
|
|
2014-05-20 14:45:58 +02:00
|
|
|
|
(define* (delete-file-recursively dir
|
|
|
|
|
#:key follow-mounts?)
|
|
|
|
|
"Delete DIR recursively, like `rm -rf', without following symlinks. Don't
|
|
|
|
|
follow mount points either, unless FOLLOW-MOUNTS? is true. Report but ignore
|
|
|
|
|
errors."
|
|
|
|
|
(let ((dev (stat:dev (lstat dir))))
|
|
|
|
|
(file-system-fold (lambda (dir stat result) ; enter?
|
|
|
|
|
(or follow-mounts?
|
|
|
|
|
(= dev (stat:dev stat))))
|
|
|
|
|
(lambda (file stat result) ; leaf
|
|
|
|
|
(delete-file file))
|
|
|
|
|
(const #t) ; down
|
|
|
|
|
(lambda (dir stat result) ; up
|
|
|
|
|
(rmdir dir))
|
|
|
|
|
(const #t) ; skip
|
|
|
|
|
(lambda (file stat errno result)
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"warning: failed to delete ~a: ~a~%"
|
|
|
|
|
file (strerror errno)))
|
|
|
|
|
#t
|
|
|
|
|
dir
|
|
|
|
|
|
|
|
|
|
;; Don't follow symlinks.
|
|
|
|
|
lstat)))
|
2013-03-05 18:53:53 +01:00
|
|
|
|
|
2015-03-31 22:55:41 +02:00
|
|
|
|
(define (file-name-predicate regexp)
|
|
|
|
|
"Return a predicate that returns true when passed a file name whose base
|
|
|
|
|
name matches REGEXP."
|
|
|
|
|
(let ((file-rx (if (regexp? regexp)
|
|
|
|
|
regexp
|
|
|
|
|
(make-regexp regexp))))
|
|
|
|
|
(lambda (file stat)
|
|
|
|
|
(regexp-exec file-rx (basename file)))))
|
|
|
|
|
|
2015-04-06 23:09:54 +02:00
|
|
|
|
(define* (find-files dir #:optional (pred (const #t))
|
2015-09-06 18:24:08 +02:00
|
|
|
|
#:key (stat lstat)
|
|
|
|
|
directories?
|
|
|
|
|
fail-on-error?)
|
2015-03-31 22:55:41 +02:00
|
|
|
|
"Return the lexicographically sorted list of files under DIR for which PRED
|
|
|
|
|
returns true. PRED is passed two arguments: the absolute file name, and its
|
2015-04-01 15:43:54 +02:00
|
|
|
|
stat buffer; the default predicate always returns true. PRED can also be a
|
|
|
|
|
regular expression, in which case it is equivalent to (file-name-predicate
|
2015-04-06 23:09:54 +02:00
|
|
|
|
PRED). STAT is used to obtain file information; using 'lstat' means that
|
2015-09-06 18:24:08 +02:00
|
|
|
|
symlinks are not followed. If DIRECTORIES? is true, then directories will
|
|
|
|
|
also be included. If FAIL-ON-ERROR? is true, raise an exception upon error."
|
2015-03-31 22:55:41 +02:00
|
|
|
|
(let ((pred (if (procedure? pred)
|
|
|
|
|
pred
|
|
|
|
|
(file-name-predicate pred))))
|
|
|
|
|
;; Sort the result to get deterministic results.
|
|
|
|
|
(sort (file-system-fold (const #t)
|
|
|
|
|
(lambda (file stat result) ; leaf
|
|
|
|
|
(if (pred file stat)
|
|
|
|
|
(cons file result)
|
|
|
|
|
result))
|
|
|
|
|
(lambda (dir stat result) ; down
|
2015-09-06 18:24:08 +02:00
|
|
|
|
(if (and directories?
|
|
|
|
|
(pred dir stat))
|
|
|
|
|
(cons dir result)
|
|
|
|
|
result))
|
2015-03-31 22:55:41 +02:00
|
|
|
|
(lambda (dir stat result) ; up
|
|
|
|
|
result)
|
|
|
|
|
(lambda (file stat result) ; skip
|
|
|
|
|
result)
|
|
|
|
|
(lambda (file stat errno result)
|
|
|
|
|
(format (current-error-port) "find-files: ~a: ~a~%"
|
|
|
|
|
file (strerror errno))
|
2015-09-06 18:24:08 +02:00
|
|
|
|
(when fail-on-error?
|
|
|
|
|
(error "find-files failed"))
|
2015-03-31 22:55:41 +02:00
|
|
|
|
result)
|
|
|
|
|
'()
|
2015-04-06 23:09:54 +02:00
|
|
|
|
dir
|
|
|
|
|
stat)
|
2015-03-31 22:55:41 +02:00
|
|
|
|
string<?)))
|
2012-10-17 23:17:15 +02:00
|
|
|
|
|
2018-03-11 22:04:12 +01:00
|
|
|
|
(define-syntax-rule (false-if-file-not-found exp)
|
|
|
|
|
"Evaluate EXP but return #f if it raises to 'system-error with ENOENT."
|
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda () exp)
|
|
|
|
|
(lambda args
|
|
|
|
|
(if (= ENOENT (system-error-errno args))
|
|
|
|
|
#f
|
|
|
|
|
(apply throw args)))))
|
|
|
|
|
|
2012-07-01 17:32:03 +02:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Search paths.
|
|
|
|
|
;;;
|
|
|
|
|
|
2014-12-27 12:16:18 +01:00
|
|
|
|
(define* (search-path-as-list files input-dirs
|
2014-12-27 22:55:34 +01:00
|
|
|
|
#:key (type 'directory) pattern)
|
2014-12-27 12:16:18 +01:00
|
|
|
|
"Return the list of directories among FILES of the given TYPE (a symbol as
|
|
|
|
|
returned by 'stat:type') that exist in INPUT-DIRS. Example:
|
2012-06-13 17:03:34 +02:00
|
|
|
|
|
|
|
|
|
(search-path-as-list '(\"share/emacs/site-lisp\" \"share/emacs/24.1\")
|
|
|
|
|
(list \"/package1\" \"/package2\" \"/package3\"))
|
|
|
|
|
=> (\"/package1/share/emacs/site-lisp\"
|
|
|
|
|
\"/package3/share/emacs/site-lisp\")
|
|
|
|
|
|
2014-12-27 22:55:34 +01:00
|
|
|
|
When PATTERN is true, it is a regular expression denoting file names to look
|
|
|
|
|
for under the directories designated by FILES. For example:
|
|
|
|
|
|
|
|
|
|
(search-path-as-list '(\"xml\") (list docbook-xml docbook-xsl)
|
|
|
|
|
#:type 'regular
|
|
|
|
|
#:pattern \"^catalog\\\\.xml$\")
|
|
|
|
|
=> (\"/…/xml/dtd/docbook/catalog.xml\"
|
|
|
|
|
\"/…/xml/xsl/docbook-xsl-1.78.1/catalog.xml\")
|
2012-06-13 17:03:34 +02:00
|
|
|
|
"
|
|
|
|
|
(append-map (lambda (input)
|
2014-12-27 22:55:34 +01:00
|
|
|
|
(append-map (lambda (file)
|
|
|
|
|
(let ((file (string-append input "/" file)))
|
|
|
|
|
(if pattern
|
2015-10-06 04:38:19 +02:00
|
|
|
|
(find-files file (lambda (file stat)
|
|
|
|
|
(and stat
|
|
|
|
|
(eq? type (stat:type stat))
|
|
|
|
|
((file-name-predicate pattern) file stat)))
|
|
|
|
|
#:stat stat
|
|
|
|
|
#:directories? #t)
|
2014-12-27 22:55:34 +01:00
|
|
|
|
(let ((stat (stat file #f)))
|
|
|
|
|
(if (and stat (eq? type (stat:type stat)))
|
|
|
|
|
(list file)
|
|
|
|
|
'())))))
|
2014-12-27 12:16:18 +01:00
|
|
|
|
files))
|
2015-02-04 16:48:39 +01:00
|
|
|
|
(delete-duplicates input-dirs)))
|
2012-06-13 17:03:34 +02:00
|
|
|
|
|
|
|
|
|
(define (list->search-path-as-string lst separator)
|
2017-01-22 22:42:57 +01:00
|
|
|
|
(if separator
|
|
|
|
|
(string-join lst separator)
|
|
|
|
|
(match lst
|
|
|
|
|
((head rest ...) head)
|
|
|
|
|
(() ""))))
|
2012-06-13 17:03:34 +02:00
|
|
|
|
|
2012-08-19 16:44:08 +02:00
|
|
|
|
(define* (search-path-as-string->list path #:optional (separator #\:))
|
2017-01-22 22:42:57 +01:00
|
|
|
|
(if separator
|
|
|
|
|
(string-tokenize path
|
|
|
|
|
(char-set-complement (char-set separator)))
|
|
|
|
|
(list path)))
|
2012-08-19 16:44:08 +02:00
|
|
|
|
|
2014-12-27 12:16:18 +01:00
|
|
|
|
(define* (set-path-environment-variable env-var files input-dirs
|
|
|
|
|
#:key
|
|
|
|
|
(separator ":")
|
2014-12-27 22:55:34 +01:00
|
|
|
|
(type 'directory)
|
|
|
|
|
pattern)
|
2014-12-27 12:16:18 +01:00
|
|
|
|
"Look for each of FILES of the given TYPE (a symbol as returned by
|
|
|
|
|
'stat:type') in INPUT-DIRS. Set ENV-VAR to a SEPARATOR-separated path
|
|
|
|
|
accordingly. Example:
|
2012-06-13 17:03:34 +02:00
|
|
|
|
|
|
|
|
|
(set-path-environment-variable \"PKG_CONFIG\"
|
|
|
|
|
'(\"lib/pkgconfig\")
|
|
|
|
|
(list package1 package2))
|
2014-12-27 22:55:34 +01:00
|
|
|
|
|
|
|
|
|
When PATTERN is not #f, it must be a regular expression (really a string)
|
|
|
|
|
denoting file names to look for under the directories designated by FILES:
|
|
|
|
|
|
|
|
|
|
(set-path-environment-variable \"XML_CATALOG_FILES\"
|
|
|
|
|
'(\"xml\")
|
|
|
|
|
(list docbook-xml docbook-xsl)
|
|
|
|
|
#:type 'regular
|
|
|
|
|
#:pattern \"^catalog\\\\.xml$\")
|
2012-06-13 17:03:34 +02:00
|
|
|
|
"
|
2014-12-27 12:16:18 +01:00
|
|
|
|
(let* ((path (search-path-as-list files input-dirs
|
2014-12-27 22:55:34 +01:00
|
|
|
|
#:type type
|
|
|
|
|
#:pattern pattern))
|
2012-09-06 22:57:46 +02:00
|
|
|
|
(value (list->search-path-as-string path separator)))
|
2013-06-22 16:42:46 +02:00
|
|
|
|
(if (string-null? value)
|
|
|
|
|
(begin
|
|
|
|
|
;; Never set ENV-VAR to an empty string because often, the empty
|
|
|
|
|
;; string is equivalent to ".". This is the case for
|
|
|
|
|
;; GUILE_LOAD_PATH in Guile 2.0, for instance.
|
|
|
|
|
(unsetenv env-var)
|
|
|
|
|
(format #t "environment variable `~a' unset~%" env-var))
|
|
|
|
|
(begin
|
|
|
|
|
(setenv env-var value)
|
|
|
|
|
(format #t "environment variable `~a' set to `~a'~%"
|
|
|
|
|
env-var value)))))
|
2012-07-01 17:32:03 +02:00
|
|
|
|
|
2013-01-05 16:02:32 +01:00
|
|
|
|
(define (which program)
|
|
|
|
|
"Return the complete file name for PROGRAM as found in $PATH, or #f if
|
|
|
|
|
PROGRAM could not be found."
|
|
|
|
|
(search-path (search-path-as-string->list (getenv "PATH"))
|
|
|
|
|
program))
|
|
|
|
|
|
2012-07-01 17:32:03 +02:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Phases.
|
|
|
|
|
;;;
|
|
|
|
|
;;; In (guix build gnu-build-system), there are separate phases (configure,
|
|
|
|
|
;;; build, test, install). They are represented as a list of name/procedure
|
|
|
|
|
;;; pairs. The following procedures make it easy to change the list of
|
|
|
|
|
;;; phases.
|
|
|
|
|
;;;
|
|
|
|
|
|
2015-10-03 12:28:58 +02:00
|
|
|
|
(define (every* pred lst)
|
|
|
|
|
"This is like 'every', but process all the elements of LST instead of
|
|
|
|
|
stopping as soon as PRED returns false. This is useful when PRED has side
|
|
|
|
|
effects, such as displaying warnings or error messages."
|
|
|
|
|
(let loop ((lst lst)
|
|
|
|
|
(result #t))
|
|
|
|
|
(match lst
|
|
|
|
|
(()
|
|
|
|
|
result)
|
|
|
|
|
((head . tail)
|
|
|
|
|
(loop tail (and (pred head) result))))))
|
|
|
|
|
|
2012-07-01 17:32:03 +02:00
|
|
|
|
(define* (alist-cons-before reference key value alist
|
|
|
|
|
#:optional (key=? equal?))
|
|
|
|
|
"Insert the KEY/VALUE pair before the first occurrence of a pair whose key
|
|
|
|
|
is REFERENCE in ALIST. Use KEY=? to compare keys."
|
|
|
|
|
(let-values (((before after)
|
|
|
|
|
(break (match-lambda
|
|
|
|
|
((k . _)
|
|
|
|
|
(key=? k reference)))
|
|
|
|
|
alist)))
|
|
|
|
|
(append before (alist-cons key value after))))
|
|
|
|
|
|
|
|
|
|
(define* (alist-cons-after reference key value alist
|
|
|
|
|
#:optional (key=? equal?))
|
|
|
|
|
"Insert the KEY/VALUE pair after the first occurrence of a pair whose key
|
|
|
|
|
is REFERENCE in ALIST. Use KEY=? to compare keys."
|
|
|
|
|
(let-values (((before after)
|
|
|
|
|
(break (match-lambda
|
|
|
|
|
((k . _)
|
|
|
|
|
(key=? k reference)))
|
|
|
|
|
alist)))
|
|
|
|
|
(match after
|
|
|
|
|
((reference after ...)
|
|
|
|
|
(append before (cons* reference `(,key . ,value) after)))
|
|
|
|
|
(()
|
|
|
|
|
(append before `((,key . ,value)))))))
|
|
|
|
|
|
|
|
|
|
(define* (alist-replace key value alist #:optional (key=? equal?))
|
|
|
|
|
"Replace the first pair in ALIST whose car is KEY with the KEY/VALUE pair.
|
|
|
|
|
An error is raised when no such pair exists."
|
|
|
|
|
(let-values (((before after)
|
|
|
|
|
(break (match-lambda
|
|
|
|
|
((k . _)
|
|
|
|
|
(key=? k key)))
|
|
|
|
|
alist)))
|
|
|
|
|
(match after
|
|
|
|
|
((_ after ...)
|
|
|
|
|
(append before (alist-cons key value after))))))
|
|
|
|
|
|
2015-02-26 22:48:14 +01:00
|
|
|
|
(define-syntax-rule (modify-phases phases mod-spec ...)
|
|
|
|
|
"Modify PHASES sequentially as per each MOD-SPEC, which may have one of the
|
|
|
|
|
following forms:
|
|
|
|
|
|
|
|
|
|
(delete <old-phase-name>)
|
|
|
|
|
(replace <old-phase-name> <new-phase>)
|
|
|
|
|
(add-before <old-phase-name> <new-phase-name> <new-phase>)
|
|
|
|
|
(add-after <old-phase-name> <new-phase-name> <new-phase>)
|
|
|
|
|
|
2016-06-20 23:29:12 +02:00
|
|
|
|
Where every <*-phase-name> is an expression evaluating to a symbol, and
|
|
|
|
|
<new-phase> an expression evaluating to a procedure."
|
2015-02-26 22:48:14 +01:00
|
|
|
|
(let* ((phases* phases)
|
|
|
|
|
(phases* (%modify-phases phases* mod-spec))
|
|
|
|
|
...)
|
|
|
|
|
phases*))
|
|
|
|
|
|
|
|
|
|
(define-syntax %modify-phases
|
|
|
|
|
(syntax-rules (delete replace add-before add-after)
|
|
|
|
|
((_ phases (delete old-phase-name))
|
utils: 'modify-phases' no longer introduces quotes.
Suggested by Taylan Ulrich Bayırlı/Kammer <taylanbayirli@gmail.com>.
* guix/build/utils.scm (%modify-phases): Remove quotes.
* guix/build/cmake-build-system.scm (%standard-phases): Adjust
accordingly.
* guix/build/glib-or-gtk-build-system.scm (%standard-phases): Likewise.
* guix/build/gnu-dist.scm (%dist-phases): Likewise.
* guix/build/perl-build-system.scm (%standard-phases): Likewise.
* guix/build/python-build-system.scm (%standard-phases): Likewise.
* guix/build/ruby-build-system.scm (%standard-phases): Likewise.
* guix/build/waf-build-system.scm (%standard-phases): Likewise.
* gnu/packages/bash.scm, gnu/packages/code.scm, gnu/packages/gl.scm,
gnu/packages/gnome.scm, gnu/packages/graphics.scm,
gnu/packages/image.scm, gnu/packages/key-mon.scm,
gnu/packages/ocr.scm, gnu/packages/plotutils.scm,
gnu/packages/search.scm, gnu/packages/video.scm: Likewise.
2015-03-31 22:43:01 +02:00
|
|
|
|
(alist-delete old-phase-name phases))
|
2015-02-26 22:48:14 +01:00
|
|
|
|
((_ phases (replace old-phase-name new-phase))
|
utils: 'modify-phases' no longer introduces quotes.
Suggested by Taylan Ulrich Bayırlı/Kammer <taylanbayirli@gmail.com>.
* guix/build/utils.scm (%modify-phases): Remove quotes.
* guix/build/cmake-build-system.scm (%standard-phases): Adjust
accordingly.
* guix/build/glib-or-gtk-build-system.scm (%standard-phases): Likewise.
* guix/build/gnu-dist.scm (%dist-phases): Likewise.
* guix/build/perl-build-system.scm (%standard-phases): Likewise.
* guix/build/python-build-system.scm (%standard-phases): Likewise.
* guix/build/ruby-build-system.scm (%standard-phases): Likewise.
* guix/build/waf-build-system.scm (%standard-phases): Likewise.
* gnu/packages/bash.scm, gnu/packages/code.scm, gnu/packages/gl.scm,
gnu/packages/gnome.scm, gnu/packages/graphics.scm,
gnu/packages/image.scm, gnu/packages/key-mon.scm,
gnu/packages/ocr.scm, gnu/packages/plotutils.scm,
gnu/packages/search.scm, gnu/packages/video.scm: Likewise.
2015-03-31 22:43:01 +02:00
|
|
|
|
(alist-replace old-phase-name new-phase phases))
|
2015-02-26 22:48:14 +01:00
|
|
|
|
((_ phases (add-before old-phase-name new-phase-name new-phase))
|
utils: 'modify-phases' no longer introduces quotes.
Suggested by Taylan Ulrich Bayırlı/Kammer <taylanbayirli@gmail.com>.
* guix/build/utils.scm (%modify-phases): Remove quotes.
* guix/build/cmake-build-system.scm (%standard-phases): Adjust
accordingly.
* guix/build/glib-or-gtk-build-system.scm (%standard-phases): Likewise.
* guix/build/gnu-dist.scm (%dist-phases): Likewise.
* guix/build/perl-build-system.scm (%standard-phases): Likewise.
* guix/build/python-build-system.scm (%standard-phases): Likewise.
* guix/build/ruby-build-system.scm (%standard-phases): Likewise.
* guix/build/waf-build-system.scm (%standard-phases): Likewise.
* gnu/packages/bash.scm, gnu/packages/code.scm, gnu/packages/gl.scm,
gnu/packages/gnome.scm, gnu/packages/graphics.scm,
gnu/packages/image.scm, gnu/packages/key-mon.scm,
gnu/packages/ocr.scm, gnu/packages/plotutils.scm,
gnu/packages/search.scm, gnu/packages/video.scm: Likewise.
2015-03-31 22:43:01 +02:00
|
|
|
|
(alist-cons-before old-phase-name new-phase-name new-phase phases))
|
2015-02-26 22:48:14 +01:00
|
|
|
|
((_ phases (add-after old-phase-name new-phase-name new-phase))
|
utils: 'modify-phases' no longer introduces quotes.
Suggested by Taylan Ulrich Bayırlı/Kammer <taylanbayirli@gmail.com>.
* guix/build/utils.scm (%modify-phases): Remove quotes.
* guix/build/cmake-build-system.scm (%standard-phases): Adjust
accordingly.
* guix/build/glib-or-gtk-build-system.scm (%standard-phases): Likewise.
* guix/build/gnu-dist.scm (%dist-phases): Likewise.
* guix/build/perl-build-system.scm (%standard-phases): Likewise.
* guix/build/python-build-system.scm (%standard-phases): Likewise.
* guix/build/ruby-build-system.scm (%standard-phases): Likewise.
* guix/build/waf-build-system.scm (%standard-phases): Likewise.
* gnu/packages/bash.scm, gnu/packages/code.scm, gnu/packages/gl.scm,
gnu/packages/gnome.scm, gnu/packages/graphics.scm,
gnu/packages/image.scm, gnu/packages/key-mon.scm,
gnu/packages/ocr.scm, gnu/packages/plotutils.scm,
gnu/packages/search.scm, gnu/packages/video.scm: Likewise.
2015-03-31 22:43:01 +02:00
|
|
|
|
(alist-cons-after old-phase-name new-phase-name new-phase phases))))
|
2015-02-26 22:48:14 +01:00
|
|
|
|
|
2019-01-29 11:00:42 +01:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Program invocation.
|
|
|
|
|
;;;
|
|
|
|
|
|
2018-03-16 23:29:31 +01:00
|
|
|
|
(define-condition-type &invoke-error &error
|
|
|
|
|
invoke-error?
|
|
|
|
|
(program invoke-error-program)
|
|
|
|
|
(arguments invoke-error-arguments)
|
|
|
|
|
(exit-status invoke-error-exit-status)
|
|
|
|
|
(term-signal invoke-error-term-signal)
|
|
|
|
|
(stop-signal invoke-error-stop-signal))
|
|
|
|
|
|
2017-06-01 19:04:10 +02:00
|
|
|
|
(define (invoke program . args)
|
2018-03-16 23:29:31 +01:00
|
|
|
|
"Invoke PROGRAM with the given ARGS. Raise an exception
|
|
|
|
|
if the exit code is non-zero; otherwise return #t."
|
|
|
|
|
(let ((code (apply system* program args)))
|
|
|
|
|
(unless (zero? code)
|
|
|
|
|
(raise (condition (&invoke-error
|
|
|
|
|
(program program)
|
|
|
|
|
(arguments args)
|
|
|
|
|
(exit-status (status:exit-val code))
|
|
|
|
|
(term-signal (status:term-sig code))
|
|
|
|
|
(stop-signal (status:stop-sig code))))))
|
2017-06-01 19:04:10 +02:00
|
|
|
|
#t))
|
|
|
|
|
|
2019-01-29 11:00:42 +01:00
|
|
|
|
(define* (report-invoke-error c #:optional (port (current-error-port)))
|
|
|
|
|
"Report to PORT about C, an '&invoke-error' condition, in a human-friendly
|
|
|
|
|
way."
|
|
|
|
|
(format port "command~{ ~s~} failed with ~:[signal~;status~] ~a~%"
|
|
|
|
|
(cons (invoke-error-program c)
|
|
|
|
|
(invoke-error-arguments c))
|
|
|
|
|
(invoke-error-exit-status c)
|
|
|
|
|
(or (invoke-error-exit-status c)
|
|
|
|
|
(invoke-error-term-signal c)
|
|
|
|
|
(invoke-error-stop-signal c))))
|
|
|
|
|
|
2019-06-17 16:06:27 +02:00
|
|
|
|
(define (open-pipe-with-stderr program . args)
|
|
|
|
|
"Run PROGRAM with ARGS in an input pipe, but, unlike 'open-pipe*', redirect
|
|
|
|
|
both its standard output and standard error to the pipe. Return two value:
|
|
|
|
|
the pipe to read PROGRAM's data from, and the PID of the child process running
|
|
|
|
|
PROGRAM."
|
|
|
|
|
;; 'open-pipe*' doesn't attempt to capture stderr in any way, which is why
|
|
|
|
|
;; we need to roll our own.
|
|
|
|
|
(match (pipe)
|
|
|
|
|
((input . output)
|
|
|
|
|
(match (primitive-fork)
|
|
|
|
|
(0
|
|
|
|
|
(dynamic-wind
|
|
|
|
|
(const #t)
|
|
|
|
|
(lambda ()
|
|
|
|
|
(close-port input)
|
|
|
|
|
(dup2 (fileno output) 1)
|
|
|
|
|
(dup2 (fileno output) 2)
|
|
|
|
|
(apply execlp program program args))
|
|
|
|
|
(lambda ()
|
|
|
|
|
(primitive-exit 127))))
|
|
|
|
|
(pid
|
|
|
|
|
(close-port output)
|
|
|
|
|
(values input pid))))))
|
|
|
|
|
|
|
|
|
|
(define (invoke/quiet program . args)
|
|
|
|
|
"Invoke PROGRAM with ARGS and capture PROGRAM's standard output and standard
|
|
|
|
|
error. If PROGRAM succeeds, print nothing and return the unspecified value;
|
|
|
|
|
otherwise, raise a '&message' error condition that includes the status code
|
|
|
|
|
and the output of PROGRAM."
|
|
|
|
|
(let-values (((pipe pid)
|
|
|
|
|
(apply open-pipe-with-stderr program args)))
|
|
|
|
|
(let loop ((lines '()))
|
|
|
|
|
(match (read-line pipe)
|
|
|
|
|
((? eof-object?)
|
|
|
|
|
(close-port pipe)
|
|
|
|
|
(match (waitpid pid)
|
|
|
|
|
((_ . status)
|
|
|
|
|
(unless (zero? status)
|
|
|
|
|
(let-syntax ((G_ (syntax-rules () ;for xgettext
|
|
|
|
|
((_ str) str))))
|
|
|
|
|
(raise (condition
|
|
|
|
|
(&message
|
|
|
|
|
(message (format #f (G_ "'~a~{ ~a~}' exited \
|
|
|
|
|
with status ~a; output follows:~%~%~{ ~a~%~}")
|
|
|
|
|
program args
|
|
|
|
|
(or (status:exit-val status)
|
|
|
|
|
status)
|
|
|
|
|
(reverse lines)))))))))))
|
|
|
|
|
(line
|
|
|
|
|
(loop (cons line lines)))))))
|
|
|
|
|
|
2012-07-01 17:32:03 +02:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Text substitution (aka. sed).
|
|
|
|
|
;;;
|
|
|
|
|
|
2012-10-16 17:28:11 +02:00
|
|
|
|
(define (with-atomic-file-replacement file proc)
|
|
|
|
|
"Call PROC with two arguments: an input port for FILE, and an output
|
|
|
|
|
port for the file that is going to replace FILE. Upon success, FILE is
|
|
|
|
|
atomically replaced by what has been written to the output port, and
|
|
|
|
|
PROC's result is returned."
|
|
|
|
|
(let* ((template (string-append file ".XXXXXX"))
|
2012-07-07 18:11:52 +02:00
|
|
|
|
(out (mkstemp! template))
|
|
|
|
|
(mode (stat:mode (stat file))))
|
2012-07-01 17:32:03 +02:00
|
|
|
|
(with-throw-handler #t
|
|
|
|
|
(lambda ()
|
|
|
|
|
(call-with-input-file file
|
|
|
|
|
(lambda (in)
|
2012-10-16 17:28:11 +02:00
|
|
|
|
(let ((result (proc in out)))
|
|
|
|
|
(close out)
|
|
|
|
|
(chmod template mode)
|
|
|
|
|
(rename-file template file)
|
|
|
|
|
result))))
|
2012-07-01 17:32:03 +02:00
|
|
|
|
(lambda (key . args)
|
|
|
|
|
(false-if-exception (delete-file template))))))
|
|
|
|
|
|
2012-10-16 17:28:11 +02:00
|
|
|
|
(define (substitute file pattern+procs)
|
2014-11-25 21:52:13 +01:00
|
|
|
|
"PATTERN+PROCS is a list of regexp/two-argument-procedure pairs. For each
|
|
|
|
|
line of FILE, and for each PATTERN that it matches, call the corresponding
|
|
|
|
|
PROC as (PROC LINE MATCHES); PROC must return the line that will be written as
|
|
|
|
|
a substitution of the original line. Be careful about using '$' to match the
|
|
|
|
|
end of a line; by itself it won't match the terminating newline of a line."
|
2012-10-16 17:28:11 +02:00
|
|
|
|
(let ((rx+proc (map (match-lambda
|
|
|
|
|
(((? regexp? pattern) . proc)
|
|
|
|
|
(cons pattern proc))
|
|
|
|
|
((pattern . proc)
|
|
|
|
|
(cons (make-regexp pattern regexp/extended)
|
|
|
|
|
proc)))
|
|
|
|
|
pattern+procs)))
|
|
|
|
|
(with-atomic-file-replacement file
|
|
|
|
|
(lambda (in out)
|
|
|
|
|
(let loop ((line (read-line in 'concat)))
|
|
|
|
|
(if (eof-object? line)
|
|
|
|
|
#t
|
|
|
|
|
(let ((line (fold (lambda (r+p line)
|
|
|
|
|
(match r+p
|
|
|
|
|
((regexp . proc)
|
|
|
|
|
(match (list-matches regexp line)
|
|
|
|
|
((and m+ (_ _ ...))
|
|
|
|
|
(proc line m+))
|
|
|
|
|
(_ line)))))
|
|
|
|
|
line
|
|
|
|
|
rx+proc)))
|
|
|
|
|
(display line out)
|
|
|
|
|
(loop (read-line in 'concat)))))))))
|
|
|
|
|
|
2012-07-07 16:25:10 +02:00
|
|
|
|
|
|
|
|
|
(define-syntax let-matches
|
|
|
|
|
;; Helper macro for `substitute*'.
|
|
|
|
|
(syntax-rules (_)
|
|
|
|
|
((let-matches index match (_ vars ...) body ...)
|
|
|
|
|
(let-matches (+ 1 index) match (vars ...)
|
|
|
|
|
body ...))
|
|
|
|
|
((let-matches index match (var vars ...) body ...)
|
|
|
|
|
(let ((var (match:substring match index)))
|
|
|
|
|
(let-matches (+ 1 index) match (vars ...)
|
|
|
|
|
body ...)))
|
|
|
|
|
((let-matches index match () body ...)
|
|
|
|
|
(begin body ...))))
|
|
|
|
|
|
2012-08-25 13:12:33 +02:00
|
|
|
|
(define-syntax substitute*
|
|
|
|
|
(syntax-rules ()
|
|
|
|
|
"Substitute REGEXP in FILE by the string returned by BODY. BODY is
|
2012-07-07 16:25:10 +02:00
|
|
|
|
evaluated with each MATCH-VAR bound to the corresponding positional regexp
|
|
|
|
|
sub-expression. For example:
|
|
|
|
|
|
2012-07-07 17:12:04 +02:00
|
|
|
|
(substitute* file
|
2012-10-26 09:07:37 +02:00
|
|
|
|
((\"hello\")
|
|
|
|
|
\"good morning\\n\")
|
|
|
|
|
((\"foo([a-z]+)bar(.*)$\" all letters end)
|
|
|
|
|
(string-append \"baz\" letter end)))
|
2012-07-07 17:12:04 +02:00
|
|
|
|
|
|
|
|
|
Here, anytime a line of FILE contains \"hello\", it is replaced by \"good
|
|
|
|
|
morning\". Anytime a line of FILE matches the second regexp, ALL is bound to
|
|
|
|
|
the complete match, LETTERS is bound to the first sub-expression, and END is
|
|
|
|
|
bound to the last one.
|
|
|
|
|
|
|
|
|
|
When one of the MATCH-VAR is `_', no variable is bound to the corresponding
|
2012-10-26 09:07:37 +02:00
|
|
|
|
match substring.
|
|
|
|
|
|
|
|
|
|
Alternatively, FILE may be a list of file names, in which case they are
|
2014-11-25 21:52:13 +01:00
|
|
|
|
all subject to the substitutions.
|
|
|
|
|
|
|
|
|
|
Be careful about using '$' to match the end of a line; by itself it won't
|
|
|
|
|
match the terminating newline of a line."
|
2012-08-25 13:12:33 +02:00
|
|
|
|
((substitute* file ((regexp match-var ...) body ...) ...)
|
2012-10-26 09:07:37 +02:00
|
|
|
|
(let ()
|
|
|
|
|
(define (substitute-one-file file-name)
|
|
|
|
|
(substitute
|
|
|
|
|
file-name
|
|
|
|
|
(list (cons regexp
|
|
|
|
|
(lambda (l m+)
|
|
|
|
|
;; Iterate over matches M+ and return the
|
|
|
|
|
;; modified line based on L.
|
|
|
|
|
(let loop ((m* m+) ; matches
|
|
|
|
|
(o 0) ; offset in L
|
|
|
|
|
(r '())) ; result
|
|
|
|
|
(match m*
|
|
|
|
|
(()
|
|
|
|
|
(let ((r (cons (substring l o) r)))
|
|
|
|
|
(string-concatenate-reverse r)))
|
|
|
|
|
((m . rest)
|
|
|
|
|
(let-matches 0 m (match-var ...)
|
|
|
|
|
(loop rest
|
|
|
|
|
(match:end m)
|
|
|
|
|
(cons*
|
|
|
|
|
(begin body ...)
|
|
|
|
|
(substring l o (match:start m))
|
|
|
|
|
r))))))))
|
|
|
|
|
...)))
|
|
|
|
|
|
|
|
|
|
(match file
|
|
|
|
|
((files (... ...))
|
|
|
|
|
(for-each substitute-one-file files))
|
|
|
|
|
((? string? f)
|
|
|
|
|
(substitute-one-file f)))))))
|
2012-07-07 16:25:10 +02:00
|
|
|
|
|
2012-08-19 16:44:08 +02:00
|
|
|
|
|
|
|
|
|
;;;
|
2014-03-10 23:51:31 +01:00
|
|
|
|
;;; Patching shebangs---e.g., /bin/sh -> /gnu/store/xyz...-bash/bin/sh.
|
2012-08-19 16:44:08 +02:00
|
|
|
|
;;;
|
|
|
|
|
|
2012-12-20 01:34:42 +01:00
|
|
|
|
(define* (dump-port in out
|
|
|
|
|
#:key (buffer-size 16384)
|
|
|
|
|
(progress (lambda (t k) (k))))
|
2015-02-27 14:56:01 +01:00
|
|
|
|
"Read as much data as possible from IN and write it to OUT, using chunks of
|
|
|
|
|
BUFFER-SIZE bytes. Call PROGRESS at the beginning and after each successful
|
|
|
|
|
transfer of BUFFER-SIZE bytes or less, passing it the total number of bytes
|
|
|
|
|
transferred and the continuation of the transfer as a thunk."
|
2012-08-19 16:44:08 +02:00
|
|
|
|
(define buffer
|
|
|
|
|
(make-bytevector buffer-size))
|
|
|
|
|
|
2015-02-27 14:56:01 +01:00
|
|
|
|
(define (loop total bytes)
|
2012-08-19 16:44:08 +02:00
|
|
|
|
(or (eof-object? bytes)
|
2012-12-20 01:34:42 +01:00
|
|
|
|
(let ((total (+ total bytes)))
|
2012-08-19 16:44:08 +02:00
|
|
|
|
(put-bytevector out buffer 0 bytes)
|
2012-12-20 01:34:42 +01:00
|
|
|
|
(progress total
|
|
|
|
|
(lambda ()
|
|
|
|
|
(loop total
|
2015-02-27 14:56:01 +01:00
|
|
|
|
(get-bytevector-n! in buffer 0 buffer-size)))))))
|
|
|
|
|
|
|
|
|
|
;; Make sure PROGRESS is called when we start so that it can measure
|
|
|
|
|
;; throughput.
|
|
|
|
|
(progress 0
|
|
|
|
|
(lambda ()
|
|
|
|
|
(loop 0 (get-bytevector-n! in buffer 0 buffer-size)))))
|
2012-08-19 16:44:08 +02:00
|
|
|
|
|
2012-12-31 01:17:43 +01:00
|
|
|
|
(define (set-file-time file stat)
|
|
|
|
|
"Set the atime/mtime of FILE to that specified by STAT."
|
|
|
|
|
(utime file
|
|
|
|
|
(stat:atime stat)
|
|
|
|
|
(stat:mtime stat)
|
|
|
|
|
(stat:atimensec stat)
|
|
|
|
|
(stat:mtimensec stat)))
|
|
|
|
|
|
2015-02-28 01:01:51 +01:00
|
|
|
|
(define (get-char* p)
|
|
|
|
|
;; We call it `get-char', but that's really a binary version
|
|
|
|
|
;; thereof. (The real `get-char' cannot be used here because our
|
|
|
|
|
;; bootstrap Guile is hacked to always use UTF-8.)
|
|
|
|
|
(match (get-u8 p)
|
|
|
|
|
((? integer? x) (integer->char x))
|
|
|
|
|
(x x)))
|
|
|
|
|
|
2012-08-19 16:44:08 +02:00
|
|
|
|
(define patch-shebang
|
2020-02-17 16:52:55 +01:00
|
|
|
|
(let ((shebang-rx (make-regexp "^[[:blank:]]*(/[[:graph:]]+)[[:blank:]]*([[:graph:]]*)(.*)$")))
|
2012-08-19 21:50:03 +02:00
|
|
|
|
(lambda* (file
|
2012-12-31 01:17:43 +01:00
|
|
|
|
#:optional
|
|
|
|
|
(path (search-path-as-string->list (getenv "PATH")))
|
|
|
|
|
#:key (keep-mtime? #t))
|
2012-08-19 21:50:03 +02:00
|
|
|
|
"Replace the #! interpreter file name in FILE by a valid one found in
|
|
|
|
|
PATH, when FILE actually starts with a shebang. Return #t when FILE was
|
2012-12-31 01:17:43 +01:00
|
|
|
|
patched, #f otherwise. When KEEP-MTIME? is true, the atime/mtime of
|
|
|
|
|
FILE are kept unchanged."
|
2012-08-19 16:44:08 +02:00
|
|
|
|
(define (patch p interpreter rest-of-line)
|
|
|
|
|
(let* ((template (string-append file ".XXXXXX"))
|
|
|
|
|
(out (mkstemp! template))
|
2012-12-31 01:17:43 +01:00
|
|
|
|
(st (stat file))
|
|
|
|
|
(mode (stat:mode st)))
|
2012-08-19 16:44:08 +02:00
|
|
|
|
(with-throw-handler #t
|
|
|
|
|
(lambda ()
|
|
|
|
|
(format out "#!~a~a~%"
|
|
|
|
|
interpreter rest-of-line)
|
|
|
|
|
(dump-port p out)
|
|
|
|
|
(close out)
|
|
|
|
|
(chmod template mode)
|
|
|
|
|
(rename-file template file)
|
2012-12-31 01:17:43 +01:00
|
|
|
|
(when keep-mtime?
|
|
|
|
|
(set-file-time file st))
|
2012-08-19 16:44:08 +02:00
|
|
|
|
#t)
|
|
|
|
|
(lambda (key . args)
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"patch-shebang: ~a: error: ~a ~s~%"
|
|
|
|
|
file key args)
|
|
|
|
|
(false-if-exception (delete-file template))
|
|
|
|
|
#f))))
|
|
|
|
|
|
2012-12-21 22:31:25 +01:00
|
|
|
|
(call-with-ascii-input-file file
|
|
|
|
|
(lambda (p)
|
2015-02-28 01:01:51 +01:00
|
|
|
|
(and (eq? #\# (get-char* p))
|
|
|
|
|
(eq? #\! (get-char* p))
|
2012-12-21 22:31:25 +01:00
|
|
|
|
(let ((line (false-if-exception (read-line p))))
|
|
|
|
|
(and=> (and line (regexp-exec shebang-rx line))
|
|
|
|
|
(lambda (m)
|
2013-02-22 23:00:41 +01:00
|
|
|
|
(let* ((interp (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)))
|
2012-12-21 22:31:25 +01:00
|
|
|
|
(if bin
|
2013-02-22 23:00:41 +01:00
|
|
|
|
(if (string=? bin interp)
|
2012-12-21 22:31:25 +01:00
|
|
|
|
#f ; nothing to do
|
2013-02-22 23:00:41 +01:00
|
|
|
|
(if has-env
|
|
|
|
|
(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
|
2013-02-23 23:27:46 +01:00
|
|
|
|
(if (string-null? arg1)
|
|
|
|
|
""
|
|
|
|
|
(string-append " " arg1 rest))))))
|
2012-12-21 22:31:25 +01:00
|
|
|
|
(begin
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"patch-shebang: ~a: warning: no binary for interpreter `~a' found in $PATH~%"
|
|
|
|
|
file (basename cmd))
|
|
|
|
|
#f))))))))))))
|
|
|
|
|
|
2012-12-31 01:17:43 +01:00
|
|
|
|
(define* (patch-makefile-SHELL file #:key (keep-mtime? #t))
|
|
|
|
|
"Patch the `SHELL' variable in FILE, which is supposedly a makefile.
|
|
|
|
|
When KEEP-MTIME? is true, the atime/mtime of FILE are kept unchanged."
|
2012-12-21 22:31:25 +01:00
|
|
|
|
|
|
|
|
|
;; For instance, Gettext-generated po/Makefile.in.in do not honor $SHELL.
|
|
|
|
|
|
|
|
|
|
;; XXX: Unlike with `patch-shebang', FILE is always touched.
|
|
|
|
|
|
|
|
|
|
(define (find-shell name)
|
2014-12-13 00:04:25 +01:00
|
|
|
|
(let ((shell (which name)))
|
2012-12-21 22:31:25 +01:00
|
|
|
|
(unless shell
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"patch-makefile-SHELL: warning: no binary for shell `~a' found in $PATH~%"
|
|
|
|
|
name))
|
|
|
|
|
shell))
|
|
|
|
|
|
2012-12-31 01:17:43 +01:00
|
|
|
|
(let ((st (stat file)))
|
2015-02-28 12:25:22 +01:00
|
|
|
|
;; Consider FILE is using an 8-bit encoding to avoid errors.
|
|
|
|
|
(with-fluids ((%default-port-encoding #f))
|
|
|
|
|
(substitute* file
|
|
|
|
|
(("^ *SHELL[[:blank:]]*:?=[[:blank:]]*([[:graph:]]*/)([[:graph:]]+)(.*)$"
|
|
|
|
|
_ dir shell args)
|
|
|
|
|
(let* ((old (string-append dir shell))
|
|
|
|
|
(new (or (find-shell shell) old)))
|
|
|
|
|
(unless (string=? new old)
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"patch-makefile-SHELL: ~a: changing `SHELL' from `~a' to `~a'~%"
|
|
|
|
|
file old new))
|
|
|
|
|
(string-append "SHELL = " new args)))))
|
2012-12-31 01:17:43 +01:00
|
|
|
|
|
|
|
|
|
(when keep-mtime?
|
|
|
|
|
(set-file-time file st))))
|
2012-07-07 16:25:10 +02:00
|
|
|
|
|
2015-01-09 22:35:33 +01:00
|
|
|
|
(define* (patch-/usr/bin/file file
|
|
|
|
|
#:key
|
|
|
|
|
(file-command (which "file"))
|
|
|
|
|
(keep-mtime? #t))
|
|
|
|
|
"Patch occurrences of \"/usr/bin/file\" in FILE, replacing them with
|
|
|
|
|
FILE-COMMAND. When KEEP-MTIME? is true, keep FILE's modification time
|
|
|
|
|
unchanged."
|
|
|
|
|
(if (not file-command)
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"patch-/usr/bin/file: warning: \
|
|
|
|
|
no replacement 'file' command, doing nothing~%")
|
|
|
|
|
(let ((st (stat file)))
|
2015-02-28 12:25:22 +01:00
|
|
|
|
;; Consider FILE is using an 8-bit encoding to avoid errors.
|
|
|
|
|
(with-fluids ((%default-port-encoding #f))
|
|
|
|
|
(substitute* file
|
|
|
|
|
(("/usr/bin/file")
|
|
|
|
|
(begin
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"patch-/usr/bin/file: ~a: changing `~a' to `~a'~%"
|
|
|
|
|
file "/usr/bin/file" file-command)
|
|
|
|
|
file-command))))
|
2015-01-09 22:35:33 +01:00
|
|
|
|
|
|
|
|
|
(when keep-mtime?
|
|
|
|
|
(set-file-time file st)))))
|
|
|
|
|
|
2012-10-16 23:01:01 +02:00
|
|
|
|
(define* (fold-port-matches proc init pattern port
|
|
|
|
|
#:optional (unmatched (lambda (_ r) r)))
|
|
|
|
|
"Read from PORT character-by-character; for each match against
|
|
|
|
|
PATTERN, call (PROC MATCH RESULT), where RESULT is seeded with INIT.
|
|
|
|
|
PATTERN is a list of SRFI-14 char-sets. Call (UNMATCHED CHAR RESULT)
|
|
|
|
|
for each unmatched character."
|
|
|
|
|
(define initial-pattern
|
|
|
|
|
;; The poor developer's regexp.
|
|
|
|
|
(if (string? pattern)
|
|
|
|
|
(map char-set (string->list pattern))
|
|
|
|
|
pattern))
|
|
|
|
|
|
|
|
|
|
;; Note: we're not really striving for performance here...
|
|
|
|
|
(let loop ((chars '())
|
|
|
|
|
(pattern initial-pattern)
|
|
|
|
|
(matched '())
|
|
|
|
|
(result init))
|
|
|
|
|
(cond ((null? chars)
|
2015-02-28 01:01:51 +01:00
|
|
|
|
(loop (list (get-char* port))
|
2012-10-16 23:01:01 +02:00
|
|
|
|
pattern
|
|
|
|
|
matched
|
|
|
|
|
result))
|
|
|
|
|
((null? pattern)
|
|
|
|
|
(loop chars
|
|
|
|
|
initial-pattern
|
|
|
|
|
'()
|
|
|
|
|
(proc (list->string (reverse matched)) result)))
|
|
|
|
|
((eof-object? (car chars))
|
|
|
|
|
(fold-right unmatched result matched))
|
|
|
|
|
((char-set-contains? (car pattern) (car chars))
|
|
|
|
|
(loop (cdr chars)
|
|
|
|
|
(cdr pattern)
|
|
|
|
|
(cons (car chars) matched)
|
|
|
|
|
result))
|
|
|
|
|
((null? matched) ; common case
|
|
|
|
|
(loop (cdr chars)
|
|
|
|
|
pattern
|
|
|
|
|
matched
|
|
|
|
|
(unmatched (car chars) result)))
|
|
|
|
|
(else
|
|
|
|
|
(let ((matched (reverse matched)))
|
|
|
|
|
(loop (append (cdr matched) chars)
|
|
|
|
|
initial-pattern
|
|
|
|
|
'()
|
|
|
|
|
(unmatched (car matched) result)))))))
|
|
|
|
|
|
|
|
|
|
(define* (remove-store-references file
|
2014-03-10 23:51:31 +01:00
|
|
|
|
#:optional (store (%store-directory)))
|
2012-10-16 23:01:01 +02:00
|
|
|
|
"Remove from FILE occurrences of file names in STORE; return #t when
|
|
|
|
|
store paths were encountered in FILE, #f otherwise. This procedure is
|
|
|
|
|
known as `nuke-refs' in Nixpkgs."
|
|
|
|
|
(define pattern
|
|
|
|
|
(let ((nix-base32-chars
|
|
|
|
|
'(#\0 #\1 #\2 #\3 #\4 #\5 #\6 #\7 #\8 #\9
|
|
|
|
|
#\a #\b #\c #\d #\f #\g #\h #\i #\j #\k #\l #\m #\n
|
|
|
|
|
#\p #\q #\r #\s #\v #\w #\x #\y #\z)))
|
|
|
|
|
`(,@(map char-set (string->list store))
|
|
|
|
|
,(char-set #\/)
|
|
|
|
|
,@(make-list 32 (list->char-set nix-base32-chars))
|
|
|
|
|
,(char-set #\-))))
|
|
|
|
|
|
|
|
|
|
(with-fluids ((%default-port-encoding #f))
|
|
|
|
|
(with-atomic-file-replacement file
|
|
|
|
|
(lambda (in out)
|
|
|
|
|
;; We cannot use `regexp-exec' here because it cannot deal with
|
|
|
|
|
;; strings containing NUL characters.
|
|
|
|
|
(format #t "removing store references from `~a'...~%" file)
|
2019-01-29 09:49:33 +01:00
|
|
|
|
(setvbuf in 'block 65536)
|
|
|
|
|
(setvbuf out 'block 65536)
|
2012-10-16 23:01:01 +02:00
|
|
|
|
(fold-port-matches (lambda (match result)
|
2013-01-01 23:12:34 +01:00
|
|
|
|
(put-bytevector out (string->utf8 store))
|
|
|
|
|
(put-u8 out (char->integer #\/))
|
|
|
|
|
(put-bytevector out
|
|
|
|
|
(string->utf8
|
|
|
|
|
"eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee-"))
|
2012-10-16 23:01:01 +02:00
|
|
|
|
#t)
|
|
|
|
|
#f
|
|
|
|
|
pattern
|
|
|
|
|
in
|
|
|
|
|
(lambda (char result)
|
2013-01-01 23:12:34 +01:00
|
|
|
|
(put-u8 out (char->integer char))
|
2012-10-16 23:01:01 +02:00
|
|
|
|
result))))))
|
|
|
|
|
|
2018-01-02 21:43:07 +01:00
|
|
|
|
(define-condition-type &wrap-error &error
|
|
|
|
|
wrap-error?
|
|
|
|
|
(program wrap-error-program)
|
|
|
|
|
(type wrap-error-type))
|
|
|
|
|
|
2018-07-11 09:33:33 +02:00
|
|
|
|
(define (wrapper? prog)
|
|
|
|
|
"Return #t if PROG is a wrapper as produced by 'wrap-program'."
|
|
|
|
|
(and (file-exists? prog)
|
|
|
|
|
(let ((base (basename prog)))
|
|
|
|
|
(and (string-prefix? "." base)
|
|
|
|
|
(string-suffix? "-real" base)))))
|
|
|
|
|
|
2013-03-04 00:20:28 +01:00
|
|
|
|
(define* (wrap-program prog #:rest vars)
|
2014-09-13 08:05:03 +02:00
|
|
|
|
"Make a wrapper for PROG. VARS should look like this:
|
2013-03-04 00:20:28 +01:00
|
|
|
|
|
|
|
|
|
'(VARIABLE DELIMITER POSITION LIST-OF-DIRECTORIES)
|
|
|
|
|
|
|
|
|
|
where DELIMITER is optional. ':' will be used if DELIMITER is not given.
|
|
|
|
|
|
|
|
|
|
For example, this command:
|
|
|
|
|
|
|
|
|
|
(wrap-program \"foo\"
|
2014-09-13 08:05:03 +02:00
|
|
|
|
'(\"PATH\" \":\" = (\"/gnu/.../bar/bin\"))
|
|
|
|
|
'(\"CERT_PATH\" suffix (\"/gnu/.../baz/certs\"
|
2013-03-04 00:20:28 +01:00
|
|
|
|
\"/qux/certs\")))
|
|
|
|
|
|
|
|
|
|
will copy 'foo' to '.foo-real' and create the file 'foo' with the following
|
|
|
|
|
contents:
|
|
|
|
|
|
|
|
|
|
#!location/of/bin/bash
|
2014-09-13 08:05:03 +02:00
|
|
|
|
export PATH=\"/gnu/.../bar/bin\"
|
|
|
|
|
export CERT_PATH=\"$CERT_PATH${CERT_PATH:+:}/gnu/.../baz/certs:/qux/certs\"
|
2015-02-15 17:14:53 +01:00
|
|
|
|
exec -a $0 location/of/.foo-real \"$@\"
|
2013-03-04 00:20:28 +01:00
|
|
|
|
|
|
|
|
|
This is useful for scripts that expect particular programs to be in $PATH, for
|
|
|
|
|
programs that expect particular shared libraries to be in $LD_LIBRARY_PATH, or
|
2014-09-13 08:05:03 +02:00
|
|
|
|
modules in $GUILE_LOAD_PATH, etc.
|
|
|
|
|
|
2016-09-07 23:59:02 +02:00
|
|
|
|
If PROG has previously been wrapped by 'wrap-program', the wrapper is extended
|
|
|
|
|
with definitions for VARS."
|
|
|
|
|
(define wrapped-file
|
|
|
|
|
(string-append (dirname prog) "/." (basename prog) "-real"))
|
|
|
|
|
|
|
|
|
|
(define already-wrapped?
|
|
|
|
|
(file-exists? wrapped-file))
|
|
|
|
|
|
|
|
|
|
(define (last-line port)
|
|
|
|
|
;; Return the last line read from PORT and leave PORT's cursor right
|
|
|
|
|
;; before it.
|
|
|
|
|
(let loop ((previous-line-offset 0)
|
|
|
|
|
(previous-line "")
|
|
|
|
|
(position (seek port 0 SEEK_CUR)))
|
|
|
|
|
(match (read-line port 'concat)
|
|
|
|
|
((? eof-object?)
|
|
|
|
|
(seek port previous-line-offset SEEK_SET)
|
|
|
|
|
previous-line)
|
|
|
|
|
((? string? line)
|
|
|
|
|
(loop position line (+ (string-length line) position))))))
|
|
|
|
|
|
|
|
|
|
(define (export-variable lst)
|
|
|
|
|
;; Return a string that exports an environment variable.
|
|
|
|
|
(match lst
|
|
|
|
|
((var sep '= rest)
|
|
|
|
|
(format #f "export ~a=\"~a\""
|
|
|
|
|
var (string-join rest sep)))
|
|
|
|
|
((var sep 'prefix rest)
|
2018-08-20 16:51:04 +02:00
|
|
|
|
(format #f "export ~a=\"~a${~a:+~a}$~a\""
|
|
|
|
|
var (string-join rest sep) var sep var))
|
2016-09-07 23:59:02 +02:00
|
|
|
|
((var sep 'suffix rest)
|
2018-08-20 16:51:04 +02:00
|
|
|
|
(format #f "export ~a=\"$~a${~a+~a}~a\""
|
|
|
|
|
var var var sep (string-join rest sep)))
|
2016-09-07 23:59:02 +02:00
|
|
|
|
((var '= rest)
|
|
|
|
|
(format #f "export ~a=\"~a\""
|
|
|
|
|
var (string-join rest ":")))
|
|
|
|
|
((var 'prefix rest)
|
|
|
|
|
(format #f "export ~a=\"~a${~a:+:}$~a\""
|
|
|
|
|
var (string-join rest ":") var var))
|
|
|
|
|
((var 'suffix rest)
|
|
|
|
|
(format #f "export ~a=\"$~a${~a:+:}~a\""
|
|
|
|
|
var var var (string-join rest ":")))))
|
|
|
|
|
|
|
|
|
|
(if already-wrapped?
|
|
|
|
|
|
|
|
|
|
;; PROG is already a wrapper: add the new "export VAR=VALUE" lines just
|
|
|
|
|
;; before the last line.
|
|
|
|
|
(let* ((port (open-file prog "r+"))
|
|
|
|
|
(last (last-line port)))
|
|
|
|
|
(for-each (lambda (var)
|
|
|
|
|
(display (export-variable var) port)
|
|
|
|
|
(newline port))
|
|
|
|
|
vars)
|
|
|
|
|
(display last port)
|
|
|
|
|
(close-port port))
|
|
|
|
|
|
|
|
|
|
;; PROG is not wrapped yet: create a shell script that sets VARS.
|
|
|
|
|
(let ((prog-tmp (string-append wrapped-file "-tmp")))
|
|
|
|
|
(link prog wrapped-file)
|
|
|
|
|
|
|
|
|
|
(call-with-output-file prog-tmp
|
|
|
|
|
(lambda (port)
|
|
|
|
|
(format port
|
|
|
|
|
"#!~a~%~a~%exec -a \"$0\" \"~a\" \"$@\"~%"
|
|
|
|
|
(which "bash")
|
|
|
|
|
(string-join (map export-variable vars) "\n")
|
|
|
|
|
(canonicalize-path wrapped-file))))
|
|
|
|
|
|
|
|
|
|
(chmod prog-tmp #o755)
|
|
|
|
|
(rename-file prog-tmp prog))))
|
2013-03-04 00:20:28 +01:00
|
|
|
|
|
2018-01-02 21:43:07 +01:00
|
|
|
|
(define wrap-script
|
|
|
|
|
(let ((interpreter-regex
|
|
|
|
|
(make-regexp
|
|
|
|
|
(string-append "^#! ?(/[^ ]+/bin/("
|
|
|
|
|
(string-join '("python[^ ]*"
|
|
|
|
|
"Rscript"
|
|
|
|
|
"perl"
|
|
|
|
|
"ruby"
|
|
|
|
|
"bash"
|
|
|
|
|
"sh") "|")
|
|
|
|
|
"))( ?.*)")))
|
|
|
|
|
(coding-line-regex
|
|
|
|
|
(make-regexp
|
|
|
|
|
".*#.*coding[=:][[:space:]]*([-a-zA-Z_0-9.]+)")))
|
|
|
|
|
(lambda* (prog #:key (guile (which "guile")) #:rest vars)
|
|
|
|
|
"Wrap the script PROG such that VARS are set first. The format of VARS
|
|
|
|
|
is the same as in the WRAP-PROGRAM procedure. This procedure differs from
|
|
|
|
|
WRAP-PROGRAM in that it does not create a separate shell script. Instead,
|
|
|
|
|
PROG is modified directly by prepending a Guile script, which is interpreted
|
|
|
|
|
as a comment in the script's language.
|
|
|
|
|
|
|
|
|
|
Special encoding comments as supported by Python are recreated on the second
|
|
|
|
|
line.
|
|
|
|
|
|
|
|
|
|
Note that this procedure can only be used once per file as Guile scripts are
|
|
|
|
|
not supported."
|
|
|
|
|
(define update-env
|
|
|
|
|
(match-lambda
|
|
|
|
|
((var sep '= rest)
|
|
|
|
|
`(setenv ,var ,(string-join rest sep)))
|
|
|
|
|
((var sep 'prefix rest)
|
|
|
|
|
`(let ((current (getenv ,var)))
|
|
|
|
|
(setenv ,var (if current
|
|
|
|
|
(string-append ,(string-join rest sep)
|
|
|
|
|
,sep current)
|
|
|
|
|
,(string-join rest sep)))))
|
|
|
|
|
((var sep 'suffix rest)
|
|
|
|
|
`(let ((current (getenv ,var)))
|
|
|
|
|
(setenv ,var (if current
|
|
|
|
|
(string-append current ,sep
|
|
|
|
|
,(string-join rest sep))
|
|
|
|
|
,(string-join rest sep)))))
|
|
|
|
|
((var '= rest)
|
|
|
|
|
`(setenv ,var ,(string-join rest ":")))
|
|
|
|
|
((var 'prefix rest)
|
|
|
|
|
`(let ((current (getenv ,var)))
|
|
|
|
|
(setenv ,var (if current
|
|
|
|
|
(string-append ,(string-join rest ":")
|
|
|
|
|
":" current)
|
|
|
|
|
,(string-join rest ":")))))
|
|
|
|
|
((var 'suffix rest)
|
|
|
|
|
`(let ((current (getenv ,var)))
|
|
|
|
|
(setenv ,var (if current
|
|
|
|
|
(string-append current ":"
|
|
|
|
|
,(string-join rest ":"))
|
|
|
|
|
,(string-join rest ":")))))))
|
|
|
|
|
(let-values (((interpreter args coding-line)
|
|
|
|
|
(call-with-ascii-input-file prog
|
|
|
|
|
(lambda (p)
|
|
|
|
|
(let ((first-match
|
|
|
|
|
(false-if-exception
|
|
|
|
|
(regexp-exec interpreter-regex (read-line p)))))
|
|
|
|
|
(values (and first-match (match:substring first-match 1))
|
|
|
|
|
(and first-match (match:substring first-match 3))
|
|
|
|
|
(false-if-exception
|
|
|
|
|
(and=> (regexp-exec coding-line-regex (read-line p))
|
|
|
|
|
(lambda (m) (match:substring m 0))))))))))
|
|
|
|
|
(if interpreter
|
|
|
|
|
(let* ((header (format #f "\
|
|
|
|
|
#!~a --no-auto-compile
|
|
|
|
|
#!#; ~a
|
|
|
|
|
#\\-~s
|
|
|
|
|
#\\-~s
|
|
|
|
|
"
|
|
|
|
|
guile
|
|
|
|
|
(or coding-line "Guix wrapper")
|
|
|
|
|
(cons 'begin (map update-env
|
|
|
|
|
(match vars
|
|
|
|
|
((#:guile _ . vars) vars)
|
|
|
|
|
(_ vars))))
|
|
|
|
|
`(let ((cl (command-line)))
|
|
|
|
|
(apply execl ,interpreter
|
|
|
|
|
(car cl)
|
|
|
|
|
(cons (car cl)
|
|
|
|
|
(append
|
|
|
|
|
',(string-split args #\space)
|
|
|
|
|
cl))))))
|
|
|
|
|
(template (string-append prog ".XXXXXX"))
|
|
|
|
|
(out (mkstemp! template))
|
|
|
|
|
(st (stat prog))
|
|
|
|
|
(mode (stat:mode st)))
|
|
|
|
|
(with-throw-handler #t
|
|
|
|
|
(lambda ()
|
|
|
|
|
(call-with-ascii-input-file prog
|
|
|
|
|
(lambda (p)
|
|
|
|
|
(format out header)
|
|
|
|
|
(dump-port p out)
|
|
|
|
|
(close out)
|
|
|
|
|
(chmod template mode)
|
|
|
|
|
(rename-file template prog)
|
|
|
|
|
(set-file-time prog st))))
|
|
|
|
|
(lambda (key . args)
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"wrap-script: ~a: error: ~a ~s~%"
|
|
|
|
|
prog key args)
|
|
|
|
|
(false-if-exception (delete-file template))
|
|
|
|
|
(raise (condition
|
|
|
|
|
(&wrap-error (program prog)
|
|
|
|
|
(type key))))
|
|
|
|
|
#f)))
|
|
|
|
|
(raise (condition
|
|
|
|
|
(&wrap-error (program prog)
|
|
|
|
|
(type 'no-interpreter-found)))))))))
|
|
|
|
|
|
2019-05-26 10:15:28 +02:00
|
|
|
|
(define* (make-desktop-entry-file destination #:key
|
|
|
|
|
(type "Application") ; One of "Application", "Link" or "Directory".
|
|
|
|
|
(version "1.1")
|
|
|
|
|
name
|
|
|
|
|
(generic-name name)
|
|
|
|
|
(no-display #f)
|
|
|
|
|
comment
|
|
|
|
|
icon
|
|
|
|
|
(hidden #f)
|
|
|
|
|
only-show-in
|
|
|
|
|
not-show-in
|
|
|
|
|
(d-bus-activatable #f)
|
|
|
|
|
try-exec
|
|
|
|
|
exec
|
|
|
|
|
path
|
|
|
|
|
(terminal #f)
|
|
|
|
|
actions
|
|
|
|
|
mime-type
|
|
|
|
|
(categories "Application")
|
|
|
|
|
implements
|
|
|
|
|
keywords
|
|
|
|
|
(startup-notify #t)
|
|
|
|
|
startup-w-m-class
|
|
|
|
|
#:rest all-args)
|
|
|
|
|
"Create a desktop entry file at DESTINATION.
|
|
|
|
|
You must specify NAME.
|
|
|
|
|
|
|
|
|
|
Values can be booleans, numbers, strings or list of strings.
|
|
|
|
|
|
|
|
|
|
Additionally, locales can be specified with an alist where the key is the
|
|
|
|
|
locale. The #f key specifies the default. Example:
|
|
|
|
|
|
|
|
|
|
#:name '((#f \"I love Guix\") (\"fr\" \"J'aime Guix\"))
|
|
|
|
|
|
|
|
|
|
produces
|
|
|
|
|
|
|
|
|
|
Name=I love Guix
|
|
|
|
|
Name[fr]=J'aime Guix
|
|
|
|
|
|
|
|
|
|
For a complete description of the format, see the specifications at
|
|
|
|
|
https://specifications.freedesktop.org/desktop-entry-spec/desktop-entry-spec-latest.html."
|
|
|
|
|
(define (escape-semicolon s)
|
|
|
|
|
(string-join (string-split s #\;) "\\;"))
|
|
|
|
|
(define* (parse key value #:optional locale)
|
|
|
|
|
(set! value (match value
|
|
|
|
|
(#t "true")
|
|
|
|
|
(#f "false")
|
|
|
|
|
((? number? n) n)
|
|
|
|
|
((? string? s) (escape-semicolon s))
|
|
|
|
|
((? list? value)
|
|
|
|
|
(catch 'wrong-type-arg
|
|
|
|
|
(lambda () (string-join (map escape-semicolon value) ";"))
|
|
|
|
|
(lambda args (error "List arguments can only contain strings: ~a" args))))
|
|
|
|
|
(_ (error "Value must be a boolean, number, string or list of strings"))))
|
|
|
|
|
(format #t "~a=~a~%"
|
|
|
|
|
(if locale
|
|
|
|
|
(format #f "~a[~a]" key locale)
|
|
|
|
|
key)
|
|
|
|
|
value))
|
|
|
|
|
|
|
|
|
|
(define key-error-message "This procedure only takes key arguments beside DESTINATION")
|
|
|
|
|
|
|
|
|
|
(unless name
|
|
|
|
|
(error "Missing NAME key argument"))
|
|
|
|
|
(unless (member #:type all-args)
|
|
|
|
|
(set! all-args (append (list #:type type) all-args)))
|
|
|
|
|
(mkdir-p (dirname destination))
|
|
|
|
|
|
|
|
|
|
(with-output-to-file destination
|
|
|
|
|
(lambda ()
|
|
|
|
|
(format #t "[Desktop Entry]~%")
|
|
|
|
|
(let loop ((args all-args))
|
|
|
|
|
(match args
|
|
|
|
|
(() #t)
|
|
|
|
|
((_) (error key-error-message))
|
|
|
|
|
((key value . ...)
|
|
|
|
|
(unless (keyword? key)
|
|
|
|
|
(error key-error-message))
|
|
|
|
|
(set! key
|
|
|
|
|
(string-join (map string-titlecase
|
|
|
|
|
(string-split (symbol->string
|
|
|
|
|
(keyword->symbol key))
|
|
|
|
|
#\-))
|
|
|
|
|
""))
|
|
|
|
|
(match value
|
|
|
|
|
(((_ . _) . _)
|
|
|
|
|
(for-each (lambda (locale-subvalue)
|
|
|
|
|
(parse key
|
|
|
|
|
(if (and (list? (cdr locale-subvalue))
|
|
|
|
|
(= 1 (length (cdr locale-subvalue))))
|
|
|
|
|
;; Support both proper and improper lists for convenience.
|
|
|
|
|
(cadr locale-subvalue)
|
|
|
|
|
(cdr locale-subvalue))
|
|
|
|
|
(car locale-subvalue)))
|
|
|
|
|
value))
|
|
|
|
|
(_
|
|
|
|
|
(parse key value)))
|
|
|
|
|
(loop (cddr args))))))))
|
|
|
|
|
|
2015-02-27 14:54:00 +01:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Locales.
|
|
|
|
|
;;;
|
|
|
|
|
|
|
|
|
|
(define (locale-category->string category)
|
|
|
|
|
"Return the name of locale category CATEGORY, one of the 'LC_' constants.
|
|
|
|
|
If CATEGORY is a bitwise or of several 'LC_' constants, an approximation is
|
|
|
|
|
returned."
|
|
|
|
|
(letrec-syntax ((convert (syntax-rules ()
|
|
|
|
|
((_)
|
|
|
|
|
(number->string category))
|
|
|
|
|
((_ first rest ...)
|
|
|
|
|
(if (= first category)
|
|
|
|
|
(symbol->string 'first)
|
|
|
|
|
(convert rest ...))))))
|
|
|
|
|
(convert LC_ADDRESS LC_ALL LC_COLLATE LC_CTYPE
|
|
|
|
|
LC_IDENTIFICATION LC_MEASUREMENT LC_MESSAGES LC_MONETARY
|
|
|
|
|
LC_NAME LC_NUMERIC LC_PAPER LC_TELEPHONE
|
|
|
|
|
LC_TIME)))
|
|
|
|
|
|
2012-07-01 17:32:03 +02:00
|
|
|
|
;;; Local Variables:
|
|
|
|
|
;;; eval: (put 'call-with-output-file/atomic 'scheme-indent-function 1)
|
2014-11-22 12:44:56 +01:00
|
|
|
|
;;; eval: (put 'call-with-ascii-input-file 'scheme-indent-function 1)
|
2012-07-01 17:32:03 +02:00
|
|
|
|
;;; eval: (put 'with-throw-handler 'scheme-indent-function 1)
|
2012-09-01 19:21:06 +02:00
|
|
|
|
;;; eval: (put 'let-matches 'scheme-indent-function 3)
|
2012-10-16 17:28:11 +02:00
|
|
|
|
;;; eval: (put 'with-atomic-file-replacement 'scheme-indent-function 1)
|
2012-07-01 17:32:03 +02:00
|
|
|
|
;;; End:
|