mirror of
https://git.in.rschanz.org/ryan77627/guix.git
synced 2025-01-11 13:49:23 -05:00
grafts: 'graft-derivation' does now introduce grafts that shadow other grafts.
Partly fixes <http://bugs.gnu.org/24418>. * guix/grafts.scm (cumulative-grafts)[graft-origin?]: New procedure. [dependency-grafts]: Use it in new 'if' around recursive call. * tests/grafts.scm ("graft-derivation, grafts are not shadowed"): New test.
This commit is contained in:
parent
d0025d0144
commit
b013c33f6f
2 changed files with 82 additions and 4 deletions
|
@ -227,13 +227,29 @@ (define* (cumulative-grafts store drv grafts
|
||||||
|
|
||||||
This is a monadic procedure in %STATE-MONAD where the state is a vhash mapping
|
This is a monadic procedure in %STATE-MONAD where the state is a vhash mapping
|
||||||
derivations to the corresponding set of grafts."
|
derivations to the corresponding set of grafts."
|
||||||
|
(define (graft-origin? drv graft)
|
||||||
|
;; Return true if DRV corresponds to the origin of GRAFT.
|
||||||
|
(match graft
|
||||||
|
(($ <graft> (? derivation? origin) output)
|
||||||
|
(match (assoc-ref (derivation->output-paths drv) output)
|
||||||
|
((? string? result)
|
||||||
|
(string=? result
|
||||||
|
(derivation->output-path origin output)))
|
||||||
|
(_
|
||||||
|
#f)))
|
||||||
|
(_
|
||||||
|
#f)))
|
||||||
|
|
||||||
(define (dependency-grafts item)
|
(define (dependency-grafts item)
|
||||||
(let-values (((drv output) (item->deriver store item)))
|
(let-values (((drv output) (item->deriver store item)))
|
||||||
(if drv
|
(if drv
|
||||||
|
;; If GRAFTS already contains a graft from DRV, do not override it.
|
||||||
|
(if (find (cut graft-origin? drv <>) grafts)
|
||||||
|
(state-return grafts)
|
||||||
(cumulative-grafts store drv grafts references
|
(cumulative-grafts store drv grafts references
|
||||||
#:outputs (list output)
|
#:outputs (list output)
|
||||||
#:guile guile
|
#:guile guile
|
||||||
#:system system)
|
#:system system))
|
||||||
(state-return grafts))))
|
(state-return grafts))))
|
||||||
|
|
||||||
(define (return/cache cache value)
|
(define (return/cache cache value)
|
||||||
|
|
|
@ -218,4 +218,66 @@ (define %mkdir
|
||||||
(let ((out (derivation->output-path grafted)))
|
(let ((out (derivation->output-path grafted)))
|
||||||
(file-is-directory? (string-append out "/" repl))))))
|
(file-is-directory? (string-append out "/" repl))))))
|
||||||
|
|
||||||
|
(test-assert "graft-derivation, grafts are not shadowed"
|
||||||
|
;; We build a DAG as below, where dotted arrows represent replacements and
|
||||||
|
;; solid arrows represent dependencies:
|
||||||
|
;;
|
||||||
|
;; P1 ·············> P1R
|
||||||
|
;; |\__________________.
|
||||||
|
;; v v
|
||||||
|
;; P2 ·············> P2R
|
||||||
|
;; |
|
||||||
|
;; v
|
||||||
|
;; P3
|
||||||
|
;;
|
||||||
|
;; We want to make sure that the two grafts we want to apply to P3 are
|
||||||
|
;; honored and not shadowed by other computed grafts.
|
||||||
|
(let* ((p1 (build-expression->derivation
|
||||||
|
%store "p1"
|
||||||
|
'(mkdir (assoc-ref %outputs "out"))))
|
||||||
|
(p1r (build-expression->derivation
|
||||||
|
%store "P1"
|
||||||
|
'(let ((out (assoc-ref %outputs "out")))
|
||||||
|
(mkdir out)
|
||||||
|
(call-with-output-file (string-append out "/replacement")
|
||||||
|
(const #t)))))
|
||||||
|
(p2 (build-expression->derivation
|
||||||
|
%store "p2"
|
||||||
|
`(let ((out (assoc-ref %outputs "out")))
|
||||||
|
(mkdir out)
|
||||||
|
(chdir out)
|
||||||
|
(symlink (assoc-ref %build-inputs "p1") "p1"))
|
||||||
|
#:inputs `(("p1" ,p1))))
|
||||||
|
(p2r (build-expression->derivation
|
||||||
|
%store "P2"
|
||||||
|
`(let ((out (assoc-ref %outputs "out")))
|
||||||
|
(mkdir out)
|
||||||
|
(chdir out)
|
||||||
|
(symlink (assoc-ref %build-inputs "p1") "p1")
|
||||||
|
(call-with-output-file (string-append out "/replacement")
|
||||||
|
(const #t)))
|
||||||
|
#:inputs `(("p1" ,p1))))
|
||||||
|
(p3 (build-expression->derivation
|
||||||
|
%store "p3"
|
||||||
|
`(let ((out (assoc-ref %outputs "out")))
|
||||||
|
(mkdir out)
|
||||||
|
(chdir out)
|
||||||
|
(symlink (assoc-ref %build-inputs "p2") "p2"))
|
||||||
|
#:inputs `(("p2" ,p2))))
|
||||||
|
(p1g (graft
|
||||||
|
(origin p1)
|
||||||
|
(replacement p1r)))
|
||||||
|
(p2g (graft
|
||||||
|
(origin p2)
|
||||||
|
(replacement (graft-derivation %store p2r (list p1g)))))
|
||||||
|
(p3d (graft-derivation %store p3 (list p1g p2g))))
|
||||||
|
(and (build-derivations %store (list p3d))
|
||||||
|
(let ((out (derivation->output-path (pk p3d))))
|
||||||
|
;; Make sure OUT refers to the replacement of P2, which in turn
|
||||||
|
;; refers to the replacement of P1, as specified by P1G and P2G.
|
||||||
|
;; It used to be the case that P2G would be shadowed by a simple
|
||||||
|
;; P2->P2R graft, which is not what we want.
|
||||||
|
(and (file-exists? (string-append out "/p2/replacement"))
|
||||||
|
(file-exists? (string-append out "/p2/p1/replacement")))))))
|
||||||
|
|
||||||
(test-end)
|
(test-end)
|
||||||
|
|
Loading…
Reference in a new issue