summaryrefslogtreecommitdiff
path: root/guix/store
diff options
context:
space:
mode:
authorLudovic Courtès <ludo@gnu.org>2018-07-20 13:49:50 +0200
committerLudovic Courtès <ludo@gnu.org>2018-07-20 15:01:33 +0200
commit4f89a8eec69491b925f084381ea4de37527c9310 (patch)
tree6d4051386b58ce4b7f699b20173fe0255a51a7a5 /guix/store
parenta2662bfe9ccedc9d791696424988dcadff9247b0 (diff)
downloadgnu-guix-4f89a8eec69491b925f084381ea4de37527c9310.tar
gnu-guix-4f89a8eec69491b925f084381ea4de37527c9310.tar.gz
deduplication: Work around Guile bug in 'seek'.
Fixes <https://bugs.gnu.org/32161>. Reported by Ricardo Wurmus <rekado@elephly.net>. This mostly reverts 83099892e0cf0d9c59f5e1a0774331026e48baa8. * guix/store/deduplication.scm (counting-wrapper-port): New procedure. (nar-sha256): Use it.
Diffstat (limited to 'guix/store')
-rw-r--r--guix/store/deduplication.scm32
1 files changed, 28 insertions, 4 deletions
diff --git a/guix/store/deduplication.scm b/guix/store/deduplication.scm
index 8234819f14..8c19d7309e 100644
--- a/guix/store/deduplication.scm
+++ b/guix/store/deduplication.scm
@@ -31,14 +31,38 @@
#:export (nar-sha256
deduplicate))
+;; XXX: This port is used as a workaround on Guile <= 2.2.4 where
+;; 'port-position' throws to 'out-of-range' when the offset is great than or
+;; equal to 2^32: <https://bugs.gnu.org/32161>.
+(define (counting-wrapper-port output-port)
+ "Return two values: an output port that wraps OUTPUT-PORT, and a thunk to
+retrieve the number of bytes written to OUTPUT-PORT."
+ (let ((byte-count 0))
+ (values (make-custom-binary-output-port "counting-wrapper"
+ (lambda (bytes offset count)
+ (put-bytevector output-port bytes
+ offset count)
+ (set! byte-count
+ (+ byte-count count))
+ count)
+ (lambda ()
+ byte-count)
+ #f
+ (lambda ()
+ (close-port output-port)))
+ (lambda ()
+ byte-count))))
+
(define (nar-sha256 file)
"Gives the sha256 hash of a file and the size of the file in nar form."
- (let-values (((port get-hash) (open-sha256-port)))
- (write-file file port)
+ (let*-values (((port get-hash) (open-sha256-port))
+ ((wrapper get-size) (counting-wrapper-port port)))
+ (write-file file wrapper)
+ (force-output wrapper)
(force-output port)
(let ((hash (get-hash))
- (size (port-position port)))
- (close-port port)
+ (size (get-size)))
+ (close-port wrapper)
(values hash size))))
(define (tempname-in directory)