store: Add 'map/accumulate-builds' cutoff to address pathological cases.

Fixes <https://bugs.gnu.org/49439>.
Reported by Ricardo Wurmus <rekado@elephly.net>.

Previously, a command such as:

  guix environment pigx-scrnaseq

could lead to unbounded memory growth and could even fail to complete
when some items are missing from the store.  This was because
'map/accumulate-builds' callees would keep making .drv build requests
that were turned into <unresolved> nodes; in this case, there are often
many identical build requests.  Stopping accumulation earlier allows us
to unlock the situation by proceeding with the first few build requests
instead of spinning until we've accumulated all the build requests.

* guix/store.scm (map/accumulate-builds): Define 'accumulation-cutoff'.
Use a loop when iterating over LST and maintain a counter of unresolved
nodes met so far; return when the counter exceeds ACCUMULATION-CUTOFF.
This commit is contained in:
Ludovic Courtès 2021-08-10 12:24:44 +02:00
parent b0a6b1f13c
commit fa81971cba
No known key found for this signature in database
GPG key ID: 090B11993D9AEBB5

View file

@ -1358,11 +1358,28 @@ (define (build-accumulator continue store things mode)
(define (map/accumulate-builds store proc lst) (define (map/accumulate-builds store proc lst)
"Apply PROC over each element of LST, accumulating 'build-things' calls and "Apply PROC over each element of LST, accumulating 'build-things' calls and
coalescing them into a single call." coalescing them into a single call."
(define result (define accumulation-cutoff
(map (lambda (obj) ;; Threshold above which we stop accumulating unresolved nodes to avoid
(with-build-handler build-accumulator ;; pessimal behavior where we keep stumbling upon the same .drv build
(proc obj))) ;; requests with many incoming edges. See <https://bugs.gnu.org/49439>.
lst)) 30)
(define-values (result rest)
(let loop ((lst lst)
(result '())
(unresolved 0))
(match lst
((head . tail)
(match (with-build-handler build-accumulator
(proc head))
((? unresolved? obj)
(if (> unresolved accumulation-cutoff)
(values (reverse (cons obj result)) tail)
(loop tail (cons obj result) (+ 1 unresolved))))
(obj
(loop tail (cons obj result) unresolved))))
(()
(values (reverse result) lst)))))
(match (append-map (lambda (obj) (match (append-map (lambda (obj)
(if (unresolved? obj) (if (unresolved? obj)
@ -1370,6 +1387,7 @@ (define result
'())) '()))
result) result)
(() (()
;; REST is necessarily empty.
result) result)
(to-build (to-build
;; We've accumulated things TO-BUILD. Actually build them and resume the ;; We've accumulated things TO-BUILD. Actually build them and resume the
@ -1382,7 +1400,7 @@ (define result
;; unnecessary. ;; unnecessary.
((unresolved-continuation obj) #f) ((unresolved-continuation obj) #f)
obj)) obj))
result)))) (append result rest)))))
(define build-things (define build-things
(let ((build (operation (build-things (string-list things) (let ((build (operation (build-things (string-list things)