packages: 'patch-and-repack' returns a directory when given a directory.

Previously, 'patch-and-repack' would always create a tar.xz archive as a
result, even if the input was a directory (a checkout).  This change
reduces gratuitous CPU and storage overhead.

* guix/packages.scm (patch-and-repack)[tarxz-name]: Remove 'checkout?' case.
[build](repack): New procedure, with "tar" invocation formerly at the
top level.
If SOURCE is a directory, call 'copy-recursively'; otherwise, call
'repack'.
Change NAME to ORIGINAL-FILE-NAME when it matches 'checkout?'.
This commit is contained in:
Ludovic Courtès 2021-01-15 14:07:21 +01:00
parent 812a2931de
commit f41ff53293
No known key found for this signature in database
GPG Key ID: 090B11993D9AEBB5

View File

@ -1,5 +1,5 @@
;;; GNU Guix --- Functional package management for GNU ;;; GNU Guix --- Functional package management for GNU
;;; Copyright © 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020 Ludovic Courtès <ludo@gnu.org> ;;; Copyright © 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020, 2021 Ludovic Courtès <ludo@gnu.org>
;;; Copyright © 2014, 2015, 2017, 2018 Mark H Weaver <mhw@netris.org> ;;; Copyright © 2014, 2015, 2017, 2018 Mark H Weaver <mhw@netris.org>
;;; Copyright © 2015 Eric Bavier <bavier@member.fsf.org> ;;; Copyright © 2015 Eric Bavier <bavier@member.fsf.org>
;;; Copyright © 2016 Alex Kost <alezost@gmail.com> ;;; Copyright © 2016 Alex Kost <alezost@gmail.com>
@ -635,11 +635,9 @@ specifies modules in scope when evaluating SNIPPET."
(define (tarxz-name file-name) (define (tarxz-name file-name)
;; Return a '.tar.xz' file name based on FILE-NAME. ;; Return a '.tar.xz' file name based on FILE-NAME.
(let ((base (cond ((numeric-extension? file-name) (let ((base (if (numeric-extension? file-name)
original-file-name) original-file-name
((checkout? file-name) (file-sans-extension file-name))))
(string-drop-right file-name 9))
(else (file-sans-extension file-name)))))
(string-append base (string-append base
(if (equal? (file-extension base) "tar") (if (equal? (file-extension base) "tar")
".xz" ".xz"
@ -689,6 +687,29 @@ specifies modules in scope when evaluating SNIPPET."
(lambda (name) (lambda (name)
(not (member name '("." ".."))))))) (not (member name '("." "..")))))))
(define (repack directory output)
;; Write to OUTPUT a compressed tarball containing DIRECTORY.
(unless tar-supports-sort?
(call-with-output-file ".file_list"
(lambda (port)
(for-each (lambda (name)
(format port "~a~%" name))
(find-files directory
#:directories? #t
#:fail-on-error? #t)))))
(apply invoke #+(file-append tar "/bin/tar")
"cvfa" output
;; Avoid non-determinism in the archive. Set the mtime
;; to 1 as is the case in the store (software like gzip
;; behaves differently when it stumbles upon mtime = 0).
"--mtime=@1"
"--owner=root:0" "--group=root:0"
(if tar-supports-sort?
`("--sort=name" ,directory)
'("--no-recursion"
"--files-from=.file_list"))))
;; Encoding/decoding errors shouldn't be silent. ;; Encoding/decoding errors shouldn't be silent.
(fluid-set! %default-port-conversion-strategy 'error) (fluid-set! %default-port-conversion-strategy 'error)
@ -742,30 +763,16 @@ specifies modules in scope when evaluating SNIPPET."
(chdir "..") (chdir "..")
(unless tar-supports-sort? ;; If SOURCE is a directory (such as a checkout), return a
(call-with-output-file ".file_list" ;; directory. Otherwise create a tarball.
(lambda (port) (if (file-is-directory? #+source)
(for-each (lambda (name) (copy-recursively directory #$output
(format port "~a~%" name)) #:log (%make-void-port "w"))
(find-files directory (repack directory #$output))))))
#:directories? #t
#:fail-on-error? #t)))))
(apply invoke
(string-append #+tar "/bin/tar")
"cvfa" #$output
;; Avoid non-determinism in the archive. Set the mtime
;; to 1 as is the case in the store (software like gzip
;; behaves differently when it stumbles upon mtime = 0).
"--mtime=@1"
"--owner=root:0"
"--group=root:0"
(if tar-supports-sort?
`("--sort=name"
,directory)
'("--no-recursion"
"--files-from=.file_list")))))))
(let ((name (tarxz-name original-file-name))) (let ((name (if (checkout? original-file-name)
original-file-name
(tarxz-name original-file-name))))
(gexp->derivation name build (gexp->derivation name build
#:graft? #f #:graft? #f
#:system system #:system system