2015-08-27 00:36:41 +02:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
2023-08-21 15:13:59 +02:00
|
|
|
|
;;; Copyright © 2015-2023 Ludovic Courtès <ludo@gnu.org>
|
2015-08-27 00:36:41 +02:00
|
|
|
|
;;;
|
|
|
|
|
;;; This file is part of GNU Guix.
|
|
|
|
|
;;;
|
|
|
|
|
;;; GNU Guix is free software; you can redistribute it and/or modify it
|
|
|
|
|
;;; 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.
|
|
|
|
|
;;;
|
|
|
|
|
;;; GNU Guix is distributed in the hope that it will be useful, but
|
|
|
|
|
;;; 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
|
|
|
|
|
;;; along with GNU Guix. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
(define-module (test-graph)
|
|
|
|
|
#:use-module (guix tests)
|
2015-11-21 13:12:02 +01:00
|
|
|
|
#:use-module (guix graph)
|
2015-08-27 00:36:41 +02:00
|
|
|
|
#:use-module (guix scripts graph)
|
|
|
|
|
#:use-module (guix packages)
|
|
|
|
|
#:use-module (guix derivations)
|
|
|
|
|
#:use-module (guix store)
|
|
|
|
|
#:use-module (guix monads)
|
|
|
|
|
#:use-module (guix build-system gnu)
|
2015-11-21 14:48:34 +01:00
|
|
|
|
#:use-module (guix build-system trivial)
|
2015-08-27 00:36:41 +02:00
|
|
|
|
#:use-module (guix gexp)
|
2015-11-21 14:48:34 +01:00
|
|
|
|
#:use-module (guix utils)
|
2015-08-27 00:36:41 +02:00
|
|
|
|
#:use-module (gnu packages)
|
2015-11-21 14:48:34 +01:00
|
|
|
|
#:use-module (gnu packages base)
|
2020-05-10 00:04:59 +02:00
|
|
|
|
#:use-module (gnu packages bootstrap)
|
2015-11-21 14:48:34 +01:00
|
|
|
|
#:use-module (gnu packages guile)
|
2017-01-04 16:16:17 +01:00
|
|
|
|
#:use-module (gnu packages libunistring)
|
2015-08-27 00:36:41 +02:00
|
|
|
|
#:use-module (gnu packages bootstrap)
|
|
|
|
|
#:use-module (ice-9 match)
|
2022-01-24 19:15:44 +01:00
|
|
|
|
#:use-module (ice-9 sandbox)
|
2015-08-27 00:36:41 +02:00
|
|
|
|
#:use-module (srfi srfi-1)
|
|
|
|
|
#:use-module (srfi srfi-11)
|
|
|
|
|
#:use-module (srfi srfi-26)
|
|
|
|
|
#:use-module (srfi srfi-64))
|
|
|
|
|
|
|
|
|
|
(define %store
|
|
|
|
|
(open-connection-for-tests))
|
|
|
|
|
|
2016-03-06 21:53:28 +01:00
|
|
|
|
;; Globally disable grafts because they can trigger early builds.
|
|
|
|
|
(%graft? #f)
|
|
|
|
|
|
2015-08-27 00:36:41 +02:00
|
|
|
|
(define (make-recording-backend)
|
|
|
|
|
"Return a <graph-backend> and a thunk that returns the recorded nodes and
|
|
|
|
|
edges."
|
|
|
|
|
(let ((nodes '())
|
|
|
|
|
(edges '()))
|
|
|
|
|
(define (record-node id label port)
|
|
|
|
|
(set! nodes (cons (list id label) nodes)))
|
|
|
|
|
(define (record-edge source target port)
|
|
|
|
|
(set! edges (cons (list source target) edges)))
|
|
|
|
|
(define (return)
|
|
|
|
|
(values (reverse nodes) (reverse edges)))
|
|
|
|
|
|
2016-10-20 08:47:03 +02:00
|
|
|
|
(values (graph-backend "test" "This is the test backend."
|
|
|
|
|
(const #t) (const #t)
|
2015-08-27 00:36:41 +02:00
|
|
|
|
record-node record-edge)
|
|
|
|
|
return)))
|
|
|
|
|
|
|
|
|
|
(define (package->tuple package)
|
|
|
|
|
"Return a tuple representing PACKAGE as produced by %PACKAGE-NODE-TYPE."
|
|
|
|
|
(list (object-address package)
|
|
|
|
|
(package-full-name package)))
|
|
|
|
|
|
|
|
|
|
(define (edge->tuple source target)
|
|
|
|
|
"Likewise for an edge from SOURCE to TARGET."
|
|
|
|
|
(list (object-address source)
|
|
|
|
|
(object-address target)))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
(test-begin "graph")
|
|
|
|
|
|
|
|
|
|
(test-assert "package DAG"
|
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(let* ((p1 (dummy-package "p1"))
|
|
|
|
|
(p2 (dummy-package "p2" (inputs `(("p1" ,p1)))))
|
|
|
|
|
(p3 (dummy-package "p3" (inputs `(("p2" ,p2) ("p1", p1))))))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(export-graph (list p3) 'port
|
|
|
|
|
#:node-type %package-node-type
|
|
|
|
|
#:backend backend))
|
|
|
|
|
;; We should see nothing more than these 3 packages.
|
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(and (equal? nodes (map package->tuple (list p3 p2 p1)))
|
|
|
|
|
(equal? edges
|
|
|
|
|
(map edge->tuple
|
|
|
|
|
(list p3 p3 p2)
|
|
|
|
|
(list p2 p1 p1))))))))
|
|
|
|
|
|
2021-09-17 10:13:15 +02:00
|
|
|
|
(test-assert "package DAG, limited depth"
|
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(let* ((p1 (dummy-package "p1"))
|
|
|
|
|
(p2 (dummy-package "p2" (inputs `(("p1" ,p1)))))
|
|
|
|
|
(p3 (dummy-package "p3" (inputs `(("p1" ,p1)))))
|
|
|
|
|
(p4 (dummy-package "p4" (inputs `(("p2" ,p2) ("p3" ,p3))))))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(export-graph (list p4) 'port
|
|
|
|
|
#:max-depth 1
|
|
|
|
|
#:node-type %package-node-type
|
|
|
|
|
#:backend backend))
|
|
|
|
|
;; We should see nothing more than these 3 packages.
|
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(and (equal? nodes (map package->tuple (list p4 p2 p3)))
|
|
|
|
|
(equal? edges
|
|
|
|
|
(map edge->tuple
|
|
|
|
|
(list p4 p4)
|
|
|
|
|
(list p2 p3))))))))
|
|
|
|
|
|
2022-01-24 19:15:44 +01:00
|
|
|
|
(test-assert "package DAG, oops it was a cycle"
|
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(letrec ((p1 (dummy-package "p1" (inputs `(("p3" ,p3)))))
|
|
|
|
|
(p2 (dummy-package "p2" (inputs `(("p1" ,p1)))))
|
|
|
|
|
(p3 (dummy-package "p3" (inputs `(("p2" ,p2) ("p1", p1))))))
|
|
|
|
|
(call-with-time-limit
|
|
|
|
|
600 ;; If ever this test should fail, we still want it to terminate
|
|
|
|
|
(lambda ()
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(export-graph (list p3) 'port
|
|
|
|
|
#:node-type %package-node-type
|
|
|
|
|
#:backend backend)))
|
|
|
|
|
(lambda ()
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(export-graph
|
|
|
|
|
(list (dummy-package "timeout-reached"))
|
|
|
|
|
'port
|
|
|
|
|
#:node-type %package-node-type
|
|
|
|
|
#:backend backend))))
|
|
|
|
|
;; We should see nothing more than these 3 packages.
|
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(and (equal? nodes (map package->tuple (list p3 p2 p1)))
|
|
|
|
|
(equal? edges
|
|
|
|
|
(map edge->tuple
|
|
|
|
|
(list p3 p3 p2 p1)
|
|
|
|
|
(list p2 p1 p1 p3))))))))
|
|
|
|
|
|
2017-01-04 16:16:17 +01:00
|
|
|
|
(test-assert "reverse package DAG"
|
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(export-graph (list libunistring) 'port
|
|
|
|
|
#:node-type %reverse-package-node-type
|
|
|
|
|
#:backend backend))
|
|
|
|
|
;; We should see nothing more than these 3 packages.
|
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(and (member (package->tuple guile-2.0) nodes)
|
|
|
|
|
(->bool (member (edge->tuple libunistring guile-2.0) edges))))))
|
|
|
|
|
|
2015-08-27 00:36:41 +02:00
|
|
|
|
(test-assert "bag-emerged DAG"
|
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
2016-01-02 22:12:36 +01:00
|
|
|
|
(let* ((o (dummy-origin (method (lambda _
|
|
|
|
|
(text-file "foo" "bar")))))
|
|
|
|
|
(p (dummy-package "p" (source o)))
|
|
|
|
|
(implicit (map (match-lambda
|
2017-12-17 22:27:21 +01:00
|
|
|
|
((label package) package)
|
|
|
|
|
((label package output) package))
|
2016-01-02 22:12:36 +01:00
|
|
|
|
(standard-packages))))
|
2015-08-27 00:36:41 +02:00
|
|
|
|
(run-with-store %store
|
|
|
|
|
(export-graph (list p) 'port
|
|
|
|
|
#:node-type %bag-emerged-node-type
|
|
|
|
|
#:backend backend))
|
|
|
|
|
;; We should see exactly P and IMPLICIT, with one edge from P to each
|
2017-12-17 22:27:21 +01:00
|
|
|
|
;; element of IMPLICIT. O must not appear among NODES. Note: IMPLICIT
|
|
|
|
|
;; contains "glibc" twice, once for "out" and a second time for
|
|
|
|
|
;; "static", hence the 'delete-duplicates' call below.
|
2015-08-27 00:36:41 +02:00
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(and (equal? (match nodes
|
|
|
|
|
(((labels names) ...)
|
|
|
|
|
names))
|
2017-12-17 22:27:21 +01:00
|
|
|
|
(map package-full-name
|
|
|
|
|
(cons p (delete-duplicates implicit))))
|
2015-08-27 00:36:41 +02:00
|
|
|
|
(equal? (match edges
|
|
|
|
|
(((sources destinations) ...)
|
|
|
|
|
(zip (map store-path-package-name sources)
|
|
|
|
|
(map store-path-package-name destinations))))
|
|
|
|
|
(map (lambda (destination)
|
|
|
|
|
(list "p-0.drv"
|
|
|
|
|
(string-append
|
2018-04-09 01:51:42 +02:00
|
|
|
|
(package-full-name destination "-")
|
2015-08-27 00:36:41 +02:00
|
|
|
|
".drv")))
|
|
|
|
|
implicit)))))))
|
|
|
|
|
|
2015-11-21 14:48:34 +01:00
|
|
|
|
(test-assert "bag DAG" ;a big town in Iraq
|
2015-08-27 00:36:41 +02:00
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(let ((p (dummy-package "p")))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(export-graph (list p) 'port
|
|
|
|
|
#:node-type %bag-node-type
|
|
|
|
|
#:backend backend))
|
|
|
|
|
;; We should see P, its implicit inputs as well as the whole DAG, which
|
|
|
|
|
;; should include bootstrap binaries.
|
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(every (lambda (name)
|
|
|
|
|
(find (cut string=? name <>)
|
|
|
|
|
(match nodes
|
|
|
|
|
(((labels names) ...)
|
|
|
|
|
names))))
|
2018-09-09 11:54:27 +02:00
|
|
|
|
(match (%bootstrap-inputs)
|
2015-08-27 00:36:41 +02:00
|
|
|
|
(((labels packages) ...)
|
2018-09-09 11:54:27 +02:00
|
|
|
|
(map package-full-name (filter package? packages)))))))))
|
2015-08-27 00:36:41 +02:00
|
|
|
|
|
2015-11-23 23:31:53 +01:00
|
|
|
|
(test-assert "bag DAG, including origins"
|
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(let* ((m (lambda* (uri hash-type hash name #:key system)
|
|
|
|
|
(text-file "foo-1.2.3.tar.gz" "This is a fake!")))
|
2020-05-19 15:55:08 +02:00
|
|
|
|
(o (origin
|
|
|
|
|
(method m) (uri "the-uri")
|
|
|
|
|
(sha256
|
|
|
|
|
(base32
|
|
|
|
|
"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"))))
|
2015-11-23 23:31:53 +01:00
|
|
|
|
(p (dummy-package "p" (source o))))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(export-graph (list p) 'port
|
|
|
|
|
#:node-type %bag-with-origins-node-type
|
|
|
|
|
#:backend backend))
|
|
|
|
|
;; We should see O among the nodes, with an edge coming from P.
|
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(mlet %store-monad ((o* (lower-object o))
|
2016-01-02 22:22:57 +01:00
|
|
|
|
(p* (lower-object p))
|
|
|
|
|
(g (lower-object (default-guile))))
|
2015-11-23 23:31:53 +01:00
|
|
|
|
(return
|
|
|
|
|
(and (find (match-lambda
|
|
|
|
|
((file "the-uri") #t)
|
|
|
|
|
(_ #f))
|
|
|
|
|
nodes)
|
|
|
|
|
(find (match-lambda
|
|
|
|
|
((source target)
|
|
|
|
|
(and (string=? source (derivation-file-name p*))
|
|
|
|
|
(string=? target o*))))
|
2016-01-02 22:22:57 +01:00
|
|
|
|
edges)
|
|
|
|
|
|
|
|
|
|
;; There must also be an edge from O to G.
|
|
|
|
|
(find (match-lambda
|
|
|
|
|
((source target)
|
|
|
|
|
(and (string=? source o*)
|
|
|
|
|
(string=? target (derivation-file-name g)))))
|
2015-11-23 23:31:53 +01:00
|
|
|
|
edges)))))))))
|
|
|
|
|
|
2019-03-23 15:04:44 +01:00
|
|
|
|
(test-assert "reverse bag DAG"
|
2022-07-01 23:25:59 +02:00
|
|
|
|
(let-values (((dune camomile utop)
|
|
|
|
|
(values (specification->package "dune")
|
|
|
|
|
(specification->package "ocaml-camomile")
|
|
|
|
|
(specification->package "ocaml-utop")))
|
2019-03-23 15:04:44 +01:00
|
|
|
|
((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(export-graph (list dune) 'port
|
|
|
|
|
#:node-type %reverse-bag-node-type
|
|
|
|
|
#:backend backend))
|
|
|
|
|
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(mlet %store-monad ((dune-drv (package->derivation dune))
|
2022-07-01 23:25:59 +02:00
|
|
|
|
(camomile-drv (package->derivation camomile))
|
|
|
|
|
(utop-drv (package->derivation utop)))
|
|
|
|
|
;; CAMOMILE uses 'dune-build-system' so DUNE is a direct dependency.
|
|
|
|
|
;; UTOP is much higher in the stack but it should be there.
|
2019-03-23 15:04:44 +01:00
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(return
|
2022-07-01 23:25:59 +02:00
|
|
|
|
(and (member `(,(derivation-file-name camomile-drv)
|
|
|
|
|
,(package-full-name camomile))
|
2019-03-23 15:04:44 +01:00
|
|
|
|
nodes)
|
|
|
|
|
(->bool (member (map derivation-file-name
|
2022-07-01 23:25:59 +02:00
|
|
|
|
(list dune-drv utop-drv))
|
2019-03-23 15:04:44 +01:00
|
|
|
|
edges)))))))))
|
|
|
|
|
|
2015-08-27 00:36:41 +02:00
|
|
|
|
(test-assert "derivation DAG"
|
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(mlet* %store-monad ((txt (text-file "text-file" "Hello!"))
|
|
|
|
|
(guile (package->derivation %bootstrap-guile))
|
|
|
|
|
(drv (gexp->derivation "output"
|
|
|
|
|
#~(symlink #$txt #$output)
|
|
|
|
|
#:guile-for-build
|
|
|
|
|
guile)))
|
|
|
|
|
;; We should get at least these 3 nodes and corresponding edges.
|
|
|
|
|
(mbegin %store-monad
|
|
|
|
|
(export-graph (list drv) 'port
|
|
|
|
|
#:node-type %derivation-node-type
|
|
|
|
|
#:backend backend)
|
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
;; XXX: For some reason we need to throw in some 'basename'.
|
|
|
|
|
(return (and (match nodes
|
|
|
|
|
(((ids labels) ...)
|
|
|
|
|
(let ((ids (map basename ids)))
|
|
|
|
|
(every (lambda (item)
|
|
|
|
|
(member (basename item) ids))
|
|
|
|
|
(list txt
|
|
|
|
|
(derivation-file-name drv)
|
|
|
|
|
(derivation-file-name guile))))))
|
|
|
|
|
(every (cut member <>
|
|
|
|
|
(map (lambda (edge)
|
|
|
|
|
(map basename edge))
|
|
|
|
|
edges))
|
|
|
|
|
(list (map (compose basename derivation-file-name)
|
|
|
|
|
(list drv guile))
|
|
|
|
|
(list (basename (derivation-file-name drv))
|
|
|
|
|
(basename txt))))))))))))
|
|
|
|
|
|
|
|
|
|
(test-assert "reference DAG"
|
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(mlet* %store-monad ((txt (text-file "text-file" "Hello!"))
|
|
|
|
|
(guile (package->derivation %bootstrap-guile))
|
|
|
|
|
(drv (gexp->derivation "output"
|
|
|
|
|
#~(symlink #$txt #$output)
|
|
|
|
|
#:guile-for-build
|
|
|
|
|
guile))
|
|
|
|
|
(out -> (derivation->output-path drv)))
|
|
|
|
|
;; We should see only OUT and TXT, with an edge from the former to the
|
|
|
|
|
;; latter.
|
|
|
|
|
(mbegin %store-monad
|
|
|
|
|
(built-derivations (list drv))
|
|
|
|
|
(export-graph (list (derivation->output-path drv)) 'port
|
|
|
|
|
#:node-type %reference-node-type
|
|
|
|
|
#:backend backend)
|
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(return
|
|
|
|
|
(and (equal? (match nodes
|
|
|
|
|
(((ids labels) ...)
|
|
|
|
|
ids))
|
|
|
|
|
(list out txt))
|
|
|
|
|
(equal? edges `((,out ,txt)))))))))))
|
|
|
|
|
|
2016-10-15 22:47:42 +02:00
|
|
|
|
(test-assert "referrer DAG"
|
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(mlet* %store-monad ((txt (text-file "referrer-node" (random-text)))
|
|
|
|
|
(drv (gexp->derivation "referrer"
|
|
|
|
|
#~(symlink #$txt #$output)))
|
|
|
|
|
(out -> (derivation->output-path drv)))
|
|
|
|
|
;; We should see only TXT and OUT, with an edge from the former to the
|
|
|
|
|
;; latter.
|
|
|
|
|
(mbegin %store-monad
|
|
|
|
|
(built-derivations (list drv))
|
|
|
|
|
(export-graph (list txt) 'port
|
|
|
|
|
#:node-type %referrer-node-type
|
|
|
|
|
#:backend backend)
|
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(return
|
|
|
|
|
(and (equal? (match nodes
|
|
|
|
|
(((ids labels) ...)
|
|
|
|
|
ids))
|
|
|
|
|
(list txt out))
|
|
|
|
|
(equal? edges `((,txt ,out)))))))))))
|
|
|
|
|
|
2018-03-27 14:00:48 +02:00
|
|
|
|
(test-assert "module graph"
|
|
|
|
|
(let-values (((backend nodes+edges) (make-recording-backend)))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(export-graph '((gnu packages guile)) 'port
|
|
|
|
|
#:node-type %module-node-type
|
|
|
|
|
#:backend backend))
|
|
|
|
|
|
|
|
|
|
(let-values (((nodes edges) (nodes+edges)))
|
|
|
|
|
(and (member '(gnu packages guile)
|
|
|
|
|
(match nodes
|
|
|
|
|
(((ids labels) ...) ids)))
|
|
|
|
|
(->bool (and (member (list '(gnu packages guile)
|
|
|
|
|
'(gnu packages libunistring))
|
|
|
|
|
edges)
|
|
|
|
|
(member (list '(gnu packages guile)
|
|
|
|
|
'(gnu packages bdw-gc))
|
|
|
|
|
edges)))))))
|
|
|
|
|
|
2015-11-21 14:48:34 +01:00
|
|
|
|
(test-assert "node-edges"
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(let ((packages (fold-packages cons '())))
|
|
|
|
|
(mlet %store-monad ((edges (node-edges %package-node-type packages)))
|
2018-06-06 13:58:43 +02:00
|
|
|
|
(return (and (null? (edges hello))
|
2015-11-21 14:48:34 +01:00
|
|
|
|
(lset= eq?
|
|
|
|
|
(edges guile-2.0)
|
|
|
|
|
(match (package-direct-inputs guile-2.0)
|
|
|
|
|
(((labels packages _ ...) ...)
|
|
|
|
|
packages)))))))))
|
|
|
|
|
|
2023-08-21 15:13:59 +02:00
|
|
|
|
(test-equal "node-transitive-edges + node-back-edges"
|
|
|
|
|
'()
|
2015-11-21 14:48:34 +01:00
|
|
|
|
(run-with-store %store
|
|
|
|
|
(let ((packages (fold-packages cons '()))
|
|
|
|
|
(bootstrap? (lambda (package)
|
|
|
|
|
(string-contains
|
|
|
|
|
(location-file (package-location package))
|
|
|
|
|
"bootstrap.scm")))
|
|
|
|
|
(trivial? (lambda (package)
|
|
|
|
|
(eq? (package-build-system package)
|
2023-08-21 15:13:59 +02:00
|
|
|
|
trivial-build-system)))
|
|
|
|
|
(system-specific? (lambda (package)
|
|
|
|
|
(memq #:system (package-arguments package)))))
|
2015-11-21 14:48:34 +01:00
|
|
|
|
(mlet %store-monad ((edges (node-back-edges %bag-node-type packages)))
|
|
|
|
|
(let* ((glibc (canonical-package glibc))
|
|
|
|
|
(dependents (node-transitive-edges (list glibc) edges))
|
|
|
|
|
(diff (lset-difference eq? packages dependents)))
|
2023-08-21 15:13:59 +02:00
|
|
|
|
;; All the packages depend on libc, except bootstrap packages, some
|
|
|
|
|
;; packages that use TRIVIAL-BUILD-SYSTEM, and some that target a
|
|
|
|
|
;; specific system and thus may depend on a different libc package
|
|
|
|
|
;; object.
|
|
|
|
|
(return (remove (lambda (package)
|
|
|
|
|
(or (trivial? package)
|
|
|
|
|
(bootstrap? package)
|
|
|
|
|
(system-specific? package)))
|
|
|
|
|
diff)))))))
|
2015-11-21 14:48:34 +01:00
|
|
|
|
|
2015-12-13 21:41:52 +01:00
|
|
|
|
(test-assert "node-transitive-edges, no duplicates"
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(let* ((p0 (dummy-package "p0"))
|
|
|
|
|
(p1a (dummy-package "p1a" (inputs `(("p0" ,p0)))))
|
|
|
|
|
(p1b (dummy-package "p1b" (inputs `(("p0" ,p0)))))
|
|
|
|
|
(p2 (dummy-package "p2" (inputs `(("p1a" ,p1a) ("p1b" ,p1b))))))
|
|
|
|
|
(mlet %store-monad ((edges (node-edges %package-node-type
|
|
|
|
|
(list p2 p1a p1b p0))))
|
|
|
|
|
(return (lset= eq? (node-transitive-edges (list p2) edges)
|
|
|
|
|
(list p1a p1b p0)))))))
|
|
|
|
|
|
2020-05-10 00:04:59 +02:00
|
|
|
|
(test-assert "node-transitive-edges, references"
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(mlet* %store-monad ((d0 (package->derivation %bootstrap-guile))
|
|
|
|
|
(d1 (gexp->derivation "d1"
|
|
|
|
|
#~(begin
|
|
|
|
|
(mkdir #$output)
|
|
|
|
|
(symlink #$%bootstrap-guile
|
|
|
|
|
(string-append
|
|
|
|
|
#$output "/l")))))
|
|
|
|
|
(d2 (gexp->derivation "d2"
|
|
|
|
|
#~(begin
|
|
|
|
|
(mkdir #$output)
|
|
|
|
|
(symlink #$d1
|
|
|
|
|
(string-append
|
|
|
|
|
#$output "/l")))))
|
|
|
|
|
(_ (built-derivations (list d2)))
|
|
|
|
|
(->node -> (node-type-convert %reference-node-type))
|
|
|
|
|
(o2 (->node (derivation->output-path d2)))
|
|
|
|
|
(o1 (->node (derivation->output-path d1)))
|
|
|
|
|
(o0 (->node (derivation->output-path d0)))
|
|
|
|
|
(edges (node-edges %reference-node-type
|
|
|
|
|
(append o0 o1 o2)))
|
|
|
|
|
(reqs ((store-lift requisites) o2)))
|
|
|
|
|
(return (lset= string=?
|
|
|
|
|
(append o2 (node-transitive-edges o2 edges)) reqs)))))
|
|
|
|
|
|
2016-05-23 23:03:23 +02:00
|
|
|
|
(test-equal "node-reachable-count"
|
|
|
|
|
'(3 3)
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(let* ((p0 (dummy-package "p0"))
|
|
|
|
|
(p1a (dummy-package "p1a" (inputs `(("p0" ,p0)))))
|
|
|
|
|
(p1b (dummy-package "p1b" (inputs `(("p0" ,p0)))))
|
|
|
|
|
(p2 (dummy-package "p2" (inputs `(("p1a" ,p1a) ("p1b" ,p1b))))))
|
|
|
|
|
(mlet* %store-monad ((all -> (list p2 p1a p1b p0))
|
|
|
|
|
(edges (node-edges %package-node-type all))
|
|
|
|
|
(back (node-back-edges %package-node-type all)))
|
|
|
|
|
(return (list (node-reachable-count (list p2) edges)
|
|
|
|
|
(node-reachable-count (list p0) back)))))))
|
|
|
|
|
|
2020-05-10 00:09:05 +02:00
|
|
|
|
(test-equal "shortest-path, packages + derivations"
|
|
|
|
|
'(("p5" "p4" "p1" "p0")
|
|
|
|
|
("p3" "p2" "p1" "p0")
|
|
|
|
|
#f
|
|
|
|
|
("p5-0.drv" "p4-0.drv" "p1-0.drv" "p0-0.drv"))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(let* ((p0 (dummy-package "p0"))
|
|
|
|
|
(p1 (dummy-package "p1" (inputs `(("p0" ,p0)))))
|
|
|
|
|
(p2 (dummy-package "p2" (inputs `(("p1" ,p1)))))
|
|
|
|
|
(p3 (dummy-package "p3" (inputs `(("p2" ,p2)))))
|
|
|
|
|
(p4 (dummy-package "p4" (inputs `(("p1" ,p1)))))
|
|
|
|
|
(p5 (dummy-package "p5" (inputs `(("p4" ,p4) ("p3" ,p3))))))
|
|
|
|
|
(mlet* %store-monad ((path1 (shortest-path p5 p0 %package-node-type))
|
|
|
|
|
(path2 (shortest-path p3 p0 %package-node-type))
|
|
|
|
|
(nope (shortest-path p3 p4 %package-node-type))
|
|
|
|
|
(drv5 (package->derivation p5))
|
|
|
|
|
(drv0 (package->derivation p0))
|
|
|
|
|
(path3 (shortest-path drv5 drv0
|
|
|
|
|
%derivation-node-type)))
|
|
|
|
|
(return (append (map (lambda (path)
|
|
|
|
|
(and path (map package-name path)))
|
|
|
|
|
(list path1 path2 nope))
|
|
|
|
|
(list (map (node-type-label %derivation-node-type)
|
|
|
|
|
path3))))))))
|
|
|
|
|
|
|
|
|
|
(test-equal "shortest-path, reverse packages"
|
|
|
|
|
'("libffi" "guile" "guile-json")
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(mlet %store-monad ((path (shortest-path (specification->package "libffi")
|
2023-02-27 01:11:34 +01:00
|
|
|
|
guile-json-1
|
2020-05-10 00:09:05 +02:00
|
|
|
|
%reverse-package-node-type)))
|
|
|
|
|
(return (map package-name path)))))
|
|
|
|
|
|
|
|
|
|
(test-equal "shortest-path, references"
|
|
|
|
|
`(("d2" "d1" ,(package-full-name %bootstrap-guile "-"))
|
|
|
|
|
(,(package-full-name %bootstrap-guile "-") "d1" "d2"))
|
|
|
|
|
(run-with-store %store
|
|
|
|
|
(mlet* %store-monad ((d0 (package->derivation %bootstrap-guile))
|
|
|
|
|
(d1 (gexp->derivation "d1"
|
|
|
|
|
#~(begin
|
|
|
|
|
(mkdir #$output)
|
|
|
|
|
(symlink #$%bootstrap-guile
|
|
|
|
|
(string-append
|
|
|
|
|
#$output "/l")))))
|
|
|
|
|
(d2 (gexp->derivation "d2"
|
|
|
|
|
#~(begin
|
|
|
|
|
(mkdir #$output)
|
|
|
|
|
(symlink #$d1
|
|
|
|
|
(string-append
|
|
|
|
|
#$output "/l")))))
|
|
|
|
|
(_ (built-derivations (list d2)))
|
|
|
|
|
(->node -> (node-type-convert %reference-node-type))
|
|
|
|
|
(o2 (->node (derivation->output-path d2)))
|
|
|
|
|
(o0 (->node (derivation->output-path d0)))
|
|
|
|
|
(path (shortest-path (first o2) (first o0)
|
|
|
|
|
%reference-node-type))
|
|
|
|
|
(rpath (shortest-path (first o0) (first o2)
|
|
|
|
|
%referrer-node-type)))
|
|
|
|
|
(return (list (map (node-type-label %reference-node-type) path)
|
|
|
|
|
(map (node-type-label %referrer-node-type) rpath))))))
|
|
|
|
|
|
2015-08-27 00:36:41 +02:00
|
|
|
|
(test-end "graph")
|