123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531 |
- ;;; GNU Guix --- Functional package management for GNU
- ;;; Copyright © 2015, 2016, 2017, 2019, 2020, 2021 Ludovic Courtès <ludo@gnu.org>
- ;;;
- ;;; 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 (guix scripts challenge)
- #:use-module (guix ui)
- #:use-module (guix scripts)
- #:use-module (guix store)
- #:use-module (guix utils)
- #:use-module (guix grafts)
- #:use-module (guix monads)
- #:use-module (guix base32)
- #:use-module (guix packages)
- #:use-module ((guix progress) #:hide (dump-port*))
- #:use-module (guix serialization)
- #:use-module (guix substitutes)
- #:use-module (guix narinfo)
- #:use-module (rnrs bytevectors)
- #:autoload (guix http-client) (http-fetch)
- #:use-module ((guix build syscalls) #:select (terminal-columns))
- #:use-module (gcrypt hash)
- #:use-module (srfi srfi-1)
- #:use-module (srfi srfi-9)
- #:use-module (srfi srfi-26)
- #:use-module (srfi srfi-34)
- #:use-module (srfi srfi-37)
- #:use-module (srfi srfi-71)
- #:use-module (ice-9 match)
- #:use-module (ice-9 vlist)
- #:use-module (ice-9 format)
- #:use-module (ice-9 ftw)
- #:use-module (web uri)
- #:export (compare-contents
- comparison-report?
- comparison-report-item
- comparison-report-result
- comparison-report-local-sha256
- comparison-report-narinfos
- comparison-report-match?
- comparison-report-mismatch?
- comparison-report-inconclusive?
- differing-files
- call-with-mismatches
- guix-challenge))
- ;;; Commentary:
- ;;;
- ;;; Challenge substitute servers, checking whether they provide the same
- ;;; binaries as those built locally.
- ;;;
- ;;; Here we completely bypass the daemon to access substitutes. This is
- ;;; because we want to be able to report fine-grain information about
- ;;; discrepancies: We need to show the URL of the offending nar, its hash, and
- ;;; so on.
- ;;;
- ;;; Code:
- (define ensure-store-item ;XXX: move to (guix ui)?
- (@@ (guix scripts size) ensure-store-item))
- ;; Representation of a comparison report for ITEM.
- (define-record-type <comparison-report>
- (%comparison-report item result local-sha256 narinfos)
- comparison-report?
- (item comparison-report-item) ;string, /gnu/store/… item
- (result comparison-report-result) ;'match | 'mismatch | 'inconclusive
- (local-sha256 comparison-report-local-sha256) ;bytevector | #f
- (narinfos comparison-report-narinfos)) ;list of <narinfo>
- (define-syntax comparison-report
- ;; Some sort of a an enum to make sure 'result' is correct.
- (syntax-rules (match mismatch inconclusive)
- ((_ item 'match rest ...)
- (%comparison-report item 'match rest ...))
- ((_ item 'mismatch rest ...)
- (%comparison-report item 'mismatch rest ...))
- ((_ item 'inconclusive rest ...)
- (%comparison-report item 'inconclusive rest ...))))
- (define (comparison-report-predicate result)
- "Return a predicate that returns true when pass a REPORT that has RESULT."
- (lambda (report)
- (eq? (comparison-report-result report) result)))
- (define comparison-report-mismatch?
- (comparison-report-predicate 'mismatch))
- (define comparison-report-match?
- (comparison-report-predicate 'match))
- (define comparison-report-inconclusive?
- (comparison-report-predicate 'inconclusive))
- (define (locally-built? store item)
- "Return true if ITEM was built locally."
- ;; XXX: For now approximate it by checking whether there's a build log for
- ;; ITEM. There could be false negatives, if logs have been removed.
- (->bool (log-file store item)))
- (define (query-locally-built-hash item)
- "Return the hash of ITEM, a store item, if ITEM was built locally.
- Otherwise return #f."
- (lambda (store)
- (guard (c ((store-protocol-error? c)
- (values #f store)))
- (if (locally-built? store item)
- (values (query-path-hash store item) store)
- (values #f store)))))
- (define-syntax-rule (report args ...)
- (format (current-error-port) args ...))
- (define (compare-contents items servers)
- "Challenge the substitute servers whose URLs are listed in SERVERS by
- comparing the hash of the substitutes of ITEMS that they serve. Return the
- list of <comparison-report> objects.
- This procedure does not authenticate narinfos from SERVERS, nor does it verify
- that they are signed by an authorized public keys. The reason is that, by
- definition, we may want to target unknown servers. Furthermore, no risk is
- taken since we do not import the archives."
- (define (compare item reference)
- ;; Return a procedure to compare the hash of ITEM with REFERENCE.
- (lambda (narinfo url)
- (or (not narinfo)
- (let ((value (narinfo-hash->sha256 (narinfo-hash narinfo))))
- (bytevector=? reference value)))))
- (define (select-reference item narinfos urls)
- ;; Return a "reference" narinfo among NARINFOS.
- (match narinfos
- ((first narinfos ...)
- (match servers
- ((url urls ...)
- (if (not first)
- (select-reference item narinfos urls)
- (narinfo-hash->sha256 (narinfo-hash first))))))))
- (mlet* %store-monad ((local (mapm %store-monad
- query-locally-built-hash items))
- (remote -> (append-map (cut lookup-narinfos <> items)
- servers))
- ;; No 'assert-valid-narinfo' on purpose.
- (narinfos -> (fold (lambda (narinfo vhash)
- (vhash-cons (narinfo-path narinfo) narinfo
- vhash))
- vlist-null
- remote)))
- (return (map (lambda (item local)
- (match (vhash-fold* cons '() item narinfos)
- (() ;no substitutes
- (comparison-report item 'inconclusive local '()))
- ((narinfo)
- (if local
- (if ((compare item local) narinfo (first servers))
- (comparison-report item 'match
- local (list narinfo))
- (comparison-report item 'mismatch
- local (list narinfo)))
- (comparison-report item 'inconclusive
- local (list narinfo))))
- ((narinfos ...)
- (let ((reference
- (or local (select-reference item narinfos
- servers))))
- (if (every (compare item reference) narinfos servers)
- (comparison-report item 'match
- local narinfos)
- (comparison-report item 'mismatch
- local narinfos))))))
- items
- local))))
- ;;;
- ;;; Reporting.
- ;;;
- (define (port-sha256* port size)
- ;; Like 'port-sha256', but limited to SIZE bytes.
- (let ((out get (open-sha256-port)))
- (dump-port* port out size)
- (close-port out)
- (get)))
- (define (archive-contents port)
- "Return a list representing the files contained in the nar read from PORT.
- The list is sorted in canonical order--i.e., the order in which entries appear
- in the nar."
- (reverse
- (fold-archive (lambda (file type contents result)
- (match type
- ((or 'regular 'executable)
- (match contents
- ((port . size)
- (cons `(,file ,type ,(port-sha256* port size))
- result))))
- ('directory result)
- ('directory-complete result)
- ('symlink
- (cons `(,file ,type ,contents) result))))
- '()
- port
- "")))
- (define (store-item-contents item)
- "Return a list of files and contents for ITEM in the same format as
- 'archive-contents'."
- (let loop ((file item))
- (define stat
- (lstat file))
- (define short
- (string-drop file (string-length item)))
- (match (stat:type stat)
- ('regular
- (let ((size (stat:size stat))
- (type (if (zero? (logand (stat:mode stat)
- #o100))
- 'regular
- 'executable)))
- `((,short ,type
- ,(call-with-input-file file
- (cut port-sha256* <> size))))))
- ('symlink
- `((,short symlink ,(readlink file))))
- ('directory
- (append-map (match-lambda
- ((or "." "..")
- '())
- (entry
- (loop (string-append file "/" entry))))
- ;; Traverse entries in canonical order, the same as the
- ;; order of entries in nars.
- (scandir file (const #t) string<?))))))
- (define (call-with-nar narinfo proc)
- "Call PROC with an input port from which it can read the nar pointed to by
- NARINFO."
- (let* ((uri compression size (narinfo-best-uri narinfo))
- (port actual-size (http-fetch uri)))
- (define reporter
- (progress-reporter/file (narinfo-path narinfo)
- (and size
- (max size (or actual-size 0))) ;defensive
- #:abbreviation (const (uri-host uri))))
- (define result
- (call-with-decompressed-port (string->symbol compression)
- (progress-report-port reporter port)
- proc))
- (close-port port)
- (erase-current-line (current-output-port))
- result))
- (define (narinfo-contents narinfo)
- "Fetch the nar described by NARINFO and return a list representing the file
- it contains."
- (call-with-nar narinfo archive-contents))
- (define (differing-files comparison-report)
- "Return a list of files that differ among the nars and possibly the local
- store item specified in COMPARISON-REPORT."
- (define contents
- (map narinfo-contents
- (comparison-report-narinfos comparison-report)))
- (define local-contents
- (and (comparison-report-local-sha256 comparison-report)
- (store-item-contents (comparison-report-item comparison-report))))
- (match (apply lset-difference equal?
- (take (delete-duplicates
- (if local-contents
- (cons local-contents contents)
- contents))
- 2))
- (((files _ ...) ...)
- files)))
- (define (report-differing-files comparison-report)
- "Report differences among the nars and possibly the local store item
- specified in COMPARISON-REPORT."
- (match (differing-files comparison-report)
- (()
- #t)
- ((files ...)
- (format #t (N_ " differing file:~%"
- " differing files:~%"
- (length files)))
- (format #t "~{ ~a~%~}" files))))
- (define (call-with-mismatches comparison-report proc)
- "Call PROC with two directories containing the mismatching store items."
- (define local-hash
- (comparison-report-local-sha256 comparison-report))
- (define narinfos
- (comparison-report-narinfos comparison-report))
- (call-with-temporary-directory
- (lambda (directory1)
- (call-with-temporary-directory
- (lambda (directory2)
- (define narinfo1
- (if local-hash
- (find (lambda (narinfo)
- (not (bytevector=? (narinfo-hash->sha256
- (narinfo-hash narinfo))
- local-hash)))
- narinfos)
- (first (comparison-report-narinfos comparison-report))))
- (define narinfo2
- (and (not local-hash)
- (find (lambda (narinfo)
- (not (eq? narinfo narinfo1)))
- narinfos)))
- (rmdir directory1)
- (call-with-nar narinfo1 (cut restore-file <> directory1))
- (when narinfo2
- (rmdir directory2)
- (call-with-nar narinfo2 (cut restore-file <> directory2)))
- (proc directory1
- (if local-hash
- (comparison-report-item comparison-report)
- directory2)))))))
- (define %diffoscope-command
- ;; Default external diff command. Pass "--exclude-directory-metadata" so
- ;; that the mtime/ctime differences are ignored.
- '("diffoscope" "--exclude-directory-metadata=yes"))
- (define* (report-differing-files/external comparison-report
- #:optional
- (command %diffoscope-command))
- "Run COMMAND to show the file-level differences for the mismatches in
- COMPARISON-REPORT."
- (call-with-mismatches comparison-report
- (lambda (directory1 directory2)
- (apply system*
- (append command
- (list directory1 directory2))))))
- (define* (summarize-report comparison-report
- #:key
- (report-differences (const #f))
- (hash->string bytevector->nix-base32-string)
- verbose?)
- "Write to the current error port a summary of COMPARISON-REPORT, a
- <comparison-report> object. When VERBOSE?, display matches in addition to
- mismatches and inconclusive reports. Upon mismatch, call REPORT-DIFFERENCES
- with COMPARISON-REPORT."
- (define (report-hashes item local narinfos)
- (if local
- (report (G_ " local hash: ~a~%") (hash->string local))
- (report (G_ " no local build for '~a'~%") item))
- (for-each (lambda (narinfo)
- (report (G_ " ~50a: ~a~%")
- (uri->string (narinfo-best-uri narinfo))
- (hash->string
- (narinfo-hash->sha256 (narinfo-hash narinfo)))))
- narinfos))
- (match comparison-report
- (($ <comparison-report> item 'mismatch local (narinfos ...))
- (report (G_ "~a contents differ:~%") item)
- (report-hashes item local narinfos)
- (report-differences comparison-report))
- (($ <comparison-report> item 'inconclusive #f narinfos)
- (warning (G_ "could not challenge '~a': no local build~%") item))
- (($ <comparison-report> item 'inconclusive locals ())
- (warning (G_ "could not challenge '~a': no substitutes~%") item))
- (($ <comparison-report> item 'match local (narinfos ...))
- (when verbose?
- (report (G_ "~a contents match:~%") item)
- (report-hashes item local narinfos)))))
- (define (summarize-report-list reports)
- "Display the overall summary of REPORTS."
- (let ((total (length reports))
- (inconclusive (count comparison-report-inconclusive? reports))
- (matches (count comparison-report-match? reports))
- (discrepancies (count comparison-report-mismatch? reports)))
- (report (G_ "~h store items were analyzed:~%") total)
- (report (G_ " - ~h (~,1f%) were identical~%")
- matches (* 100. (/ matches total)))
- (report (G_ " - ~h (~,1f%) differed~%")
- discrepancies (* 100. (/ discrepancies total)))
- (report (G_ " - ~h (~,1f%) were inconclusive~%")
- inconclusive (* 100. (/ inconclusive total)))))
- ;;;
- ;;; Command-line options.
- ;;;
- (define (show-help)
- (display (G_ "Usage: guix challenge [PACKAGE...]
- Challenge the substitutes for PACKAGE... provided by one or more servers.\n"))
- (display (G_ "
- --substitute-urls=URLS
- compare build results with those at URLS"))
- (display (G_ "
- -v, --verbose show details about successful comparisons"))
- (display (G_ "
- --diff=MODE show differences according to MODE"))
- (newline)
- (display (G_ "
- -h, --help display this help and exit"))
- (display (G_ "
- -V, --version display version information and exit"))
- (newline)
- (show-bug-report-information))
- (define %options
- (list (option '(#\h "help") #f #f
- (lambda args
- (show-help)
- (exit 0)))
- (option '(#\V "version") #f #f
- (lambda args
- (show-version-and-exit "guix challenge")))
- (option '("diff") #t #f
- (lambda (opt name arg result . rest)
- (define mode
- (match arg
- ("none" (const #t))
- ("simple" report-differing-files)
- ("diffoscope" report-differing-files/external)
- ((and (? (cut string-prefix? "/" <>)) command)
- (cute report-differing-files/external <>
- (string-tokenize command)))
- (_ (leave (G_ "~a: unknown diff mode~%") arg))))
- (apply values
- (alist-cons 'difference-report mode result)
- rest)))
- (option '("substitute-urls") #t #f
- (lambda (opt name arg result . rest)
- (apply values
- (alist-cons 'substitute-urls
- (string-tokenize arg)
- (alist-delete 'substitute-urls result))
- rest)))
- (option '("verbose" #\v) #f #f
- (lambda (opt name arg result . rest)
- (apply values
- (alist-cons 'verbose? #t result)
- rest)))))
- (define %default-options
- `((system . ,(%current-system))
- (substitute-urls . ,%default-substitute-urls)
- (difference-report . ,report-differing-files)))
- ;;;
- ;;; Entry point.
- ;;;
- (define-command (guix-challenge . args)
- (category packaging)
- (synopsis "challenge substitute servers, comparing their binaries")
- (with-error-handling
- (let* ((opts (parse-command-line args %options (list %default-options)
- #:build-options? #f))
- (files (filter-map (match-lambda
- (('argument . file) file)
- (_ #f))
- opts))
- (system (assoc-ref opts 'system))
- (urls (assoc-ref opts 'substitute-urls))
- (diff (assoc-ref opts 'difference-report))
- (verbose? (assoc-ref opts 'verbose?)))
- (leave-on-EPIPE
- (with-store store
- ;; Disable grafts since substitute servers normally provide only
- ;; ungrafted stuff.
- (parameterize ((%graft? #f)
- (current-terminal-columns (terminal-columns)))
- (let ((files (match files
- (()
- (filter (cut locally-built? store <>)
- (live-paths store)))
- (x
- files))))
- (set-build-options store
- #:use-substitutes? #f)
- (run-with-store store
- (mlet* %store-monad ((items (mapm %store-monad
- ensure-store-item files))
- (reports (compare-contents items urls)))
- (for-each (cut summarize-report <> #:verbose? verbose?
- #:report-differences diff)
- reports)
- (report "\n")
- (summarize-report-list reports)
- (exit (cond ((any comparison-report-mismatch? reports) 2)
- ((every comparison-report-match? reports) 0)
- (else 1))))
- #:system system))))))))
- ;;; challenge.scm ends here
|