mirror of
https://git.in.rschanz.org/ryan77627/guix.git
synced 2025-01-12 06:06:53 -05:00
ui: Factorize command-line + env. var. option parsing.
* guix/ui.scm (%default-argument-handler, parse-command-line): New procedures. (environment-build-options): Make private. * guix/scripts/archive.scm (guix-archive)[parse-options, parse-options-from]: Remove. Use 'parse-command-line' instead. * guix/scripts/build.scm (guix-build): Likewise. * guix/scripts/environment.scm (guix-environment): Likewise. * guix/scripts/package.scm (guix-package): Likewise. * guix/scripts/system.scm (guix-system): Likewise. * tests/ui.scm (with-environment-variable): New macro. ("parse-command-line"): New test.
This commit is contained in:
parent
72bfebf58d
commit
b3f213893b
7 changed files with 85 additions and 83 deletions
|
@ -297,20 +297,6 @@ (define (read-key)
|
||||||
(cut write-acl acl <>)))))
|
(cut write-acl acl <>)))))
|
||||||
|
|
||||||
(define (guix-archive . args)
|
(define (guix-archive . args)
|
||||||
(define (parse-options)
|
|
||||||
;; Return the alist of option values.
|
|
||||||
(append (parse-options-from args)
|
|
||||||
(parse-options-from (environment-build-options))))
|
|
||||||
|
|
||||||
(define (parse-options-from args)
|
|
||||||
;; Actual parsing takes place here.
|
|
||||||
(args-fold* args %options
|
|
||||||
(lambda (opt name arg result)
|
|
||||||
(leave (_ "~A: unrecognized option~%") name))
|
|
||||||
(lambda (arg result)
|
|
||||||
(alist-cons 'argument arg result))
|
|
||||||
%default-options))
|
|
||||||
|
|
||||||
(define (lines port)
|
(define (lines port)
|
||||||
;; Return lines read from PORT.
|
;; Return lines read from PORT.
|
||||||
(let loop ((line (read-line port))
|
(let loop ((line (read-line port))
|
||||||
|
@ -324,7 +310,7 @@ (define (lines port)
|
||||||
;; Ask for absolute file names so that .drv file names passed from the
|
;; Ask for absolute file names so that .drv file names passed from the
|
||||||
;; user to 'read-derivation' are absolute when it returns.
|
;; user to 'read-derivation' are absolute when it returns.
|
||||||
(with-fluids ((%file-port-name-canonicalization 'absolute))
|
(with-fluids ((%file-port-name-canonicalization 'absolute))
|
||||||
(let ((opts (parse-options)))
|
(let ((opts (parse-command-line args %options (list %default-options))))
|
||||||
(cond ((assoc-ref opts 'generate-key)
|
(cond ((assoc-ref opts 'generate-key)
|
||||||
=>
|
=>
|
||||||
generate-key-pair)
|
generate-key-pair)
|
||||||
|
|
|
@ -405,25 +405,12 @@ (define new-sources
|
||||||
;;;
|
;;;
|
||||||
|
|
||||||
(define (guix-build . args)
|
(define (guix-build . args)
|
||||||
(define (parse-options)
|
|
||||||
;; Return the alist of option values.
|
|
||||||
(append (parse-options-from args)
|
|
||||||
(parse-options-from (environment-build-options))))
|
|
||||||
|
|
||||||
(define (parse-options-from args)
|
|
||||||
;; Actual parsing takes place here.
|
|
||||||
(args-fold* args %options
|
|
||||||
(lambda (opt name arg result)
|
|
||||||
(leave (_ "~A: unrecognized option~%") name))
|
|
||||||
(lambda (arg result)
|
|
||||||
(alist-cons 'argument arg result))
|
|
||||||
%default-options))
|
|
||||||
|
|
||||||
(with-error-handling
|
(with-error-handling
|
||||||
;; Ask for absolute file names so that .drv file names passed from the
|
;; Ask for absolute file names so that .drv file names passed from the
|
||||||
;; user to 'read-derivation' are absolute when it returns.
|
;; user to 'read-derivation' are absolute when it returns.
|
||||||
(with-fluids ((%file-port-name-canonicalization 'absolute))
|
(with-fluids ((%file-port-name-canonicalization 'absolute))
|
||||||
(let* ((opts (parse-options))
|
(let* ((opts (parse-command-line args %options
|
||||||
|
(list %default-options)))
|
||||||
(store (open-connection))
|
(store (open-connection))
|
||||||
(drv (options->derivations store opts))
|
(drv (options->derivations store opts))
|
||||||
(roots (filter-map (match-lambda
|
(roots (filter-map (match-lambda
|
||||||
|
|
|
@ -217,22 +217,12 @@ (define (build-inputs inputs opts)
|
||||||
|
|
||||||
;; Entry point.
|
;; Entry point.
|
||||||
(define (guix-environment . args)
|
(define (guix-environment . args)
|
||||||
(define (parse-options)
|
(define (handle-argument arg result)
|
||||||
;; Return the alist of option values.
|
(alist-cons 'package arg result))
|
||||||
(append (parse-options-from args)
|
|
||||||
(parse-options-from (environment-build-options))))
|
|
||||||
|
|
||||||
(define (parse-options-from args)
|
|
||||||
;; Actual parsing takes place here.
|
|
||||||
(args-fold* args %options
|
|
||||||
(lambda (opt name arg result)
|
|
||||||
(leave (_ "~A: unrecognized option~%") name))
|
|
||||||
(lambda (arg result)
|
|
||||||
(alist-cons 'package arg result))
|
|
||||||
%default-options))
|
|
||||||
|
|
||||||
(with-store store
|
(with-store store
|
||||||
(let* ((opts (parse-options))
|
(let* ((opts (parse-command-line args %options (list %default-options)
|
||||||
|
#:argument-handler handle-argument))
|
||||||
(pure? (assoc-ref opts 'pure))
|
(pure? (assoc-ref opts 'pure))
|
||||||
(command (assoc-ref opts 'exec))
|
(command (assoc-ref opts 'exec))
|
||||||
(inputs (packages->transitive-inputs
|
(inputs (packages->transitive-inputs
|
||||||
|
|
|
@ -692,22 +692,11 @@ (define (readlink* file)
|
||||||
;;;
|
;;;
|
||||||
|
|
||||||
(define (guix-package . args)
|
(define (guix-package . args)
|
||||||
(define (parse-options)
|
(define (handle-argument arg result arg-handler)
|
||||||
;; Return the alist of option values.
|
;; Process non-option argument ARG by calling back ARG-HANDLER.
|
||||||
(append (parse-options-from args)
|
(if arg-handler
|
||||||
(parse-options-from (environment-build-options))))
|
(arg-handler arg result)
|
||||||
|
(leave (_ "~A: extraneous argument~%") arg)))
|
||||||
(define (parse-options-from args)
|
|
||||||
;; Actual parsing takes place here.
|
|
||||||
(args-fold* args %options
|
|
||||||
(lambda (opt name arg result arg-handler)
|
|
||||||
(leave (_ "~A: unrecognized option~%") name))
|
|
||||||
(lambda (arg result arg-handler)
|
|
||||||
(if arg-handler
|
|
||||||
(arg-handler arg result)
|
|
||||||
(leave (_ "~A: extraneous argument~%") arg)))
|
|
||||||
%default-options
|
|
||||||
#f))
|
|
||||||
|
|
||||||
(define (ensure-default-profile)
|
(define (ensure-default-profile)
|
||||||
;; Ensure the default profile symlink and directory exist and are
|
;; Ensure the default profile symlink and directory exist and are
|
||||||
|
@ -987,7 +976,8 @@ (define (list-generation number)
|
||||||
|
|
||||||
(_ #f))))
|
(_ #f))))
|
||||||
|
|
||||||
(let ((opts (parse-options)))
|
(let ((opts (parse-command-line args %options (list %default-options #f)
|
||||||
|
#:argument-handler handle-argument)))
|
||||||
(with-error-handling
|
(with-error-handling
|
||||||
(or (process-query opts)
|
(or (process-query opts)
|
||||||
(parameterize ((%store (open-connection)))
|
(parameterize ((%store (open-connection)))
|
||||||
|
|
|
@ -487,26 +487,15 @@ (define %default-options
|
||||||
;;;
|
;;;
|
||||||
|
|
||||||
(define (guix-system . args)
|
(define (guix-system . args)
|
||||||
(define (parse-options)
|
(define (parse-sub-command arg result)
|
||||||
;; Return the alist of option values.
|
;; Parse sub-command ARG and augment RESULT accordingly.
|
||||||
(append (parse-options-from args)
|
(if (assoc-ref result 'action)
|
||||||
(parse-options-from (environment-build-options))))
|
(alist-cons 'argument arg result)
|
||||||
|
(let ((action (string->symbol arg)))
|
||||||
(define (parse-options-from args)
|
(case action
|
||||||
;; Actual parsing takes place here.
|
((build vm vm-image disk-image reconfigure init)
|
||||||
(args-fold* args %options
|
(alist-cons 'action action result))
|
||||||
(lambda (opt name arg result)
|
(else (leave (_ "~a: unknown action~%") action))))))
|
||||||
(leave (_ "~A: unrecognized option~%") name))
|
|
||||||
(lambda (arg result)
|
|
||||||
(if (assoc-ref result 'action)
|
|
||||||
(alist-cons 'argument arg result)
|
|
||||||
(let ((action (string->symbol arg)))
|
|
||||||
(case action
|
|
||||||
((build vm vm-image disk-image reconfigure init)
|
|
||||||
(alist-cons 'action action result))
|
|
||||||
(else (leave (_ "~a: unknown action~%")
|
|
||||||
action))))))
|
|
||||||
%default-options))
|
|
||||||
|
|
||||||
(define (match-pair car)
|
(define (match-pair car)
|
||||||
;; Return a procedure that matches a pair with CAR.
|
;; Return a procedure that matches a pair with CAR.
|
||||||
|
@ -534,7 +523,10 @@ (define (fail)
|
||||||
args))
|
args))
|
||||||
|
|
||||||
(with-error-handling
|
(with-error-handling
|
||||||
(let* ((opts (parse-options))
|
(let* ((opts (parse-command-line args %options
|
||||||
|
(list %default-options)
|
||||||
|
#:argument-handler
|
||||||
|
parse-sub-command))
|
||||||
(args (option-arguments opts))
|
(args (option-arguments opts))
|
||||||
(file (first args))
|
(file (first args))
|
||||||
(action (assoc-ref opts 'action))
|
(action (assoc-ref opts 'action))
|
||||||
|
|
28
guix/ui.scm
28
guix/ui.scm
|
@ -66,7 +66,7 @@ (define-module (guix ui)
|
||||||
string->generations
|
string->generations
|
||||||
string->duration
|
string->duration
|
||||||
args-fold*
|
args-fold*
|
||||||
environment-build-options
|
parse-command-line
|
||||||
run-guix-command
|
run-guix-command
|
||||||
program-name
|
program-name
|
||||||
guix-warning-port
|
guix-warning-port
|
||||||
|
@ -754,6 +754,32 @@ (define (environment-build-options)
|
||||||
"Return additional build options passed as environment variables."
|
"Return additional build options passed as environment variables."
|
||||||
(arguments-from-environment-variable "GUIX_BUILD_OPTIONS"))
|
(arguments-from-environment-variable "GUIX_BUILD_OPTIONS"))
|
||||||
|
|
||||||
|
(define %default-argument-handler
|
||||||
|
;; The default handler for non-option command-line arguments.
|
||||||
|
(lambda (arg result)
|
||||||
|
(alist-cons 'argument arg result)))
|
||||||
|
|
||||||
|
(define* (parse-command-line args options seeds
|
||||||
|
#:key
|
||||||
|
(argument-handler %default-argument-handler))
|
||||||
|
"Parse the command-line arguments ARGS as well as arguments passed via the
|
||||||
|
'GUIX_BUILD_OPTIONS' environment variable according to OPTIONS (a list of
|
||||||
|
SRFI-37 options) and return the result, seeded by SEEDS.
|
||||||
|
Command-line options take precedence those passed via 'GUIX_BUILD_OPTIONS'.
|
||||||
|
|
||||||
|
ARGUMENT-HANDLER is called for non-option arguments, like the 'operand-proc'
|
||||||
|
parameter of 'args-fold'."
|
||||||
|
(define (parse-options-from args)
|
||||||
|
;; Actual parsing takes place here.
|
||||||
|
(apply args-fold* args options
|
||||||
|
(lambda (opt name arg . rest)
|
||||||
|
(leave (_ "~A: unrecognized option~%") name))
|
||||||
|
argument-handler
|
||||||
|
seeds))
|
||||||
|
|
||||||
|
(append (parse-options-from args)
|
||||||
|
(parse-options-from (environment-build-options))))
|
||||||
|
|
||||||
(define (show-guix-usage)
|
(define (show-guix-usage)
|
||||||
(format (current-error-port)
|
(format (current-error-port)
|
||||||
(_ "Try `guix --help' for more information.~%"))
|
(_ "Try `guix --help' for more information.~%"))
|
||||||
|
|
31
tests/ui.scm
31
tests/ui.scm
|
@ -22,6 +22,8 @@ (define-module (test-ui)
|
||||||
#:use-module (guix profiles)
|
#:use-module (guix profiles)
|
||||||
#:use-module (guix store)
|
#:use-module (guix store)
|
||||||
#:use-module (guix derivations)
|
#:use-module (guix derivations)
|
||||||
|
#:use-module ((guix scripts build)
|
||||||
|
#:select (%standard-build-options))
|
||||||
#:use-module (srfi srfi-1)
|
#:use-module (srfi srfi-1)
|
||||||
#:use-module (srfi srfi-11)
|
#:use-module (srfi srfi-11)
|
||||||
#:use-module (srfi srfi-19)
|
#:use-module (srfi srfi-19)
|
||||||
|
@ -52,9 +54,34 @@ (define guile-2.0.9
|
||||||
(item "/gnu/store/...")
|
(item "/gnu/store/...")
|
||||||
(output "out")))
|
(output "out")))
|
||||||
|
|
||||||
|
(define-syntax-rule (with-environment-variable variable value body ...)
|
||||||
|
"Run BODY with VARIABLE set to VALUE."
|
||||||
|
(let ((orig (getenv variable)))
|
||||||
|
(dynamic-wind
|
||||||
|
(lambda ()
|
||||||
|
(setenv variable value))
|
||||||
|
(lambda ()
|
||||||
|
body ...)
|
||||||
|
(lambda ()
|
||||||
|
(if orig
|
||||||
|
(setenv variable orig)
|
||||||
|
(unsetenv variable))))))
|
||||||
|
|
||||||
|
|
||||||
(test-begin "ui")
|
(test-begin "ui")
|
||||||
|
|
||||||
|
(test-equal "parse-command-line"
|
||||||
|
'((argument . "bar") (argument . "foo")
|
||||||
|
(cores . 10) ;takes precedence
|
||||||
|
(substitutes? . #f) (keep-failed? . #t)
|
||||||
|
(max-jobs . 77) (cores . 42))
|
||||||
|
|
||||||
|
(with-environment-variable "GUIX_BUILD_OPTIONS" "-c 42 -M 77"
|
||||||
|
(parse-command-line '("--keep-failed" "--no-substitutes"
|
||||||
|
"--cores=10" "foo" "bar")
|
||||||
|
%standard-build-options
|
||||||
|
(list '()))))
|
||||||
|
|
||||||
(test-assert "fill-paragraph"
|
(test-assert "fill-paragraph"
|
||||||
(every (lambda (column)
|
(every (lambda (column)
|
||||||
(every (lambda (width)
|
(every (lambda (width)
|
||||||
|
@ -246,3 +273,7 @@ (define guile-2.0.9
|
||||||
|
|
||||||
|
|
||||||
(exit (= (test-runner-fail-count (test-runner-current)) 0))
|
(exit (= (test-runner-fail-count (test-runner-current)) 0))
|
||||||
|
|
||||||
|
;;; Local Variables:
|
||||||
|
;;; eval: (put 'with-environment-variable 'scheme-indent-function 2)
|
||||||
|
;;; End:
|
||||||
|
|
Loading…
Reference in a new issue