123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104 |
- ;;; GNU Guix --- Functional package management for GNU
- ;;; Copyright © 2012-2022 Ludovic Courtès <ludo@gnu.org>
- ;;; Copyright © 2013 Nikita Karetnikov <nikita@karetnikov.org>
- ;;; Copyright © 2013, 2015 Mark H Weaver <mhw@netris.org>
- ;;; Copyright © 2014, 2016 Alex Kost <alezost@gmail.com>
- ;;; Copyright © 2016 Roel Janssen <roel@gnu.org>
- ;;; Copyright © 2016 Benz Schenk <benz.schenk@uzh.ch>
- ;;; Copyright © 2016 Chris Marusich <cmmarusich@gmail.com>
- ;;; Copyright © 2019 Tobias Geerinckx-Rice <me@tobias.gr>
- ;;; Copyright © 2020 Ricardo Wurmus <rekado@elephly.net>
- ;;; Copyright © 2020 Simon Tournier <zimon.toutoune@gmail.com>
- ;;; Copyright © 2018 Steve Sprang <scs@stevesprang.com>
- ;;;
- ;;; 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 package)
- #:use-module (guix ui)
- #:use-module ((guix status) #:select (with-status-verbosity))
- #:use-module ((guix build syscalls) #:select (terminal-rows))
- #:use-module (guix store)
- #:use-module (guix grafts)
- #:use-module (guix derivations)
- #:use-module (guix packages)
- #:use-module (guix profiles)
- #:use-module (guix search-paths)
- #:autoload (guix import json) (json->scheme-file)
- #:use-module (guix monads)
- #:use-module (guix utils)
- #:use-module (guix config)
- #:use-module (guix scripts)
- #:use-module (guix scripts build)
- #:use-module (guix transformations)
- #:autoload (guix describe) (manifest-entry-provenance
- manifest-entry-with-provenance)
- #:autoload (guix channels) (channel-name channel-commit channel->code)
- #:autoload (guix store roots) (gc-roots user-owned?)
- #:use-module ((guix build utils)
- #:select (directory-exists? mkdir-p))
- #:use-module (ice-9 format)
- #:use-module (ice-9 match)
- #:autoload (ice-9 pretty-print) (pretty-print)
- #:use-module (ice-9 regex)
- #:use-module (ice-9 vlist)
- #:use-module (srfi srfi-1)
- #:use-module (srfi srfi-11)
- #:use-module (srfi srfi-26)
- #:use-module (srfi srfi-34)
- #:use-module (srfi srfi-35)
- #:use-module (srfi srfi-37)
- #:use-module (gnu packages)
- #:autoload (gnu packages bootstrap) (%bootstrap-guile)
- #:export (build-and-use-profile
- delete-generations
- delete-matching-generations
- guix-package
- search-path-environment-variables
- manifest-entry-version-prefix
- transaction-upgrade-entry ;mostly for testing
- (%options . %package-options)
- (%default-options . %package-default-options)
- guix-package*))
- (define %store
- (make-parameter #f))
- ;;;
- ;;; Profiles.
- ;;;
- (define (ensure-default-profile)
- "Ensure the default profile symlink and directory exist and are writable."
- (ensure-profile-directory)
- ;; Try to create ~/.guix-profile if it doesn't exist yet.
- (when (and %user-profile-directory
- %current-profile
- (not (false-if-exception
- (lstat %user-profile-directory))))
- (catch 'system-error
- (lambda ()
- (symlink %current-profile %user-profile-directory))
- (const #t))))
- (define (delete-generations store profile generations)
- "Delete GENERATIONS from PROFILE.
- GENERATIONS is a list of generation numbers."
- (for-each (cut delete-generation* store profile <>)
- generations))
- (define (delete-matching-generations store profile pattern)
- "Delete from PROFILE all the generations matching PATTERN. PATTERN must be
- a string denoting a set of generations: the empty list means \"all generations
- but the current one\", a number designates a generation, and other patterns
- denote ranges as interpreted by 'matching-generations'."
- (let ((current (generation-number profile)))
- (cond ((not (file-exists? profile)) ; XXX: race condition
- (raise (condition (&profile-not-found-error
- (profile profile)))))
- ((not pattern)
- (delete-generations store profile
- (delv current (profile-generations profile))))
- ;; Do not delete the zeroth generation.
- ((equal? 0 (string->number pattern))
- #t)
- ;; If PATTERN is a duration, match generations that are
- ;; older than the specified duration.
- ((matching-generations pattern profile
- #:duration-relation >)
- =>
- (lambda (numbers)
- (when (memv current numbers)
- (warning (G_ "not removing generation ~a, which is current~%")
- current))
- ;; Make sure we don't inadvertently remove the current
- ;; generation.
- (let ((numbers (delv current numbers)))
- (when (null-list? numbers)
- (leave (G_ "no matching generation~%")))
- (delete-generations store profile numbers)))))))
- (define* (build-and-use-profile store profile manifest
- #:key
- dry-run?
- (hooks %default-profile-hooks)
- allow-collisions?
- bootstrap?)
- "Build a new generation of PROFILE, a file name, using the packages
- specified in MANIFEST, a manifest object. When ALLOW-COLLISIONS? is true,
- do not treat collisions in MANIFEST as an error. HOOKS is a list of \"profile
- hooks\" run when building the profile."
- (let* ((prof-drv (run-with-store store
- (profile-derivation manifest
- #:allow-collisions? allow-collisions?
- #:hooks (if bootstrap? '() hooks)
- #:locales? (not bootstrap?))))
- (prof (derivation->output-path prof-drv)))
- (cond
- (dry-run? #t)
- ((and (file-exists? profile)
- (and=> (readlink* profile) (cut string=? prof <>)))
- (format (current-error-port) (G_ "nothing to be done~%")))
- (else
- (let* ((number (generation-number profile))
- ;; Always use NUMBER + 1 for the new profile, possibly
- ;; overwriting a "previous future generation".
- (name (generation-file-name profile (+ 1 number))))
- (and (build-derivations store (list prof-drv))
- (let* ((entries (manifest-entries manifest))
- (count (length entries)))
- (switch-symlinks name prof)
- (switch-symlinks profile (basename name))
- (unless (string=? profile %current-profile)
- (register-gc-root store name))
- (display-search-path-hint entries profile)))
- (warn-about-disk-space profile))))))
- ;;;
- ;;; Package specifications.
- ;;;
- (define (find-packages-by-description regexps)
- "Return a list of pairs: packages whose name, synopsis, description,
- or output matches at least one of REGEXPS sorted by relevance, and its
- non-zero relevance score."
- (let ((matches (fold-packages (lambda (package result)
- (if (package-superseded package)
- result
- (match (package-relevance package
- regexps)
- ((? zero?)
- result)
- (score
- (cons (cons package score)
- result)))))
- '())))
- (sort matches
- (lambda (m1 m2)
- (match m1
- ((package1 . score1)
- (match m2
- ((package2 . score2)
- (if (= score1 score2)
- (string>? (package-full-name package1)
- (package-full-name package2))
- (> score1 score2))))))))))
- (define (transaction-upgrade-entry store entry transaction)
- "Return a variant of TRANSACTION that accounts for the upgrade of ENTRY, a
- <manifest-entry>."
- (define (lower-manifest-entry* entry)
- (run-with-store store
- (lower-manifest-entry entry (%current-system))))
- (define (supersede old new)
- (info (G_ "package '~a' has been superseded by '~a'~%")
- (manifest-entry-name old) (package-name new))
- (manifest-transaction-install-entry
- (package->manifest-entry* new (manifest-entry-output old))
- (manifest-transaction-remove-pattern
- (manifest-pattern
- (name (manifest-entry-name old))
- (version (manifest-entry-version old))
- (output (manifest-entry-output old)))
- transaction)))
- (define (upgrade entry transform)
- (match entry
- (($ <manifest-entry> name version output (? string? path))
- (match (find-best-packages-by-name name #f)
- ((pkg . rest)
- (let* ((pkg (transform pkg))
- (candidate-version (package-version pkg)))
- (match (package-superseded pkg)
- ((? package? new)
- (supersede entry new))
- (#f
- (case (version-compare candidate-version version)
- ((>)
- (manifest-transaction-install-entry
- (package->manifest-entry* pkg output)
- transaction))
- ((<)
- transaction)
- ((=)
- (let* ((new (package->manifest-entry* pkg output)))
- ;; Here we want to determine whether the NEW actually
- ;; differs from ENTRY, but we need to intercept
- ;; 'build-things' calls because they would prevent us from
- ;; displaying the list of packages to install/upgrade
- ;; upfront. Thus, if lowering NEW triggers a build (due
- ;; to grafts), assume NEW differs from ENTRY.
- (if (with-build-handler (const #f)
- (manifest-entry=? (lower-manifest-entry* new)
- entry))
- transaction
- (manifest-transaction-install-entry
- new transaction)))))))))
- (()
- (warning (G_ "package '~a' no longer exists~%") name)
- transaction)))))
- (if (manifest-transaction-removal-candidate? entry transaction)
- transaction
- ;; Upgrade ENTRY, preserving transformation options listed in its
- ;; properties.
- (let ((transform (options->transformation
- (or (assq-ref (manifest-entry-properties entry)
- 'transformations)
- '()))))
- (upgrade entry transform))))
- ;;;
- ;;; Search paths.
- ;;;
- (define* (search-path-environment-variables entries profiles
- #:optional (getenv getenv)
- #:key (kind 'exact))
- "Return environment variable definitions that may be needed for the use of
- ENTRIES, a list of manifest entries, in PROFILES. Use GETENV to determine the
- current settings and report only settings not already effective. KIND
- must be one of 'exact, 'prefix, or 'suffix, depending on the kind of search
- path definition to be returned."
- (let ((search-paths (delete-duplicates
- (cons $PATH
- (append-map manifest-entry-search-paths
- entries)))))
- (filter-map (match-lambda
- ((spec . value)
- (let ((variable (search-path-specification-variable spec))
- (sep (search-path-specification-separator spec)))
- (environment-variable-definition variable value
- #:separator sep
- #:kind kind))))
- (evaluate-search-paths search-paths profiles
- getenv))))
- (define (absolutize file)
- "Return an absolute file name equivalent to FILE, but without resolving
- symlinks like 'canonicalize-path' would do."
- (if (string-prefix? "/" file)
- file
- (string-append (getcwd) "/" file)))
- (define (display-search-path-hint entries profile)
- "Display a hint on how to set environment variables to use ENTRIES, a list
- of manifest entries, in the context of PROFILE."
- (let* ((profile (user-friendly-profile (absolutize profile)))
- (settings (search-path-environment-variables entries (list profile)
- #:kind 'prefix)))
- (unless (null? settings)
- (display-hint (format #f (G_ "Consider setting the necessary environment
- variables by running:
- @example
- GUIX_PROFILE=\"~a\"
- . \"$GUIX_PROFILE/etc/profile\"
- @end example
- Alternately, see @command{guix package --search-paths -p ~s}.")
- profile profile)))))
- ;;;
- ;;; Export a manifest.
- ;;;
- (define (manifest-entry-version-prefix entry)
- "Search among all the versions of ENTRY's package that are available, and
- return the shortest unambiguous version prefix for this package. If only one
- version of ENTRY's package is available, return the empty string."
- (package-unique-version-prefix (manifest-entry-name entry)
- (manifest-entry-version entry)))
- (define* (export-manifest manifest
- #:optional (port (current-output-port)))
- "Write to PORT a manifest corresponding to MANIFEST."
- (match (manifest->code manifest
- #:entry-package-version
- manifest-entry-version-prefix)
- (('begin exp ...)
- (format port (G_ "\
- ;; This \"manifest\" file can be passed to 'guix package -m' to reproduce
- ;; the content of your profile. This is \"symbolic\": it only specifies
- ;; package names. To reproduce the exact same profile, you also need to
- ;; capture the channels being used, as returned by \"guix describe\".
- ;; See the \"Replicating Guix\" section in the manual.\n"))
- (for-each (lambda (exp)
- (newline port)
- (pretty-print exp port))
- exp))))
- (define (channel=? a b)
- (and (channel-commit a) (channel-commit b)
- (string=? (channel-commit a) (channel-commit b))))
- (define* (export-channels manifest
- #:optional (port (current-output-port)))
- (define channels
- (delete-duplicates
- (append-map manifest-entry-provenance (manifest-entries manifest))
- channel=?))
- (define channel-names
- (delete-duplicates (map channel-name channels)))
- (define table
- (fold (lambda (channel table)
- (vhash-consq (channel-name channel) channel table))
- vlist-null
- channels))
- (when (null? channels)
- (leave (G_ "no provenance information for this profile~%")))
- (format port (G_ "\
- ;; This channel file can be passed to 'guix pull -C' or to
- ;; 'guix time-machine -C' to obtain the Guix revision that was
- ;; used to populate this profile.\n"))
- (newline port)
- (display "(list\n" port)
- (for-each (lambda (name)
- (define indent " ")
- (match (vhash-foldq* cons '() name table)
- ((channel extra ...)
- (unless (null? extra)
- (display indent port)
- (format port (G_ "\
- ;; Note: these other commits were also used to install \
- some of the packages in this profile:~%"))
- (for-each (lambda (channel)
- (format port "~a;; ~s~%"
- indent (channel-commit channel)))
- extra))
- (pretty-print (channel->code channel) port
- #:per-line-prefix indent))))
- channel-names)
- (display ")\n" port)
- #t)
- ;;;
- ;;; Command-line options.
- ;;;
- (define %default-options
- ;; Alist of default option values.
- `((verbosity . 1)
- (debug . 0)
- (graft? . #t)
- (substitutes? . #t)
- (offload? . #t)
- (print-build-trace? . #t)
- (print-extended-build-trace? . #t)
- (multiplexed-build-output? . #t)))
- (define (show-help)
- (display (G_ "Usage: guix package [OPTION]...
- Install, remove, or upgrade packages in a single transaction.\n"))
- (display (G_ "
- -i, --install PACKAGE ...
- install PACKAGEs"))
- (display (G_ "
- -e, --install-from-expression=EXP
- install the package EXP evaluates to"))
- (display (G_ "
- -f, --install-from-file=FILE
- install the package that the code within FILE
- evaluates to"))
- (display (G_ "
- -r, --remove PACKAGE ...
- remove PACKAGEs"))
- (display (G_ "
- -u, --upgrade[=REGEXP] upgrade all the installed packages matching REGEXP"))
- (display (G_ "
- -m, --manifest=FILE create a new profile generation with the manifest
- from FILE"))
- (display (G_ "
- --do-not-upgrade[=REGEXP] do not upgrade any packages matching REGEXP"))
- (display (G_ "
- --roll-back roll back to the previous generation"))
- (display (G_ "
- --search-paths[=KIND]
- display needed environment variable definitions"))
- (display (G_ "
- -l, --list-generations[=PATTERN]
- list generations matching PATTERN"))
- (display (G_ "
- -d, --delete-generations[=PATTERN]
- delete generations matching PATTERN"))
- (display (G_ "
- -S, --switch-generation=PATTERN
- switch to a generation matching PATTERN"))
- (display (G_ "
- --export-manifest print a manifest for the chosen profile"))
- (display (G_ "
- --export-channels print channels for the chosen profile"))
- (display (G_ "
- -p, --profile=PROFILE use PROFILE instead of the user's default profile"))
- (display (G_ "
- --list-profiles list the user's profiles"))
- (newline)
- (display (G_ "
- --allow-collisions do not treat collisions in the profile as an error"))
- (display (G_ "
- --bootstrap use the bootstrap Guile to build the profile"))
- (display (G_ "
- -v, --verbosity=LEVEL use the given verbosity LEVEL"))
- (newline)
- (display (G_ "
- -s, --search=REGEXP search in synopsis and description using REGEXP"))
- (display (G_ "
- -I, --list-installed[=REGEXP]
- list installed packages matching REGEXP"))
- (display (G_ "
- -A, --list-available[=REGEXP]
- list available packages matching REGEXP"))
- (display (G_ "
- --show=PACKAGE show details about PACKAGE"))
- (newline)
- (show-build-options-help)
- (newline)
- (show-transformation-options-help)
- (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
- ;; Specification of the command-line options.
- (cons* (option '(#\h "help") #f #f
- (lambda args
- (show-help)
- (exit 0)))
- (option '(#\V "version") #f #f
- (lambda args
- (show-version-and-exit "guix package")))
- (option '(#\i "install") #f #t
- (lambda (opt name arg result arg-handler)
- (let arg-handler ((arg arg) (result result))
- (values (if arg
- (alist-cons 'install arg result)
- result)
- arg-handler))))
- (option '(#\e "install-from-expression") #t #f
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'install (read/eval-package-expression arg)
- result)
- #f)))
- (option '(#\f "install-from-file") #t #f
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'install
- (let ((file (or (and (string-suffix? ".json" arg)
- (json->scheme-file arg))
- arg)))
- (load* file (make-user-module '())))
- result)
- #f)))
- (option '(#\r "remove") #f #t
- (lambda (opt name arg result arg-handler)
- (let arg-handler ((arg arg) (result result))
- (values (if arg
- (alist-cons 'remove arg result)
- result)
- arg-handler))))
- (option '(#\u "upgrade") #f #t
- (lambda (opt name arg result arg-handler)
- (when (and arg (string-prefix? "-" arg))
- (warning (G_ "upgrade regexp '~a' looks like a \
- command-line option~%")
- arg)
- (warning (G_ "is this intended?~%")))
- (let arg-handler ((arg arg) (result result))
- (values (alist-cons 'upgrade arg
- ;; Delete any prior "upgrade all"
- ;; command, or else "--upgrade gcc"
- ;; would upgrade everything.
- (delete '(upgrade . #f) result))
- arg-handler))))
- (option '("do-not-upgrade") #f #t
- (lambda (opt name arg result arg-handler)
- (let arg-handler ((arg arg) (result result))
- (values (if arg
- (alist-cons 'do-not-upgrade arg result)
- result)
- arg-handler))))
- (option '("roll-back" "rollback") #f #f
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'roll-back? #t result)
- #f)))
- (option '(#\m "manifest") #t #f
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'manifest arg result)
- arg-handler)))
- (option '(#\l "list-generations") #f #t
- (lambda (opt name arg result arg-handler)
- (values (cons `(query list-generations ,arg)
- result)
- #f)))
- (option '("list-profiles") #f #f
- (lambda (opt name arg result arg-handler)
- (values (cons `(query list-profiles #t)
- result)
- #f)))
- (option '(#\d "delete-generations") #f #t
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'delete-generations arg
- result)
- #f)))
- (option '(#\S "switch-generation") #t #f
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'switch-generation arg result)
- #f)))
- (option '("search-paths") #f #t
- (lambda (opt name arg result arg-handler)
- (let ((kind (match arg
- ((or "exact" "prefix" "suffix")
- (string->symbol arg))
- (#f
- 'exact)
- (x
- (leave (G_ "~a: unsupported \
- kind of search path~%")
- x)))))
- (values (cons `(query search-paths ,kind)
- result)
- #f))))
- (option '("export-manifest") #f #f
- (lambda (opt name arg result arg-handler)
- (values (cons `(query export-manifest) result)
- #f)))
- (option '("export-channels") #f #f
- (lambda (opt name arg result arg-handler)
- (values (cons `(query export-channels) result)
- #f)))
- (option '(#\p "profile") #t #f
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'profile (canonicalize-profile arg)
- result)
- #f)))
- (option '(#\n "dry-run") #f #f
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'dry-run? #t result)
- #f)))
- (option '(#\v "verbosity") #t #f
- (lambda (opt name arg result arg-handler)
- (let ((level (string->number* arg)))
- (values (alist-cons 'verbosity level
- (alist-delete 'verbosity result))
- #f))))
- (option '("bootstrap") #f #f
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'bootstrap? #t result)
- #f)))
- (option '("verbose") #f #f ;deprecated
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'verbosity 2
- (alist-delete 'verbosity
- result))
- #f)))
- (option '("allow-collisions") #f #f
- (lambda (opt name arg result arg-handler)
- (values (alist-cons 'allow-collisions? #t result)
- #f)))
- (option '(#\s "search") #t #f
- (lambda (opt name arg result arg-handler)
- (values (cons `(query search ,(or arg ""))
- result)
- #f)))
- (option '(#\I "list-installed") #f #t
- (lambda (opt name arg result arg-handler)
- (values (cons `(query list-installed ,(or arg ""))
- result)
- #f)))
- (option '(#\A "list-available") #f #t
- (lambda (opt name arg result arg-handler)
- (values (cons `(query list-available ,(or arg ""))
- result)
- #f)))
- (option '("show") #t #t
- (lambda (opt name arg result arg-handler)
- (values (cons `(query show ,arg)
- result)
- #f)))
- (append %transformation-options
- %standard-build-options)))
- (define (options->upgrade-predicate opts)
- "Return a predicate based on the upgrade/do-not-upgrade regexps in OPTS
- that, given a package name, returns true if the package is a candidate for
- upgrading, #f otherwise."
- (define upgrade-regexps
- (filter-map (match-lambda
- (('upgrade . regexp)
- (make-regexp* (or regexp "") regexp/icase))
- (_ #f))
- opts))
- (define do-not-upgrade-regexps
- (filter-map (match-lambda
- (('do-not-upgrade . regexp)
- (make-regexp* regexp regexp/icase))
- (_ #f))
- opts))
- (lambda (name)
- (and (any (cut regexp-exec <> name) upgrade-regexps)
- (not (any (cut regexp-exec <> name) do-not-upgrade-regexps)))))
- (define (store-item->manifest-entry item)
- "Return a manifest entry for ITEM, a \"/gnu/store/...\" file name."
- (let-values (((name version)
- (package-name->name+version (store-path-package-name item)
- #\-)))
- (manifest-entry
- (name name)
- (version version)
- (output "out") ;XXX: wild guess
- (item item))))
- (define (package->manifest-entry* package output)
- "Like 'package->manifest-entry', but attach PACKAGE provenance meta-data to
- the resulting manifest entry."
- (manifest-entry-with-provenance
- (package->manifest-entry package output)))
- (define (options->installable opts manifest transaction)
- "Given MANIFEST, the current manifest, and OPTS, the result of 'args-fold',
- return an variant of TRANSACTION that accounts for the specified installations
- and upgrades."
- (define upgrade?
- (options->upgrade-predicate opts))
- (define upgraded
- (fold (lambda (entry transaction)
- (if (upgrade? (manifest-entry-name entry))
- (transaction-upgrade-entry (%store) entry transaction)
- transaction))
- transaction
- (manifest-entries manifest)))
- (define to-install
- (filter-map (match-lambda
- (('install . (? package? p))
- ;; When given a package via `-e', install the first of its
- ;; outputs (XXX).
- (package->manifest-entry* p "out"))
- (('install . (? string? spec))
- (if (store-path? spec)
- (store-item->manifest-entry spec)
- (let-values (((package output)
- (specification->package+output spec)))
- (package->manifest-entry* package output))))
- (('install . obj)
- (leave (G_ "cannot install non-package object: ~s~%")
- obj))
- (_
- #f))
- opts))
- (fold manifest-transaction-install-entry
- upgraded
- to-install))
- (define (options->removable options manifest transaction)
- "Given options, return a variant of TRANSACTION augmented with the list of
- patterns of packages to remove."
- (fold (lambda (opt transaction)
- (match opt
- (('remove . spec)
- (call-with-values
- (lambda ()
- (package-specification->name+version+output spec))
- (lambda (name version output)
- (manifest-transaction-remove-pattern
- (manifest-pattern
- (name name)
- (version version)
- (output output))
- transaction))))
- (_ transaction)))
- transaction
- options))
- (define (register-gc-root store profile)
- "Register PROFILE, a profile generation symlink, as a GC root, unless it
- doesn't need it."
- (define absolute
- ;; We must pass the daemon an absolute file name for PROFILE. However, we
- ;; cannot use (canonicalize-path profile) because that would return us the
- ;; target of PROFILE in the store; using a store item as an indirect root
- ;; would mean that said store item will always remain live, which is not
- ;; what we want here.
- (if (string-prefix? "/" profile)
- profile
- (string-append (getcwd) "/" profile)))
- (add-indirect-root store absolute))
- ;;;
- ;;; Queries and actions.
- ;;;
- (define (process-query opts)
- "Process any query specified by OPTS. Return #t when a query was actually
- processed, #f otherwise."
- (let* ((profiles (delete-duplicates
- (match (filter-map (match-lambda
- (('profile . p) p)
- (_ #f))
- opts)
- (() (list %current-profile))
- (lst (reverse lst)))))
- (profile (match profiles
- ((head tail ...) head))))
- (match (assoc-ref opts 'query)
- (('list-generations pattern)
- (define (list-generation display-function number)
- (unless (zero? number)
- (display-generation profile number)
- (display-function profile number)
- (newline)))
- (define (diff-profiles profile numbers)
- (unless (null-list? (cdr numbers))
- (display-profile-content-diff profile (car numbers) (cadr numbers))
- (diff-profiles profile (cdr numbers))))
- (leave-on-EPIPE
- (cond ((not (file-exists? profile)) ; XXX: race condition
- (raise (condition (&profile-not-found-error
- (profile profile)))))
- ((not pattern)
- (match (profile-generations profile)
- (()
- #t)
- ((first rest ...)
- (list-generation display-profile-content first)
- (diff-profiles profile (cons first rest)))))
- ((matching-generations pattern profile)
- =>
- (lambda (numbers)
- (if (null-list? numbers)
- (exit 1)
- (begin
- (list-generation display-profile-content (car numbers))
- (diff-profiles profile numbers)))))))
- #t)
- (('list-installed regexp)
- (let* ((regexp (and regexp (make-regexp* regexp regexp/icase)))
- (manifest (concatenate-manifests
- (map profile-manifest profiles)))
- (installed (manifest-entries manifest)))
- (leave-on-EPIPE
- (let ((rows (filter-map
- (match-lambda
- (($ <manifest-entry> name version output path _)
- (and (regexp-exec regexp name)
- (list name (or version "?") output path))))
- installed)))
- ;; Show most recently installed packages last.
- (pretty-print-table (reverse rows)))))
- #t)
- (('list-available regexp)
- (let* ((regexp (and regexp (make-regexp* regexp regexp/icase)))
- (available (fold-available-packages
- (lambda* (name version result
- #:key outputs location
- supported? deprecated?
- #:allow-other-keys)
- (if (and supported? (not deprecated?))
- (if regexp
- (if (regexp-exec regexp name)
- (cons `(,name ,version
- ,outputs ,location)
- result)
- result)
- (cons `(,name ,version
- ,outputs ,location)
- result))
- result))
- '())))
- (leave-on-EPIPE
- (let ((rows (map (match-lambda
- ((name version outputs location)
- (list name version (string-join outputs ",")
- (location->string location))))
- (sort available
- (match-lambda*
- (((name1 . _) (name2 . _))
- (string<? name1 name2)))))))
- (pretty-print-table rows)))
- #t))
- (('list-profiles _)
- (let ((profiles (delete-duplicates
- (filter-map (lambda (root)
- (and (or (zero? (getuid))
- (user-owned? root))
- (generation-profile root)))
- (gc-roots)))))
- (leave-on-EPIPE
- (for-each (lambda (profile)
- (display (user-friendly-profile profile))
- (newline))
- (sort profiles string<?)))))
- (('search _)
- (let* ((patterns (filter-map (match-lambda
- (('query 'search rx) rx)
- (_ #f))
- opts))
- (regexps (map (cut make-regexp* <> regexp/icase) patterns))
- (matches (find-packages-by-description regexps)))
- (leave-on-EPIPE
- (display-search-results matches (current-output-port)))
- #t))
- (('show _)
- (let ((requested-names
- (filter-map (match-lambda
- (('query 'show requested-name) requested-name)
- (_ #f))
- opts)))
- (for-each
- (lambda (requested-name)
- (let-values (((name version)
- (package-name->name+version requested-name)))
- (match (remove package-superseded
- (find-packages-by-name name version))
- (()
- (leave (G_ "~a~@[@~a~]: package not found~%") name version))
- (packages
- (leave-on-EPIPE
- (for-each (cute package->recutils <> (current-output-port))
- packages))))))
- requested-names))
- #t)
- (('search-paths kind)
- (let* ((manifests (map profile-manifest profiles))
- (entries (append-map manifest-transitive-entries
- manifests))
- (profiles (map user-friendly-profile profiles))
- (settings (search-path-environment-variables entries profiles
- (const #f)
- #:kind kind)))
- (format #t "~{~a~%~}" settings)
- #t))
- (('export-manifest)
- (let* ((manifest (concatenate-manifests
- (map profile-manifest profiles))))
- (export-manifest manifest (current-output-port))
- #t))
- (('export-channels)
- (let ((manifest (concatenate-manifests
- (map profile-manifest profiles))))
- (export-channels manifest (current-output-port))
- #t))
- (_ #f))))
- (define* (roll-back-action store profile arg opts
- #:key dry-run?)
- "Roll back PROFILE to its previous generation."
- (unless dry-run?
- (roll-back* store profile)))
- (define* (switch-generation-action store profile spec opts
- #:key dry-run?)
- "Switch PROFILE to the generation specified by SPEC."
- (unless dry-run?
- (let ((number (relative-generation-spec->number profile spec)))
- (if number
- (switch-to-generation* profile number)
- (leave (G_ "cannot switch to generation '~a'~%") spec)))))
- (define* (delete-generations-action store profile pattern opts
- #:key dry-run?)
- "Delete PROFILE's generations that match PATTERN."
- (unless dry-run?
- (delete-matching-generations store profile pattern)))
- (define (load-manifest file)
- "Load the user-profile manifest (Scheme code) from FILE and return it."
- (let ((user-module (make-user-module '((guix profiles) (gnu)))))
- (load* file user-module)))
- (define %actions
- ;; List of actions that may be processed. The car of each pair is the
- ;; action's symbol in the option list; the cdr is the action's procedure.
- `((roll-back? . ,roll-back-action)
- (switch-generation . ,switch-generation-action)
- (delete-generations . ,delete-generations-action)))
- (define (process-actions store opts)
- "Process any install/remove/upgrade action from OPTS."
- (define dry-run? (assoc-ref opts 'dry-run?))
- (define bootstrap? (assoc-ref opts 'bootstrap?))
- (define substitutes? (assoc-ref opts 'substitutes?))
- (define allow-collisions? (assoc-ref opts 'allow-collisions?))
- (define profile (or (assoc-ref opts 'profile) %current-profile))
- (define transform (options->transformation opts))
- (define (transform-entry entry)
- (let ((item (transform (manifest-entry-item entry))))
- (manifest-entry-with-transformations
- (manifest-entry
- (inherit entry)
- (item item)
- (version (if (package? item)
- (package-version item)
- (manifest-entry-version entry)))))))
- (when (equal? profile %current-profile)
- ;; Normally the daemon created %CURRENT-PROFILE when we connected, unless
- ;; it's a version that lacks the fix for <https://bugs.gnu.org/37744>
- ;; (aka. CVE-2019-18192). Ensure %CURRENT-PROFILE exists so that
- ;; 'with-profile-lock' can create its lock file below.
- (ensure-default-profile))
- ;; First, acquire a lock on the profile, to ensure only one guix process
- ;; is modifying it at a time.
- (with-profile-lock profile
- ;; Then, process roll-backs, generation removals, etc.
- (for-each (match-lambda
- ((key . arg)
- (and=> (assoc-ref %actions key)
- (lambda (proc)
- (proc store profile arg opts
- #:dry-run? dry-run?)))))
- opts)
- ;; Then, process normal package removal/installation/upgrade.
- (let* ((files (filter-map (match-lambda
- (('manifest . file) file)
- (_ #f))
- opts))
- (manifest (match files
- (() (profile-manifest profile))
- (_ (map-manifest-entries
- manifest-entry-with-provenance
- (concatenate-manifests
- (map load-manifest files))))))
- (step1 (options->removable opts manifest
- (manifest-transaction)))
- (step2 (options->installable opts manifest step1))
- (step3 (manifest-transaction
- (inherit step2)
- (install (map transform-entry
- (manifest-transaction-install step2)))))
- (new (manifest-perform-transaction manifest step3))
- (trans (if (null? files)
- step3
- (fold manifest-transaction-install-entry
- step3
- (manifest-entries manifest)))))
- (warn-about-old-distro)
- (when (and (null? files) (manifest-transaction-null? trans)
- (not (any (match-lambda
- ((key . _) (assoc-ref %actions key)))
- opts)))
- ;; We can reach this point because the user did not specify any action
- ;; (as in "guix package"), did not specify any package (as in "guix
- ;; install"), or because there's nothing to upgrade (as when running
- ;; "guix upgrade" on an up-to-date profile). We cannot distinguish
- ;; among these here; all we can say is that there's nothing to do.
- (warning (G_ "nothing to do~%")))
- (unless (manifest-transaction-null? trans)
- ;; When '--manifest' is used, display information about TRANS as if we
- ;; were starting from an empty profile.
- (show-manifest-transaction store
- (if (null? files)
- manifest
- (make-manifest '()))
- trans
- #:dry-run? dry-run?)
- (build-and-use-profile store profile new
- #:dry-run? dry-run?
- #:allow-collisions? allow-collisions?
- #:bootstrap? bootstrap?)))))
- ;;;
- ;;; Entry point.
- ;;;
- (define-command (guix-package . args)
- (synopsis "manage packages and profiles")
- (define (handle-argument arg result arg-handler)
- ;; Process non-option argument ARG by calling back ARG-HANDLER.
- (if arg-handler
- (arg-handler arg result)
- (leave (G_ "~A: extraneous argument~%") arg)))
- (define opts
- (parse-command-line args %options (list %default-options #f)
- #:argument-handler handle-argument))
- (guix-package* opts))
- (define (guix-package* opts)
- "Run the 'guix package' command on OPTS, an alist resulting for command-line
- option processing with 'parse-command-line'."
- (with-error-handling
- (or (process-query opts)
- (parameterize ((%store (open-connection))
- (%graft? (assoc-ref opts 'graft?)))
- (with-status-verbosity (assoc-ref opts 'verbosity)
- (set-build-options-from-command-line (%store) opts)
- (with-build-handler (build-notifier #:use-substitutes?
- (assoc-ref opts 'substitutes?)
- #:verbosity
- (assoc-ref opts 'verbosity)
- #:dry-run?
- (assoc-ref opts 'dry-run?))
- (parameterize ((%guile-for-build
- (package-derivation
- (%store)
- (if (assoc-ref opts 'bootstrap?)
- %bootstrap-guile
- (default-guile)))))
- (process-actions (%store) opts))))))))
|