guix package: allow multiple arguments after -i, -r, and -u.

* guix/scripts/package.scm (%options): Adapt option processors to accept and
  return a second seed value: 'arg-handler', which handles bare arguments (if
  not false).  The install, remove, and upgrade option processors return an
  arg-handler that repeat the same operation.  All other option processors
  return #f as the arg-handler.  Make the arguments to install and remove
  optional.  The upgrade option processor deletes (upgrade . #f) from the
  alist before adding a new entry.
  (guix-package): Procedures passed to 'args-fold*' accept the new seed value
  'arg-handler'.  The 'operand-proc' uses 'arg-handler' (if not false).

* doc/guix.texi (Invoking guix package): Update docs.

* tests/guix-package.sh: Add test.
This commit is contained in:
Mark H Weaver 2013-12-13 15:37:57 -05:00
parent 5839958a8f
commit 6447738c01
3 changed files with 107 additions and 64 deletions

View file

@ -501,6 +501,13 @@ the transaction. Upon completion, a new profile is created, but
previous generations of the profile remain available, should the user previous generations of the profile remain available, should the user
want to roll back. want to roll back.
For example, to remove @code{lua} and install @code{guile} and
@code{guile-cairo} in a single transaction:
@example
guix package -r lua -i guile guile-cairo
@end example
For each user, a symlink to the user's default profile is automatically For each user, a symlink to the user's default profile is automatically
created in @file{$HOME/.guix-profile}. This symlink always points to the created in @file{$HOME/.guix-profile}. This symlink always points to the
current generation of the user's default profile. Thus, users can add current generation of the user's default profile. Thus, users can add
@ -522,11 +529,11 @@ The @var{options} can be among the following:
@table @code @table @code
@item --install=@var{package} @item --install=@var{package} @dots{}
@itemx -i @var{package} @itemx -i @var{package} @dots{}
Install @var{package}. Install the specified @var{package}s.
@var{package} may specify either a simple package name, such as Each @var{package} may specify either a simple package name, such as
@code{guile}, or a package name followed by a hyphen and version number, @code{guile}, or a package name followed by a hyphen and version number,
such as @code{guile-1.8.8}. If no version number is specified, the such as @code{guile-1.8.8}. If no version number is specified, the
newest available version will be selected. In addition, @var{package} newest available version will be selected. In addition, @var{package}
@ -568,19 +575,20 @@ Note that this option installs the first output of the specified
package, which may be insufficient when needing a specific output of a package, which may be insufficient when needing a specific output of a
multiple-output package. multiple-output package.
@item --remove=@var{package} @item --remove=@var{package} @dots{}
@itemx -r @var{package} @itemx -r @var{package} @dots{}
Remove @var{package}. Remove the specified @var{package}s.
As for @code{--install}, @var{package} may specify a version number As for @code{--install}, each @var{package} may specify a version number
and/or output name in addition to the package name. For instance, and/or output name in addition to the package name. For instance,
@code{-r glibc:debug} would remove the @code{debug} output of @code{-r glibc:debug} would remove the @code{debug} output of
@code{glibc}. @code{glibc}.
@item --upgrade[=@var{regexp}] @item --upgrade[=@var{regexp} @dots{}]
@itemx -u [@var{regexp}] @itemx -u [@var{regexp} @dots{}]
Upgrade all the installed packages. When @var{regexp} is specified, upgrade Upgrade all the installed packages. If one or more @var{regexp}s are
only installed packages whose name matches @var{regexp}. specified, upgrade only installed packages whose name matches a
@var{regexp}.
Note that this upgrades package to the latest version of packages found Note that this upgrades package to the latest version of packages found
in the distribution currently installed. To update your distribution, in the distribution currently installed. To update your distribution,

View file

@ -523,70 +523,99 @@ (define %options
(lambda args (lambda args
(show-version-and-exit "guix package"))) (show-version-and-exit "guix package")))
(option '(#\i "install") #t #f (option '(#\i "install") #f #t
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'install arg result))) (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 (option '(#\e "install-from-expression") #t #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'install (read/eval-package-expression arg) (values (alist-cons 'install (read/eval-package-expression arg)
result))) result)
(option '(#\r "remove") #t #f #f)))
(lambda (opt name arg result) (option '(#\r "remove") #f #t
(alist-cons 'remove arg result))) (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 (option '(#\u "upgrade") #f #t
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'upgrade arg result))) (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 (option '("roll-back") #f #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'roll-back? #t result))) (values (alist-cons 'roll-back? #t result)
#f)))
(option '(#\l "list-generations") #f #t (option '(#\l "list-generations") #f #t
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(cons `(query list-generations ,(or arg "")) (values (cons `(query list-generations ,(or arg ""))
result))) result)
#f)))
(option '(#\d "delete-generations") #f #t (option '(#\d "delete-generations") #f #t
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'delete-generations (or arg "") (values (alist-cons 'delete-generations (or arg "")
result))) result)
#f)))
(option '("search-paths") #f #f (option '("search-paths") #f #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(cons `(query search-paths) result))) (values (cons `(query search-paths) result)
#f)))
(option '(#\p "profile") #t #f (option '(#\p "profile") #t #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'profile arg (values (alist-cons 'profile arg
(alist-delete 'profile result)))) (alist-delete 'profile result))
#f)))
(option '(#\n "dry-run") #f #f (option '(#\n "dry-run") #f #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'dry-run? #t result))) (values (alist-cons 'dry-run? #t result)
#f)))
(option '("fallback") #f #f (option '("fallback") #f #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'fallback? #t (values (alist-cons 'fallback? #t
(alist-delete 'fallback? result)))) (alist-delete 'fallback? result))
#f)))
(option '("no-substitutes") #f #f (option '("no-substitutes") #f #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'substitutes? #f (values (alist-cons 'substitutes? #f
(alist-delete 'substitutes? result)))) (alist-delete 'substitutes? result))
#f)))
(option '("max-silent-time") #t #f (option '("max-silent-time") #t #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'max-silent-time (string->number* arg) (values (alist-cons 'max-silent-time (string->number* arg)
result))) result)
#f)))
(option '("bootstrap") #f #f (option '("bootstrap") #f #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'bootstrap? #t result))) (values (alist-cons 'bootstrap? #t result)
#f)))
(option '("verbose") #f #f (option '("verbose") #f #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(alist-cons 'verbose? #t result))) (values (alist-cons 'verbose? #t result)
#f)))
(option '(#\s "search") #t #f (option '(#\s "search") #t #f
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(cons `(query search ,(or arg "")) (values (cons `(query search ,(or arg ""))
result))) result)
#f)))
(option '(#\I "list-installed") #f #t (option '(#\I "list-installed") #f #t
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(cons `(query list-installed ,(or arg "")) (values (cons `(query list-installed ,(or arg ""))
result))) result)
#f)))
(option '(#\A "list-available") #f #t (option '(#\A "list-available") #f #t
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(cons `(query list-available ,(or arg "")) (values (cons `(query list-available ,(or arg ""))
result))))) result)
#f)))))
(define (options->installable opts manifest) (define (options->installable opts manifest)
"Given MANIFEST, the current manifest, and OPTS, the result of 'args-fold', "Given MANIFEST, the current manifest, and OPTS, the result of 'args-fold',
@ -717,11 +746,14 @@ (define (guix-package . args)
(define (parse-options) (define (parse-options)
;; Return the alist of option values. ;; Return the alist of option values.
(args-fold* args %options (args-fold* args %options
(lambda (opt name arg result) (lambda (opt name arg result arg-handler)
(leave (_ "~A: unrecognized option~%") name)) (leave (_ "~A: unrecognized option~%") name))
(lambda (arg result) (lambda (arg result arg-handler)
(leave (_ "~A: extraneous argument~%") arg)) (if arg-handler
%default-options)) (arg-handler arg result)
(leave (_ "~A: extraneous argument~%") arg)))
%default-options
#f))
(define (guile-missing?) (define (guile-missing?)
;; Return #t if %GUILE-FOR-BUILD is not available yet. ;; Return #t if %GUILE-FOR-BUILD is not available yet.

View file

@ -155,6 +155,9 @@ then
guix package -p "$profile" --delete-generations=0 guix package -p "$profile" --delete-generations=0
fi fi
# Make sure multiple arguments to -i works.
guix package --bootstrap -i guile gcc -p "$profile" -n
# Make sure the `:' syntax works. # Make sure the `:' syntax works.
guix package --bootstrap -i "binutils:lib" -p "$profile" -n guix package --bootstrap -i "binutils:lib" -p "$profile" -n