2013-01-05 18:47:50 -05:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
2014-01-04 16:42:42 -05:00
|
|
|
|
;;; Copyright © 2012, 2013, 2014 Ludovic Courtès <ludo@gnu.org>
|
2013-01-17 16:20:42 -05:00
|
|
|
|
;;; Copyright © 2013 Nikita Karetnikov <nikita@karetnikov.org>
|
Build newest versions unless specified, and implement upgrades.
* gnu/packages.scm (find-newest-available-packages):
New exported procedure.
* guix-build.in (newest-available-packages, find-best-packages-by-name):
New procedures.
(find-package): Use find-best-packages-by-name, to guarantee that
if a version number is not specified, only the newest versions will
be considered.
* guix-package.in (%options): Add --upgrade/-u option.
(newest-available-packages, find-best-packages-by-name, upgradeable?):
New procedures.
(find-package): Use find-best-packages-by-name, to guarantee that
if a version number is not specified, only the newest versions will
be considered.
(process-actions): Implement upgrade option.
* doc/guix.texi (Invoking guix-package): In the description of --install,
mention that if no version number is specified, the newest available
version will be selected.
2013-02-12 01:24:21 -05:00
|
|
|
|
;;; Copyright © 2013 Mark H Weaver <mhw@netris.org>
|
2012-10-31 20:46:15 -04:00
|
|
|
|
;;;
|
2013-01-05 18:47:50 -05:00
|
|
|
|
;;; This file is part of GNU Guix.
|
2012-10-31 20:46:15 -04:00
|
|
|
|
;;;
|
2013-01-05 18:47:50 -05:00
|
|
|
|
;;; GNU Guix is free software; you can redistribute it and/or modify it
|
2012-10-31 20:46:15 -04:00
|
|
|
|
;;; 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.
|
|
|
|
|
;;;
|
2013-01-05 18:47:50 -05:00
|
|
|
|
;;; GNU Guix is distributed in the hope that it will be useful, but
|
2012-10-31 20:46:15 -04:00
|
|
|
|
;;; 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
|
2013-01-05 18:47:50 -05:00
|
|
|
|
;;; along with GNU Guix. If not, see <http://www.gnu.org/licenses/>.
|
2012-10-31 20:46:15 -04:00
|
|
|
|
|
Replace individual scripts with master 'guix' script.
* scripts/guix.in: New script.
* Makefile.am (bin_SCRIPTS): Add 'scripts/guix'. Remove 'guix-build',
'guix-download', 'guix-import', 'guix-package', and 'guix-gc'.
(MODULES): Add 'guix/scripts/build.scm', 'guix/scripts/download.scm',
'guix/scripts/import.scm', 'guix/scripts/package.scm', and
'guix/scripts/gc.scm'.
* configure.ac (AC_CONFIG_FILES): Add 'scripts/guix'. Remove 'guix-build',
'guix-download', 'guix-import', 'guix-package', and 'guix-gc'.
* guix-build.in, guix-download.in, guix-gc.in, guix-import.in,
guix-package.in: Remove shell script boilerplate. Move to guix-COMMAND.in
to guix/scripts/COMMAND.scm. Rename module from (guix-COMMAND) to
(guix scripts COMMAND). Change "guix-COMMAND" to "guix COMMAND" in
usage help string.
* pre-inst-env.in: Add "@abs_top_builddir@/scripts" to the front of $PATH.
Export $GUIX_UNINSTALLED.
* tests/guix-build.sh, tests/guix-daemon.sh, tests/guix-download.sh,
tests/guix-gc.sh, tests/guix-package.sh: Use "guix COMMAND" instead of
"guix-COMMAND".
* doc/guix.texi: Replace all occurrences of "guix-COMMAND" with
"guix COMMAND".
* po/POTFILES.in: Update.
2013-02-14 04:15:25 -05:00
|
|
|
|
(define-module (guix scripts package)
|
2012-11-03 16:19:43 -04:00
|
|
|
|
#:use-module (guix ui)
|
2012-10-31 20:46:15 -04:00
|
|
|
|
#:use-module (guix store)
|
|
|
|
|
#:use-module (guix derivations)
|
|
|
|
|
#:use-module (guix packages)
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
#:use-module (guix profiles)
|
2014-07-26 16:08:10 -04:00
|
|
|
|
#:use-module (guix monads)
|
2012-10-31 20:46:15 -04:00
|
|
|
|
#:use-module (guix utils)
|
2012-12-13 16:53:05 -05:00
|
|
|
|
#:use-module (guix config)
|
2014-03-01 12:29:29 -05:00
|
|
|
|
#:use-module (guix scripts build)
|
2013-01-14 17:44:58 -05:00
|
|
|
|
#:use-module ((guix build utils) #:select (directory-exists? mkdir-p))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
#:use-module (ice-9 format)
|
|
|
|
|
#:use-module (ice-9 match)
|
|
|
|
|
#:use-module (ice-9 regex)
|
Build newest versions unless specified, and implement upgrades.
* gnu/packages.scm (find-newest-available-packages):
New exported procedure.
* guix-build.in (newest-available-packages, find-best-packages-by-name):
New procedures.
(find-package): Use find-best-packages-by-name, to guarantee that
if a version number is not specified, only the newest versions will
be considered.
* guix-package.in (%options): Add --upgrade/-u option.
(newest-available-packages, find-best-packages-by-name, upgradeable?):
New procedures.
(find-package): Use find-best-packages-by-name, to guarantee that
if a version number is not specified, only the newest versions will
be considered.
(process-actions): Implement upgrade option.
* doc/guix.texi (Invoking guix-package): In the description of --install,
mention that if no version number is specified, the newest available
version will be selected.
2013-02-12 01:24:21 -05:00
|
|
|
|
#:use-module (ice-9 vlist)
|
2012-10-31 20:46:15 -04:00
|
|
|
|
#:use-module (srfi srfi-1)
|
|
|
|
|
#:use-module (srfi srfi-11)
|
2013-09-19 07:07:39 -04:00
|
|
|
|
#:use-module (srfi srfi-19)
|
2012-10-31 20:46:15 -04:00
|
|
|
|
#:use-module (srfi srfi-26)
|
|
|
|
|
#:use-module (srfi srfi-37)
|
2013-01-17 19:06:47 -05:00
|
|
|
|
#:use-module (gnu packages)
|
2013-01-17 19:06:24 -05:00
|
|
|
|
#:use-module ((gnu packages base) #:select (guile-final))
|
|
|
|
|
#:use-module ((gnu packages bootstrap) #:select (%bootstrap-guile))
|
2013-12-21 19:08:21 -05:00
|
|
|
|
#:export (specification->package+output
|
|
|
|
|
guix-package))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
|
|
|
|
|
(define %store
|
2013-01-15 16:39:03 -05:00
|
|
|
|
(make-parameter #f))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;;
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
;;; Profiles.
|
2012-10-31 20:46:15 -04:00
|
|
|
|
;;;
|
|
|
|
|
|
2013-10-30 12:31:05 -04:00
|
|
|
|
(define %user-profile-directory
|
2012-10-31 20:46:15 -04:00
|
|
|
|
(and=> (getenv "HOME")
|
|
|
|
|
(cut string-append <> "/.guix-profile")))
|
|
|
|
|
|
|
|
|
|
(define %profile-directory
|
2014-01-08 15:12:55 -05:00
|
|
|
|
(string-append %state-directory "/profiles/"
|
2014-07-05 08:36:44 -04:00
|
|
|
|
(or (and=> (or (getenv "USER")
|
|
|
|
|
(getenv "LOGNAME"))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
(cut string-append "per-user/" <>))
|
|
|
|
|
"default")))
|
|
|
|
|
|
|
|
|
|
(define %current-profile
|
2012-12-13 17:02:22 -05:00
|
|
|
|
;; Call it `guix-profile', not `profile', to allow Guix profiles to
|
|
|
|
|
;; coexist with Nix profiles.
|
|
|
|
|
(string-append %profile-directory "/guix-profile"))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
|
2014-07-05 08:56:08 -04:00
|
|
|
|
(define (canonicalize-profile profile)
|
|
|
|
|
"If PROFILE is %USER-PROFILE-DIRECTORY, return %CURRENT-PROFILE. Otherwise
|
|
|
|
|
return PROFILE unchanged. The goal is to treat '-p ~/.guix-profile' as if
|
|
|
|
|
'-p' was omitted." ; see <http://bugs.gnu.org/17939>
|
|
|
|
|
(if (and %user-profile-directory
|
|
|
|
|
(string=? (canonicalize-path (dirname profile))
|
|
|
|
|
(dirname %user-profile-directory))
|
|
|
|
|
(string=? (basename profile) (basename %user-profile-directory)))
|
|
|
|
|
%current-profile
|
|
|
|
|
profile))
|
|
|
|
|
|
2013-09-24 23:34:49 -04:00
|
|
|
|
(define (link-to-empty-profile generation)
|
|
|
|
|
"Link GENERATION, a string, to the empty profile."
|
2014-07-26 16:08:10 -04:00
|
|
|
|
(let* ((drv (run-with-store (%store)
|
|
|
|
|
(profile-derivation (manifest '()))))
|
2013-09-24 23:34:49 -04:00
|
|
|
|
(prof (derivation->output-path drv "out")))
|
|
|
|
|
(when (not (build-derivations (%store) (list drv)))
|
|
|
|
|
(leave (_ "failed to build the empty profile~%")))
|
|
|
|
|
|
|
|
|
|
(switch-symlinks generation prof)))
|
|
|
|
|
|
2013-09-25 22:36:24 -04:00
|
|
|
|
(define (switch-to-previous-generation profile)
|
|
|
|
|
"Atomically switch PROFILE to the previous generation."
|
|
|
|
|
(let* ((number (generation-number profile))
|
|
|
|
|
(previous-number (previous-generation-number profile number))
|
2013-10-30 17:08:35 -04:00
|
|
|
|
(previous-generation (generation-file-name profile previous-number)))
|
2013-09-25 22:36:24 -04:00
|
|
|
|
(format #t (_ "switching from generation ~a to ~a~%")
|
|
|
|
|
number previous-number)
|
|
|
|
|
(switch-symlinks profile previous-generation)))
|
|
|
|
|
|
2013-01-17 16:20:42 -05:00
|
|
|
|
(define (roll-back profile)
|
|
|
|
|
"Roll back to the previous generation of PROFILE."
|
2013-09-12 13:31:53 -04:00
|
|
|
|
(let* ((number (generation-number profile))
|
|
|
|
|
(previous-number (previous-generation-number profile number))
|
2014-08-11 12:58:17 -04:00
|
|
|
|
(previous-generation (generation-file-name profile previous-number)))
|
2013-09-25 22:36:24 -04:00
|
|
|
|
(cond ((not (file-exists? profile)) ; invalid profile
|
|
|
|
|
(leave (_ "profile '~a' does not exist~%")
|
2013-04-11 16:30:06 -04:00
|
|
|
|
profile))
|
2013-09-25 22:36:24 -04:00
|
|
|
|
((zero? number) ; empty profile
|
2013-04-12 12:07:17 -04:00
|
|
|
|
(format (current-error-port)
|
|
|
|
|
(_ "nothing to do: already at the empty profile~%")))
|
2013-09-25 22:36:24 -04:00
|
|
|
|
((or (zero? previous-number) ; going to emptiness
|
2013-09-12 13:31:53 -04:00
|
|
|
|
(not (file-exists? previous-generation)))
|
2013-09-24 23:34:49 -04:00
|
|
|
|
(link-to-empty-profile previous-generation)
|
2013-09-25 22:36:24 -04:00
|
|
|
|
(switch-to-previous-generation profile))
|
|
|
|
|
(else
|
|
|
|
|
(switch-to-previous-generation profile))))) ; anything else
|
2013-01-17 16:20:42 -05:00
|
|
|
|
|
2013-09-26 19:17:01 -04:00
|
|
|
|
(define* (matching-generations str #:optional (profile %current-profile)
|
|
|
|
|
#:key (duration-relation <=))
|
2013-09-19 07:07:39 -04:00
|
|
|
|
"Return the list of available generations matching a pattern in STR. See
|
2013-09-26 19:17:01 -04:00
|
|
|
|
'string->generations' and 'string->duration' for the list of valid patterns.
|
|
|
|
|
When STR is a duration pattern, return all the generations whose ctime has
|
|
|
|
|
DURATION-RELATION with the current time."
|
2013-09-19 07:07:39 -04:00
|
|
|
|
(define (valid-generations lst)
|
|
|
|
|
(define (valid-generation? n)
|
|
|
|
|
(any (cut = n <>) (generation-numbers profile)))
|
|
|
|
|
|
|
|
|
|
(fold-right (lambda (x acc)
|
|
|
|
|
(if (valid-generation? x)
|
|
|
|
|
(cons x acc)
|
|
|
|
|
acc))
|
|
|
|
|
'()
|
|
|
|
|
lst))
|
|
|
|
|
|
|
|
|
|
(define (filter-generations generations)
|
|
|
|
|
(match generations
|
|
|
|
|
(() '())
|
|
|
|
|
(('>= n)
|
|
|
|
|
(drop-while (cut > n <>)
|
|
|
|
|
(generation-numbers profile)))
|
|
|
|
|
(('<= n)
|
|
|
|
|
(valid-generations (iota n 1)))
|
|
|
|
|
((lst ..1)
|
|
|
|
|
(valid-generations lst))
|
|
|
|
|
(_ #f)))
|
|
|
|
|
|
|
|
|
|
(define (filter-by-duration duration)
|
|
|
|
|
(define (time-at-midnight time)
|
|
|
|
|
;; Return TIME at midnight by setting nanoseconds, seconds, minutes, and
|
|
|
|
|
;; hours to zeros.
|
|
|
|
|
(let ((d (time-utc->date time)))
|
|
|
|
|
(date->time-utc
|
|
|
|
|
(make-date 0 0 0 0
|
|
|
|
|
(date-day d) (date-month d)
|
|
|
|
|
(date-year d) (date-zone-offset d)))))
|
|
|
|
|
|
|
|
|
|
(define generation-ctime-alist
|
|
|
|
|
(map (lambda (number)
|
|
|
|
|
(cons number
|
|
|
|
|
(time-second
|
|
|
|
|
(time-at-midnight
|
|
|
|
|
(generation-time profile number)))))
|
|
|
|
|
(generation-numbers profile)))
|
|
|
|
|
|
|
|
|
|
(match duration
|
|
|
|
|
(#f #f)
|
|
|
|
|
(res
|
|
|
|
|
(let ((s (time-second
|
|
|
|
|
(subtract-duration (time-at-midnight (current-time))
|
|
|
|
|
duration))))
|
|
|
|
|
(delete #f (map (lambda (x)
|
2013-09-26 19:17:01 -04:00
|
|
|
|
(and (duration-relation s (cdr x))
|
2013-09-19 07:07:39 -04:00
|
|
|
|
(first x)))
|
|
|
|
|
generation-ctime-alist))))))
|
|
|
|
|
|
|
|
|
|
(cond ((string->generations str)
|
|
|
|
|
=>
|
|
|
|
|
filter-generations)
|
|
|
|
|
((string->duration str)
|
|
|
|
|
=>
|
|
|
|
|
filter-by-duration)
|
|
|
|
|
(else #f)))
|
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Package specifications.
|
|
|
|
|
;;;
|
|
|
|
|
|
2013-01-28 01:29:10 -05:00
|
|
|
|
(define (find-packages-by-description rx)
|
2013-10-15 16:59:50 -04:00
|
|
|
|
"Return the list of packages whose name, synopsis, or description matches
|
|
|
|
|
RX."
|
2013-01-28 01:29:10 -05:00
|
|
|
|
(define (same-location? p1 p2)
|
|
|
|
|
;; Compare locations of two packages.
|
|
|
|
|
(equal? (package-location p1) (package-location p2)))
|
|
|
|
|
|
|
|
|
|
(delete-duplicates
|
|
|
|
|
(sort
|
|
|
|
|
(fold-packages (lambda (package result)
|
|
|
|
|
(define matches?
|
|
|
|
|
(cut regexp-exec rx <>))
|
|
|
|
|
|
2014-06-13 11:30:40 -04:00
|
|
|
|
(if (or (matches? (package-name package))
|
2013-10-15 16:59:50 -04:00
|
|
|
|
(and=> (package-synopsis package)
|
2014-06-13 11:30:40 -04:00
|
|
|
|
(compose matches? P_))
|
2013-01-28 01:29:10 -05:00
|
|
|
|
(and=> (package-description package)
|
2014-06-13 11:30:40 -04:00
|
|
|
|
(compose matches? P_)))
|
2013-01-28 01:29:10 -05:00
|
|
|
|
(cons package result)
|
|
|
|
|
result))
|
|
|
|
|
'())
|
|
|
|
|
(lambda (p1 p2)
|
|
|
|
|
(string<? (package-name p1)
|
|
|
|
|
(package-name p2))))
|
|
|
|
|
same-location?))
|
|
|
|
|
|
2014-01-04 16:42:42 -05:00
|
|
|
|
(define-syntax-rule (leave-on-EPIPE exp ...)
|
|
|
|
|
"Run EXP... in a context when EPIPE errors are caught and lead to 'exit'
|
|
|
|
|
with successful exit code. This is useful when writing to the standard output
|
|
|
|
|
may lead to EPIPE, because the standard output is piped through 'head' or
|
|
|
|
|
similar."
|
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
exp ...)
|
|
|
|
|
(lambda args
|
|
|
|
|
;; We really have to exit this brutally, otherwise Guile eventually
|
|
|
|
|
;; attempts to flush all the ports, leading to an uncaught EPIPE down
|
|
|
|
|
;; the path.
|
|
|
|
|
(if (= EPIPE (system-error-errno args))
|
|
|
|
|
(primitive-_exit 0)
|
|
|
|
|
(apply throw args)))))
|
|
|
|
|
|
2013-10-30 10:26:14 -04:00
|
|
|
|
(define* (specification->package+output spec #:optional (output "out"))
|
2013-12-21 19:08:21 -05:00
|
|
|
|
"Return the package and output specified by SPEC, or #f and #f; SPEC may
|
2013-10-30 10:26:14 -04:00
|
|
|
|
optionally contain a version number and an output name, as in these examples:
|
2013-10-29 17:03:02 -04:00
|
|
|
|
|
2013-10-30 10:26:14 -04:00
|
|
|
|
guile
|
|
|
|
|
guile-2.0.9
|
|
|
|
|
guile:debug
|
|
|
|
|
guile-2.0.9:debug
|
|
|
|
|
|
|
|
|
|
If SPEC does not specify a version number, return the preferred newest
|
|
|
|
|
version; if SPEC does not specify an output, return OUTPUT."
|
2013-10-29 17:03:02 -04:00
|
|
|
|
(define (ensure-output p sub-drv)
|
|
|
|
|
(if (member sub-drv (package-outputs p))
|
2013-10-30 10:26:14 -04:00
|
|
|
|
sub-drv
|
2013-10-29 17:03:02 -04:00
|
|
|
|
(leave (_ "package `~a' lacks output `~a'~%")
|
|
|
|
|
(package-full-name p)
|
|
|
|
|
sub-drv)))
|
|
|
|
|
|
2013-11-01 11:57:48 -04:00
|
|
|
|
(let-values (((name version sub-drv)
|
2013-11-24 09:30:06 -05:00
|
|
|
|
(package-specification->name+version+output spec output)))
|
2013-10-29 17:03:02 -04:00
|
|
|
|
(match (find-best-packages-by-name name version)
|
|
|
|
|
((p)
|
2013-10-30 10:26:14 -04:00
|
|
|
|
(values p (ensure-output p sub-drv)))
|
2013-10-29 17:03:02 -04:00
|
|
|
|
((p p* ...)
|
|
|
|
|
(warning (_ "ambiguous package specification `~a'~%")
|
2013-10-30 10:26:14 -04:00
|
|
|
|
spec)
|
2013-10-29 17:03:02 -04:00
|
|
|
|
(warning (_ "choosing ~a from ~a~%")
|
|
|
|
|
(package-full-name p)
|
|
|
|
|
(location->string (package-location p)))
|
2013-10-30 10:26:14 -04:00
|
|
|
|
(values p (ensure-output p sub-drv)))
|
2013-10-29 17:03:02 -04:00
|
|
|
|
(()
|
2013-10-30 10:26:14 -04:00
|
|
|
|
(leave (_ "~a: package not found~%") spec)))))
|
2013-10-29 17:03:02 -04:00
|
|
|
|
|
|
|
|
|
(define (upgradeable? name current-version current-path)
|
|
|
|
|
"Return #t if there's a version of package NAME newer than CURRENT-VERSION,
|
|
|
|
|
or if the newest available version is equal to CURRENT-VERSION but would have
|
|
|
|
|
an output path different than CURRENT-PATH."
|
2013-12-21 16:36:32 -05:00
|
|
|
|
(match (vhash-assoc name (find-newest-available-packages))
|
2013-10-29 17:03:02 -04:00
|
|
|
|
((_ candidate-version pkg . rest)
|
|
|
|
|
(case (version-compare candidate-version current-version)
|
|
|
|
|
((>) #t)
|
|
|
|
|
((<) #f)
|
|
|
|
|
((=) (let ((candidate-path (derivation->output-path
|
|
|
|
|
(package-derivation (%store) pkg))))
|
|
|
|
|
(not (string=? current-path candidate-path))))))
|
|
|
|
|
(#f #f)))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Search paths.
|
|
|
|
|
;;;
|
|
|
|
|
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(define* (search-path-environment-variables entries profile
|
2013-04-28 17:05:57 -04:00
|
|
|
|
#:optional (getenv getenv))
|
|
|
|
|
"Return environment variable definitions that may be needed for the use of
|
2013-10-30 12:13:27 -04:00
|
|
|
|
ENTRIES, a list of manifest entries, in PROFILE. Use GETENV to determine the
|
|
|
|
|
current settings and report only settings not already effective."
|
2013-04-28 17:05:57 -04:00
|
|
|
|
|
2013-05-09 16:58:23 -04:00
|
|
|
|
;; Prefer ~/.guix-profile to the real profile directory name.
|
2013-10-30 12:31:05 -04:00
|
|
|
|
(let ((profile (if (and %user-profile-directory
|
2013-05-09 16:58:23 -04:00
|
|
|
|
(false-if-exception
|
2013-10-30 12:31:05 -04:00
|
|
|
|
(string=? (readlink %user-profile-directory)
|
2013-05-09 16:58:23 -04:00
|
|
|
|
profile)))
|
2013-10-30 12:31:05 -04:00
|
|
|
|
%user-profile-directory
|
2013-05-09 16:58:23 -04:00
|
|
|
|
profile)))
|
|
|
|
|
|
|
|
|
|
;; The search path info is not stored in the manifest. Thus, we infer the
|
|
|
|
|
;; search paths from same-named packages found in the distro.
|
|
|
|
|
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(define manifest-entry->package
|
2013-05-09 16:58:23 -04:00
|
|
|
|
(match-lambda
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(($ <manifest-entry> name version)
|
2014-04-02 10:01:32 -04:00
|
|
|
|
;; Use 'find-best-packages-by-name' and not 'find-packages-by-name';
|
|
|
|
|
;; the former traverses the module tree only once and then allows for
|
|
|
|
|
;; efficient access via a vhash.
|
|
|
|
|
(match (or (find-best-packages-by-name name version)
|
|
|
|
|
(find-best-packages-by-name name #f))
|
2013-05-09 16:58:23 -04:00
|
|
|
|
((p _ ...) p)
|
|
|
|
|
(_ #f)))))
|
|
|
|
|
|
|
|
|
|
(define search-path-definition
|
|
|
|
|
(match-lambda
|
|
|
|
|
(($ <search-path-specification> variable directories separator)
|
|
|
|
|
(let ((values (or (and=> (getenv variable)
|
|
|
|
|
(cut string-tokenize* <> separator))
|
|
|
|
|
'()))
|
|
|
|
|
(directories (filter file-exists?
|
|
|
|
|
(map (cut string-append profile
|
|
|
|
|
"/" <>)
|
|
|
|
|
directories))))
|
|
|
|
|
(if (every (cut member <> values) directories)
|
|
|
|
|
#f
|
|
|
|
|
(format #f "export ~a=\"~a\""
|
|
|
|
|
variable
|
|
|
|
|
(string-join directories separator)))))))
|
|
|
|
|
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(let* ((packages (filter-map manifest-entry->package entries))
|
2013-05-09 16:58:23 -04:00
|
|
|
|
(search-paths (delete-duplicates
|
|
|
|
|
(append-map package-native-search-paths
|
|
|
|
|
packages))))
|
|
|
|
|
(filter-map search-path-definition search-paths))))
|
2013-04-28 17:05:57 -04:00
|
|
|
|
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(define (display-search-paths entries profile)
|
2013-04-28 17:05:57 -04:00
|
|
|
|
"Display the search path environment variables that may need to be set for
|
2013-10-30 12:13:27 -04:00
|
|
|
|
ENTRIES, a list of manifest entries, in the context of PROFILE."
|
|
|
|
|
(let ((settings (search-path-environment-variables entries profile)))
|
2013-04-28 17:05:57 -04:00
|
|
|
|
(unless (null? settings)
|
|
|
|
|
(format #t (_ "The following environment variable definitions may be needed:~%"))
|
2013-05-09 16:58:23 -04:00
|
|
|
|
(format #t "~{ ~a~%~}" settings))))
|
2013-04-28 17:05:57 -04:00
|
|
|
|
|
2012-10-31 20:46:15 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Command-line options.
|
|
|
|
|
;;;
|
|
|
|
|
|
|
|
|
|
(define %default-options
|
|
|
|
|
;; Alist of default option values.
|
2013-04-12 09:43:55 -04:00
|
|
|
|
`((profile . ,%current-profile)
|
2013-05-20 12:14:55 -04:00
|
|
|
|
(max-silent-time . 3600)
|
2014-03-01 12:29:29 -05:00
|
|
|
|
(verbosity . 0)
|
2013-04-12 09:43:55 -04:00
|
|
|
|
(substitutes? . #t)))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
|
|
|
|
|
(define (show-help)
|
Replace individual scripts with master 'guix' script.
* scripts/guix.in: New script.
* Makefile.am (bin_SCRIPTS): Add 'scripts/guix'. Remove 'guix-build',
'guix-download', 'guix-import', 'guix-package', and 'guix-gc'.
(MODULES): Add 'guix/scripts/build.scm', 'guix/scripts/download.scm',
'guix/scripts/import.scm', 'guix/scripts/package.scm', and
'guix/scripts/gc.scm'.
* configure.ac (AC_CONFIG_FILES): Add 'scripts/guix'. Remove 'guix-build',
'guix-download', 'guix-import', 'guix-package', and 'guix-gc'.
* guix-build.in, guix-download.in, guix-gc.in, guix-import.in,
guix-package.in: Remove shell script boilerplate. Move to guix-COMMAND.in
to guix/scripts/COMMAND.scm. Rename module from (guix-COMMAND) to
(guix scripts COMMAND). Change "guix-COMMAND" to "guix COMMAND" in
usage help string.
* pre-inst-env.in: Add "@abs_top_builddir@/scripts" to the front of $PATH.
Export $GUIX_UNINSTALLED.
* tests/guix-build.sh, tests/guix-daemon.sh, tests/guix-download.sh,
tests/guix-gc.sh, tests/guix-package.sh: Use "guix COMMAND" instead of
"guix-COMMAND".
* doc/guix.texi: Replace all occurrences of "guix-COMMAND" with
"guix COMMAND".
* po/POTFILES.in: Update.
2013-02-14 04:15:25 -05:00
|
|
|
|
(display (_ "Usage: guix package [OPTION]... PACKAGES...
|
2012-10-31 20:46:15 -04:00
|
|
|
|
Install, remove, or upgrade PACKAGES in a single transaction.\n"))
|
|
|
|
|
(display (_ "
|
|
|
|
|
-i, --install=PACKAGE install PACKAGE"))
|
|
|
|
|
(display (_ "
|
2013-03-01 15:12:32 -05:00
|
|
|
|
-e, --install-from-expression=EXP
|
|
|
|
|
install the package EXP evaluates to"))
|
|
|
|
|
(display (_ "
|
2012-10-31 20:46:15 -04:00
|
|
|
|
-r, --remove=PACKAGE remove PACKAGE"))
|
|
|
|
|
(display (_ "
|
2013-04-15 17:23:27 -04:00
|
|
|
|
-u, --upgrade[=REGEXP] upgrade all the installed packages matching REGEXP"))
|
2013-01-17 16:20:42 -05:00
|
|
|
|
(display (_ "
|
|
|
|
|
--roll-back roll back to the previous generation"))
|
2013-04-28 17:05:57 -04:00
|
|
|
|
(display (_ "
|
|
|
|
|
--search-paths display needed environment variable definitions"))
|
2013-09-19 07:07:39 -04:00
|
|
|
|
(display (_ "
|
|
|
|
|
-l, --list-generations[=PATTERN]
|
|
|
|
|
list generations matching PATTERN"))
|
2013-09-25 22:36:24 -04:00
|
|
|
|
(display (_ "
|
|
|
|
|
-d, --delete-generations[=PATTERN]
|
|
|
|
|
delete generations matching PATTERN"))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
(display (_ "
|
|
|
|
|
-p, --profile=PROFILE use PROFILE instead of the user's default profile"))
|
2014-03-01 12:29:29 -05:00
|
|
|
|
(newline)
|
2012-10-31 20:46:15 -04:00
|
|
|
|
(display (_ "
|
2013-01-07 16:54:54 -05:00
|
|
|
|
--bootstrap use the bootstrap Guile to build the profile"))
|
2012-12-12 08:59:16 -05:00
|
|
|
|
(display (_ "
|
|
|
|
|
--verbose produce verbose output"))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
(newline)
|
|
|
|
|
(display (_ "
|
2013-01-28 01:29:10 -05:00
|
|
|
|
-s, --search=REGEXP search in synopsis and description using REGEXP"))
|
|
|
|
|
(display (_ "
|
2012-11-19 16:39:45 -05:00
|
|
|
|
-I, --list-installed[=REGEXP]
|
|
|
|
|
list installed packages matching REGEXP"))
|
2012-11-19 17:02:59 -05:00
|
|
|
|
(display (_ "
|
|
|
|
|
-A, --list-available[=REGEXP]
|
|
|
|
|
list available packages matching REGEXP"))
|
2014-07-16 20:36:09 -04:00
|
|
|
|
(display (_ "
|
|
|
|
|
--show=PACKAGE show details about PACKAGE"))
|
2012-11-19 16:39:45 -05:00
|
|
|
|
(newline)
|
2014-03-01 12:29:29 -05:00
|
|
|
|
(show-build-options-help)
|
|
|
|
|
(newline)
|
2012-11-19 16:39:45 -05:00
|
|
|
|
(display (_ "
|
2012-10-31 20:46:15 -04:00
|
|
|
|
-h, --help display this help and exit"))
|
|
|
|
|
(display (_ "
|
|
|
|
|
-V, --version display version information and exit"))
|
|
|
|
|
(newline)
|
2013-01-05 09:55:47 -05:00
|
|
|
|
(show-bug-report-information))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
|
|
|
|
|
(define %options
|
|
|
|
|
;; Specification of the command-line options.
|
2014-03-01 12:29:29 -05:00
|
|
|
|
(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 '(#\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)
|
|
|
|
|
(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 '("roll-back") #f #f
|
|
|
|
|
(lambda (opt name arg result arg-handler)
|
|
|
|
|
(values (alist-cons 'roll-back? #t result)
|
|
|
|
|
#f)))
|
|
|
|
|
(option '(#\l "list-generations") #f #t
|
|
|
|
|
(lambda (opt name arg result arg-handler)
|
|
|
|
|
(values (cons `(query list-generations ,(or arg ""))
|
|
|
|
|
result)
|
|
|
|
|
#f)))
|
|
|
|
|
(option '(#\d "delete-generations") #f #t
|
|
|
|
|
(lambda (opt name arg result arg-handler)
|
|
|
|
|
(values (alist-cons 'delete-generations (or arg "")
|
|
|
|
|
result)
|
|
|
|
|
#f)))
|
|
|
|
|
(option '("search-paths") #f #f
|
|
|
|
|
(lambda (opt name arg result arg-handler)
|
|
|
|
|
(values (cons `(query search-paths) result)
|
|
|
|
|
#f)))
|
|
|
|
|
(option '(#\p "profile") #t #f
|
|
|
|
|
(lambda (opt name arg result arg-handler)
|
2014-07-05 08:56:08 -04:00
|
|
|
|
(values (alist-cons 'profile (canonicalize-profile arg)
|
2014-03-01 12:29:29 -05:00
|
|
|
|
(alist-delete 'profile result))
|
|
|
|
|
#f)))
|
|
|
|
|
(option '(#\n "dry-run") #f #f
|
|
|
|
|
(lambda (opt name arg result arg-handler)
|
|
|
|
|
(values (alist-cons 'dry-run? #t result)
|
|
|
|
|
#f)))
|
|
|
|
|
(option '("bootstrap") #f #f
|
|
|
|
|
(lambda (opt name arg result arg-handler)
|
|
|
|
|
(values (alist-cons 'bootstrap? #t result)
|
|
|
|
|
#f)))
|
|
|
|
|
(option '("verbose") #f #f
|
|
|
|
|
(lambda (opt name arg result arg-handler)
|
|
|
|
|
(values (alist-cons 'verbose? #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)))
|
2014-07-16 20:36:09 -04:00
|
|
|
|
(option '("show") #t #t
|
|
|
|
|
(lambda (opt name arg result arg-handler)
|
|
|
|
|
(values (cons `(query show ,arg)
|
|
|
|
|
result)
|
|
|
|
|
#f)))
|
2014-03-01 12:29:29 -05:00
|
|
|
|
|
|
|
|
|
%standard-build-options))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(define (options->installable opts manifest)
|
|
|
|
|
"Given MANIFEST, the current manifest, and OPTS, the result of 'args-fold',
|
2013-10-30 17:01:43 -04:00
|
|
|
|
return the new list of manifest entries."
|
2014-07-26 16:54:40 -04:00
|
|
|
|
(define (package->manifest-entry* package output)
|
|
|
|
|
(check-package-freshness package)
|
2013-10-30 10:26:14 -04:00
|
|
|
|
;; When given a package via `-e', install the first of its
|
|
|
|
|
;; outputs (XXX).
|
2014-07-26 16:54:40 -04:00
|
|
|
|
(package->manifest-entry package output))
|
2013-10-30 10:26:14 -04:00
|
|
|
|
|
|
|
|
|
(define upgrade-regexps
|
|
|
|
|
(filter-map (match-lambda
|
|
|
|
|
(('upgrade . regexp)
|
|
|
|
|
(make-regexp (or regexp "")))
|
|
|
|
|
(_ #f))
|
|
|
|
|
opts))
|
|
|
|
|
|
|
|
|
|
(define packages-to-upgrade
|
|
|
|
|
(match upgrade-regexps
|
|
|
|
|
(()
|
|
|
|
|
'())
|
|
|
|
|
((_ ...)
|
2014-08-11 12:58:17 -04:00
|
|
|
|
(filter-map (match-lambda
|
|
|
|
|
(($ <manifest-entry> name version output path _)
|
|
|
|
|
(and (any (cut regexp-exec <> name)
|
|
|
|
|
upgrade-regexps)
|
|
|
|
|
(upgradeable? name version path)
|
|
|
|
|
(let ((output (or output "out")))
|
|
|
|
|
(call-with-values
|
|
|
|
|
(lambda ()
|
|
|
|
|
(specification->package+output name output))
|
|
|
|
|
list))))
|
|
|
|
|
(_ #f))
|
|
|
|
|
(manifest-entries manifest)))))
|
2013-10-30 10:26:14 -04:00
|
|
|
|
|
|
|
|
|
(define to-upgrade
|
|
|
|
|
(map (match-lambda
|
|
|
|
|
((package output)
|
2014-07-26 16:54:40 -04:00
|
|
|
|
(package->manifest-entry* package output)))
|
2013-10-30 10:26:14 -04:00
|
|
|
|
packages-to-upgrade))
|
|
|
|
|
|
|
|
|
|
(define packages-to-install
|
|
|
|
|
(filter-map (match-lambda
|
|
|
|
|
(('install . (? package? p))
|
|
|
|
|
(list p "out"))
|
|
|
|
|
(('install . (? string? spec))
|
|
|
|
|
(and (not (store-path? spec))
|
|
|
|
|
(let-values (((package output)
|
|
|
|
|
(specification->package+output spec)))
|
|
|
|
|
(and package (list package output)))))
|
|
|
|
|
(_ #f))
|
|
|
|
|
opts))
|
|
|
|
|
|
|
|
|
|
(define to-install
|
|
|
|
|
(append (map (match-lambda
|
|
|
|
|
((package output)
|
2014-07-26 16:54:40 -04:00
|
|
|
|
(package->manifest-entry* package output)))
|
2013-10-30 10:26:14 -04:00
|
|
|
|
packages-to-install)
|
|
|
|
|
(filter-map (match-lambda
|
|
|
|
|
(('install . (? package?))
|
|
|
|
|
#f)
|
|
|
|
|
(('install . (? store-path? path))
|
|
|
|
|
(let-values (((name version)
|
|
|
|
|
(package-name->name+version
|
|
|
|
|
(store-path-package-name path))))
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(manifest-entry
|
|
|
|
|
(name name)
|
|
|
|
|
(version version)
|
|
|
|
|
(output #f)
|
2014-07-26 16:08:10 -04:00
|
|
|
|
(item path))))
|
2013-10-30 10:26:14 -04:00
|
|
|
|
(_ #f))
|
|
|
|
|
opts)))
|
|
|
|
|
|
2013-10-30 17:01:43 -04:00
|
|
|
|
(append to-upgrade to-install))
|
2013-10-30 10:26:14 -04:00
|
|
|
|
|
2013-11-01 12:12:15 -04:00
|
|
|
|
(define (options->removable options manifest)
|
2013-11-01 18:11:17 -04:00
|
|
|
|
"Given options, return the list of manifest patterns of packages to be
|
|
|
|
|
removed from MANIFEST."
|
|
|
|
|
(filter-map (match-lambda
|
|
|
|
|
(('remove . spec)
|
|
|
|
|
(call-with-values
|
|
|
|
|
(lambda ()
|
|
|
|
|
(package-specification->name+version+output spec))
|
|
|
|
|
(lambda (name version output)
|
|
|
|
|
(manifest-pattern
|
|
|
|
|
(name name)
|
|
|
|
|
(version version)
|
|
|
|
|
(output output)))))
|
|
|
|
|
(_ #f))
|
|
|
|
|
options))
|
2013-11-01 12:12:15 -04:00
|
|
|
|
|
2014-04-02 09:44:42 -04:00
|
|
|
|
(define (maybe-register-gc-root store profile)
|
|
|
|
|
"Register PROFILE as a GC root, unless it doesn't need it."
|
|
|
|
|
(unless (string=? profile %current-profile)
|
|
|
|
|
(add-indirect-root store (canonicalize-path profile))))
|
|
|
|
|
|
2014-07-26 15:43:43 -04:00
|
|
|
|
(define (readlink* file)
|
|
|
|
|
"Call 'readlink' until the result is not a symlink."
|
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(readlink* (readlink file)))
|
|
|
|
|
(lambda args
|
|
|
|
|
(if (= EINVAL (system-error-errno args))
|
|
|
|
|
file
|
|
|
|
|
(apply throw args)))))
|
|
|
|
|
|
2012-10-31 20:46:15 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Entry point.
|
|
|
|
|
;;;
|
|
|
|
|
|
|
|
|
|
(define (guix-package . args)
|
|
|
|
|
(define (parse-options)
|
|
|
|
|
;; Return the alist of option values.
|
2013-04-27 10:46:39 -04:00
|
|
|
|
(args-fold* args %options
|
2013-12-13 15:37:57 -05:00
|
|
|
|
(lambda (opt name arg result arg-handler)
|
2013-04-27 10:46:39 -04:00
|
|
|
|
(leave (_ "~A: unrecognized option~%") name))
|
2013-12-13 15:37:57 -05:00
|
|
|
|
(lambda (arg result arg-handler)
|
|
|
|
|
(if arg-handler
|
|
|
|
|
(arg-handler arg result)
|
|
|
|
|
(leave (_ "~A: extraneous argument~%") arg)))
|
|
|
|
|
%default-options
|
|
|
|
|
#f))
|
2012-10-31 20:46:15 -04:00
|
|
|
|
|
2013-01-14 17:44:58 -05:00
|
|
|
|
(define (ensure-default-profile)
|
2013-05-16 14:04:13 -04:00
|
|
|
|
;; Ensure the default profile symlink and directory exist and are
|
|
|
|
|
;; writable.
|
|
|
|
|
|
|
|
|
|
(define (rtfm)
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
(_ "Try \"info '(guix) Invoking guix package'\" for \
|
|
|
|
|
more information.~%"))
|
|
|
|
|
(exit 1))
|
2013-01-14 17:44:58 -05:00
|
|
|
|
|
|
|
|
|
;; Create ~/.guix-profile if it doesn't exist yet.
|
2013-10-30 12:31:05 -04:00
|
|
|
|
(when (and %user-profile-directory
|
2013-01-14 17:44:58 -05:00
|
|
|
|
%current-profile
|
|
|
|
|
(not (false-if-exception
|
2013-10-30 12:31:05 -04:00
|
|
|
|
(lstat %user-profile-directory))))
|
|
|
|
|
(symlink %current-profile %user-profile-directory))
|
2013-01-14 17:44:58 -05:00
|
|
|
|
|
2013-05-16 14:04:13 -04:00
|
|
|
|
(let ((s (stat %profile-directory #f)))
|
|
|
|
|
;; Attempt to create /…/profiles/per-user/$USER if needed.
|
|
|
|
|
(unless (and s (eq? 'directory (stat:type s)))
|
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(mkdir-p %profile-directory))
|
|
|
|
|
(lambda args
|
|
|
|
|
;; Often, we cannot create %PROFILE-DIRECTORY because its
|
|
|
|
|
;; parent directory is root-owned and we're running
|
|
|
|
|
;; unprivileged.
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
(_ "error: while creating directory `~a': ~a~%")
|
|
|
|
|
%profile-directory
|
|
|
|
|
(strerror (system-error-errno args)))
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
(_ "Please create the `~a' directory, with you as the owner.~%")
|
|
|
|
|
%profile-directory)
|
|
|
|
|
(rtfm))))
|
|
|
|
|
|
|
|
|
|
;; Bail out if it's not owned by the user.
|
2013-05-24 17:14:19 -04:00
|
|
|
|
(unless (or (not s) (= (stat:uid s) (getuid)))
|
2013-05-16 14:04:13 -04:00
|
|
|
|
(format (current-error-port)
|
|
|
|
|
(_ "error: directory `~a' is not owned by you~%")
|
|
|
|
|
%profile-directory)
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
(_ "Please change the owner of `~a' to user ~s.~%")
|
2014-07-05 08:36:44 -04:00
|
|
|
|
%profile-directory (or (getenv "USER")
|
|
|
|
|
(getenv "LOGNAME")
|
|
|
|
|
(getuid)))
|
2013-05-16 14:04:13 -04:00
|
|
|
|
(rtfm))))
|
2013-01-14 17:44:58 -05:00
|
|
|
|
|
2012-11-19 16:39:45 -05:00
|
|
|
|
(define (process-actions opts)
|
|
|
|
|
;; Process any install/remove/upgrade action from OPTS.
|
2013-01-17 16:20:42 -05:00
|
|
|
|
|
|
|
|
|
(define dry-run? (assoc-ref opts 'dry-run?))
|
|
|
|
|
(define profile (assoc-ref opts 'profile))
|
|
|
|
|
|
2013-09-25 22:36:24 -04:00
|
|
|
|
(define current-generation-number
|
|
|
|
|
(generation-number profile))
|
|
|
|
|
|
|
|
|
|
(define (display-and-delete number)
|
2013-10-30 17:08:35 -04:00
|
|
|
|
(let ((generation (generation-file-name profile number)))
|
2013-09-25 22:36:24 -04:00
|
|
|
|
(unless (zero? number)
|
|
|
|
|
(format #t (_ "deleting ~a~%") generation)
|
|
|
|
|
(delete-file generation))))
|
|
|
|
|
|
|
|
|
|
(define (delete-generation number)
|
|
|
|
|
(let* ((previous-number (previous-generation-number profile number))
|
2013-10-30 17:08:35 -04:00
|
|
|
|
(previous-generation
|
|
|
|
|
(generation-file-name profile previous-number)))
|
2013-09-25 22:36:24 -04:00
|
|
|
|
(cond ((zero? number)) ; do not delete generation 0
|
|
|
|
|
((and (= number current-generation-number)
|
|
|
|
|
(not (file-exists? previous-generation)))
|
|
|
|
|
(link-to-empty-profile previous-generation)
|
|
|
|
|
(switch-to-previous-generation profile)
|
|
|
|
|
(display-and-delete number))
|
|
|
|
|
((= number current-generation-number)
|
|
|
|
|
(roll-back profile)
|
|
|
|
|
(display-and-delete number))
|
|
|
|
|
(else
|
|
|
|
|
(display-and-delete number)))))
|
|
|
|
|
|
2013-01-17 16:20:42 -05:00
|
|
|
|
;; First roll back if asked to.
|
2013-09-25 22:36:24 -04:00
|
|
|
|
(cond ((and (assoc-ref opts 'roll-back?) (not dry-run?))
|
|
|
|
|
(begin
|
|
|
|
|
(roll-back profile)
|
|
|
|
|
(process-actions (alist-delete 'roll-back? opts))))
|
|
|
|
|
((and (assoc-ref opts 'delete-generations)
|
|
|
|
|
(not dry-run?))
|
|
|
|
|
(filter-map
|
|
|
|
|
(match-lambda
|
|
|
|
|
(('delete-generations . pattern)
|
|
|
|
|
(cond ((not (file-exists? profile)) ; XXX: race condition
|
|
|
|
|
(leave (_ "profile '~a' does not exist~%")
|
|
|
|
|
profile))
|
|
|
|
|
((string-null? pattern)
|
|
|
|
|
(let ((numbers (generation-numbers profile)))
|
|
|
|
|
(if (equal? numbers '(0))
|
|
|
|
|
(exit 0)
|
|
|
|
|
(for-each display-and-delete
|
|
|
|
|
(delete current-generation-number
|
|
|
|
|
numbers)))))
|
|
|
|
|
;; Do not delete the zeroth generation.
|
|
|
|
|
((equal? 0 (string->number pattern))
|
|
|
|
|
(exit 0))
|
2013-09-26 19:17:01 -04:00
|
|
|
|
|
|
|
|
|
;; If PATTERN is a duration, match generations that are
|
|
|
|
|
;; older than the specified duration.
|
|
|
|
|
((matching-generations pattern profile
|
|
|
|
|
#:duration-relation >)
|
2013-09-25 22:36:24 -04:00
|
|
|
|
=>
|
|
|
|
|
(lambda (numbers)
|
|
|
|
|
(if (null-list? numbers)
|
|
|
|
|
(exit 1)
|
|
|
|
|
(for-each delete-generation numbers))))
|
|
|
|
|
(else
|
|
|
|
|
(leave (_ "invalid syntax: ~a~%")
|
|
|
|
|
pattern)))
|
|
|
|
|
|
|
|
|
|
(process-actions
|
|
|
|
|
(alist-delete 'delete-generations opts)))
|
|
|
|
|
(_ #f))
|
|
|
|
|
opts))
|
|
|
|
|
(else
|
2014-08-13 16:15:48 -04:00
|
|
|
|
(let* ((manifest (profile-manifest profile))
|
|
|
|
|
(install (options->installable opts manifest))
|
|
|
|
|
(remove (options->removable opts manifest))
|
|
|
|
|
(transaction (manifest-transaction (install install)
|
|
|
|
|
(remove remove)))
|
|
|
|
|
(new (manifest-perform-transaction
|
|
|
|
|
manifest transaction)))
|
2013-10-30 17:01:43 -04:00
|
|
|
|
|
|
|
|
|
(when (equal? profile %current-profile)
|
|
|
|
|
(ensure-default-profile))
|
|
|
|
|
|
2014-07-26 15:43:43 -04:00
|
|
|
|
(unless (and (null? install) (null? remove))
|
2014-07-26 16:08:10 -04:00
|
|
|
|
(let* ((prof-drv (run-with-store (%store)
|
2014-08-13 16:15:48 -04:00
|
|
|
|
(profile-derivation new)))
|
|
|
|
|
(prof (derivation->output-path prof-drv)))
|
|
|
|
|
(manifest-show-transaction (%store) manifest transaction
|
|
|
|
|
#:dry-run? dry-run?)
|
2014-07-26 15:43:43 -04:00
|
|
|
|
(show-what-to-build (%store) (list prof-drv)
|
|
|
|
|
#:use-substitutes?
|
|
|
|
|
(assoc-ref opts 'substitutes?)
|
|
|
|
|
#:dry-run? dry-run?)
|
|
|
|
|
|
|
|
|
|
(cond
|
|
|
|
|
(dry-run? #t)
|
|
|
|
|
((and (file-exists? profile)
|
|
|
|
|
(and=> (readlink* profile) (cut string=? prof <>)))
|
|
|
|
|
(format (current-error-port) (_ "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))
|
2014-08-12 10:10:32 -04:00
|
|
|
|
(let* ((entries (manifest-entries new))
|
|
|
|
|
(count (length entries)))
|
2014-07-26 15:43:43 -04:00
|
|
|
|
(switch-symlinks name prof)
|
|
|
|
|
(switch-symlinks profile name)
|
|
|
|
|
(maybe-register-gc-root (%store) profile)
|
|
|
|
|
(format #t (N_ "~a package in profile~%"
|
|
|
|
|
"~a packages in profile~%"
|
|
|
|
|
count)
|
|
|
|
|
count)
|
|
|
|
|
(display-search-paths entries
|
|
|
|
|
profile))))))))))))
|
2012-11-19 16:39:45 -05:00
|
|
|
|
|
|
|
|
|
(define (process-query opts)
|
|
|
|
|
;; Process any query specified by OPTS. Return #t when a query was
|
|
|
|
|
;; actually processed, #f otherwise.
|
|
|
|
|
(let ((profile (assoc-ref opts 'profile)))
|
|
|
|
|
(match (assoc-ref opts 'query)
|
2013-09-19 07:07:39 -04:00
|
|
|
|
(('list-generations pattern)
|
|
|
|
|
(define (list-generation number)
|
2013-09-24 18:18:09 -04:00
|
|
|
|
(unless (zero? number)
|
2013-09-24 21:55:27 -04:00
|
|
|
|
(let ((header (format #f (_ "Generation ~a\t~a") number
|
|
|
|
|
(date->string
|
|
|
|
|
(time-utc->date
|
|
|
|
|
(generation-time profile number))
|
|
|
|
|
"~b ~d ~Y ~T")))
|
|
|
|
|
(current (generation-number profile)))
|
|
|
|
|
(if (= number current)
|
|
|
|
|
(format #t (_ "~a\t(current)~%") header)
|
|
|
|
|
(format #t "~a~%" header)))
|
2013-09-19 07:07:39 -04:00
|
|
|
|
(for-each (match-lambda
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(($ <manifest-entry> name version output location _)
|
2013-09-19 07:07:39 -04:00
|
|
|
|
(format #t " ~a\t~a\t~a\t~a~%"
|
|
|
|
|
name version output location)))
|
2013-09-22 15:50:11 -04:00
|
|
|
|
|
|
|
|
|
;; Show most recently installed packages last.
|
|
|
|
|
(reverse
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(manifest-entries
|
2013-09-22 15:50:11 -04:00
|
|
|
|
(profile-manifest
|
2013-10-30 17:08:35 -04:00
|
|
|
|
(generation-file-name profile number)))))
|
2013-09-19 07:07:39 -04:00
|
|
|
|
(newline)))
|
|
|
|
|
|
|
|
|
|
(cond ((not (file-exists? profile)) ; XXX: race condition
|
|
|
|
|
(leave (_ "profile '~a' does not exist~%")
|
|
|
|
|
profile))
|
|
|
|
|
((string-null? pattern)
|
2013-09-24 18:11:16 -04:00
|
|
|
|
(let ((numbers (generation-numbers profile)))
|
2014-01-04 16:42:42 -05:00
|
|
|
|
(leave-on-EPIPE
|
|
|
|
|
(if (equal? numbers '(0))
|
|
|
|
|
(exit 0)
|
|
|
|
|
(for-each list-generation numbers)))))
|
2013-09-19 07:07:39 -04:00
|
|
|
|
((matching-generations pattern profile)
|
|
|
|
|
=>
|
2013-09-24 18:11:16 -04:00
|
|
|
|
(lambda (numbers)
|
|
|
|
|
(if (null-list? numbers)
|
|
|
|
|
(exit 1)
|
2014-01-04 16:42:42 -05:00
|
|
|
|
(leave-on-EPIPE
|
|
|
|
|
(for-each list-generation numbers)))))
|
2013-09-19 07:07:39 -04:00
|
|
|
|
(else
|
|
|
|
|
(leave (_ "invalid syntax: ~a~%")
|
|
|
|
|
pattern)))
|
|
|
|
|
#t)
|
|
|
|
|
|
2012-11-19 16:39:45 -05:00
|
|
|
|
(('list-installed regexp)
|
|
|
|
|
(let* ((regexp (and regexp (make-regexp regexp)))
|
|
|
|
|
(manifest (profile-manifest profile))
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(installed (manifest-entries manifest)))
|
2014-01-04 16:42:42 -05:00
|
|
|
|
(leave-on-EPIPE
|
|
|
|
|
(for-each (match-lambda
|
|
|
|
|
(($ <manifest-entry> name version output path _)
|
|
|
|
|
(when (or (not regexp)
|
|
|
|
|
(regexp-exec regexp name))
|
|
|
|
|
(format #t "~a\t~a\t~a\t~a~%"
|
|
|
|
|
name (or version "?") output path))))
|
|
|
|
|
|
|
|
|
|
;; Show most recently installed packages last.
|
|
|
|
|
(reverse installed)))
|
2012-11-19 17:02:59 -05:00
|
|
|
|
#t))
|
2013-01-28 01:29:10 -05:00
|
|
|
|
|
2012-11-19 17:02:59 -05:00
|
|
|
|
(('list-available regexp)
|
|
|
|
|
(let* ((regexp (and regexp (make-regexp regexp)))
|
|
|
|
|
(available (fold-packages
|
|
|
|
|
(lambda (p r)
|
|
|
|
|
(let ((n (package-name p)))
|
|
|
|
|
(if regexp
|
|
|
|
|
(if (regexp-exec regexp n)
|
|
|
|
|
(cons p r)
|
|
|
|
|
r)
|
|
|
|
|
(cons p r))))
|
|
|
|
|
'())))
|
2014-01-04 16:42:42 -05:00
|
|
|
|
(leave-on-EPIPE
|
|
|
|
|
(for-each (lambda (p)
|
|
|
|
|
(format #t "~a\t~a\t~a\t~a~%"
|
|
|
|
|
(package-name p)
|
|
|
|
|
(package-version p)
|
|
|
|
|
(string-join (package-outputs p) ",")
|
|
|
|
|
(location->string (package-location p))))
|
|
|
|
|
(sort available
|
|
|
|
|
(lambda (p1 p2)
|
|
|
|
|
(string<? (package-name p1)
|
|
|
|
|
(package-name p2))))))
|
2012-11-19 17:02:59 -05:00
|
|
|
|
#t))
|
2013-01-28 01:29:10 -05:00
|
|
|
|
|
|
|
|
|
(('search regexp)
|
2013-02-01 06:52:35 -05:00
|
|
|
|
(let ((regexp (make-regexp regexp regexp/icase)))
|
2014-01-13 12:51:07 -05:00
|
|
|
|
(leave-on-EPIPE
|
|
|
|
|
(for-each (cute package->recutils <> (current-output-port))
|
|
|
|
|
(find-packages-by-description regexp)))
|
2013-01-28 01:29:10 -05:00
|
|
|
|
#t))
|
2013-04-28 17:05:57 -04:00
|
|
|
|
|
2014-07-16 20:36:09 -04:00
|
|
|
|
(('show requested-name)
|
|
|
|
|
(let-values (((name version)
|
|
|
|
|
(package-name->name+version requested-name)))
|
|
|
|
|
(leave-on-EPIPE
|
|
|
|
|
(for-each (cute package->recutils <> (current-output-port))
|
|
|
|
|
(find-packages-by-name name version)))
|
|
|
|
|
#t))
|
|
|
|
|
|
2013-04-28 17:05:57 -04:00
|
|
|
|
(('search-paths)
|
|
|
|
|
(let* ((manifest (profile-manifest profile))
|
2013-10-30 12:13:27 -04:00
|
|
|
|
(entries (manifest-entries manifest))
|
|
|
|
|
(settings (search-path-environment-variables entries profile
|
2013-04-28 17:05:57 -04:00
|
|
|
|
(const #f))))
|
|
|
|
|
(format #t "~{~a~%~}" settings)
|
|
|
|
|
#t))
|
|
|
|
|
|
2012-11-19 16:39:45 -05:00
|
|
|
|
(_ #f))))
|
|
|
|
|
|
2012-10-31 20:46:15 -04:00
|
|
|
|
(let ((opts (parse-options)))
|
2013-01-31 17:24:10 -05:00
|
|
|
|
(or (process-query opts)
|
2013-03-07 13:29:12 -05:00
|
|
|
|
(with-error-handling
|
|
|
|
|
(parameterize ((%store (open-connection)))
|
2014-03-01 12:29:29 -05:00
|
|
|
|
(set-build-options-from-command-line (%store) opts)
|
2013-04-12 09:43:55 -04:00
|
|
|
|
|
2013-01-15 16:39:03 -05:00
|
|
|
|
(parameterize ((%guile-for-build
|
|
|
|
|
(package-derivation (%store)
|
|
|
|
|
(if (assoc-ref opts 'bootstrap?)
|
|
|
|
|
%bootstrap-guile
|
|
|
|
|
guile-final))))
|
|
|
|
|
(process-actions opts)))))))
|