123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303 |
- ;;;; getopt-long.test --- long options processing -*- scheme -*-
- ;;;; Thien-Thi Nguyen <ttn@gnu.org> --- August 2001
- ;;;;
- ;;;; Copyright (C) 2001, 2006, 2011 Free Software Foundation, Inc.
- ;;;;
- ;;;; This library is free software; you can redistribute it and/or
- ;;;; modify it under the terms of the GNU Lesser General Public
- ;;;; License as published by the Free Software Foundation; either
- ;;;; version 3 of the License, or (at your option) any later version.
- ;;;;
- ;;;; This library 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
- ;;;; Lesser General Public License for more details.
- ;;;;
- ;;;; You should have received a copy of the GNU Lesser General Public
- ;;;; License along with this library; if not, write to the Free Software
- ;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- (use-modules (test-suite lib)
- (ice-9 getopt-long)
- (ice-9 regex))
- (define-syntax pass-if-fatal-exception
- (syntax-rules ()
- ((_ name exn exp)
- (let ((port (open-output-string)))
- (with-error-to-port port
- (lambda ()
- (run-test
- name #t
- (lambda ()
- (catch (car exn)
- (lambda () exp #f)
- (lambda (k . args)
- (let ((output (get-output-string port)))
- (close-port port)
- (if (string-match (cdr exn) output)
- #t
- (error "Unexpected output" output)))))))))))))
- (defmacro deferr (name-frag re)
- (let ((name (symbol-append 'exception: name-frag)))
- `(define ,name (cons 'quit ,re))))
- (deferr no-such-option "no such option")
- (deferr option-predicate-failed "option predicate failed")
- (deferr option-does-not-support-arg "option does not support argument")
- (deferr option-must-be-specified "option must be specified")
- (deferr option-must-have-arg "option must be specified with argument")
- (with-test-prefix "exported procs"
- (pass-if "`option-ref' defined" (defined? 'option-ref))
- (pass-if "`getopt-long' defined" (defined? 'getopt-long)))
- (with-test-prefix "specifying predicate"
- (define (test1 . args)
- (getopt-long args
- `((test (value #t)
- (predicate ,(lambda (x)
- (string-match "^[0-9]+$" x)))))))
- (pass-if "valid arg"
- (equal? (test1 "foo" "bar" "--test=123")
- '((() "bar") (test . "123"))))
- (pass-if-fatal-exception "invalid arg"
- exception:option-predicate-failed
- (test1 "foo" "bar" "--test=foo"))
- (pass-if-fatal-exception "option has no arg"
- exception:option-must-have-arg
- (test1 "foo" "bar" "--test"))
- )
- (with-test-prefix "not specifying predicate"
- (define (test2 . args)
- (getopt-long args `((test (value #t)))))
- (pass-if "option has arg"
- (equal? (test2 "foo" "bar" "--test=foo")
- '((() "bar") (test . "foo"))))
- (pass-if "option has no arg"
- (equal? (test2 "foo" "bar")
- '((() "bar"))))
- )
- (with-test-prefix "value optional"
- (define (test3 . args)
- (getopt-long args '((foo (value optional) (single-char #\f))
- (bar))))
- (pass-if "long option `foo' w/ arg, long option `bar'"
- (equal? (test3 "prg" "--foo" "fooval" "--bar")
- '((()) (bar . #t) (foo . "fooval"))))
- (pass-if "short option `foo' w/ arg, long option `bar'"
- (equal? (test3 "prg" "-f" "fooval" "--bar")
- '((()) (bar . #t) (foo . "fooval"))))
- (pass-if "short option `foo', long option `bar', no args"
- (equal? (test3 "prg" "-f" "--bar")
- '((()) (bar . #t) (foo . #t))))
- (pass-if "long option `foo', long option `bar', no args"
- (equal? (test3 "prg" "--foo" "--bar")
- '((()) (bar . #t) (foo . #t))))
- (pass-if "long option `bar', short option `foo', no args"
- (equal? (test3 "prg" "--bar" "-f")
- '((()) (foo . #t) (bar . #t))))
- (pass-if "long option `bar', long option `foo', no args"
- (equal? (test3 "prg" "--bar" "--foo")
- '((()) (foo . #t) (bar . #t))))
- )
- (with-test-prefix "option-ref"
- (define (test4 option-arg . args)
- (equal? option-arg (option-ref (getopt-long
- (cons "prog" args)
- '((foo
- (value optional)
- (single-char #\f))
- (bar)))
- 'foo #f)))
- (pass-if "option-ref `--foo 4'"
- (test4 "4" "--foo" "4"))
- (pass-if "option-ref `-f 4'"
- (test4 "4" "-f" "4"))
- (pass-if "option-ref `-f4'"
- (test4 "4" "-f4"))
- (pass-if "option-ref `--foo=4'"
- (test4 "4" "--foo=4"))
- )
- (with-test-prefix "required"
- (define (test5 args specs)
- (getopt-long (cons "foo" args) specs))
- (pass-if "not mentioned, not given"
- (equal? (test5 '() '())
- '((()))))
- (pass-if-fatal-exception "not mentioned, given"
- exception:no-such-option
- (test5 '("--req") '((something))))
- (pass-if "not specified required, not given"
- (equal? (test5 '() '((req (required? #f))))
- '((()))))
- (pass-if "not specified required, given anyway"
- (equal? (test5 '("--req") '((req (required? #f))))
- '((()) (req . #t))))
- (pass-if "not specified required, but w/ value, given anyway w/ \"=\" val"
- (equal? (test5 '("--req=7") '((req (required? #f) (value #t))))
- '((()) (req . "7"))))
- (pass-if "not specified required, but w/ value, given anyway w/ non-\"=\" val"
- (equal? (test5 '("--req" "7") '((req (required? #f) (value #t))))
- '((()) (req . "7"))))
- (pass-if-fatal-exception "specified required, not given"
- exception:option-must-be-specified
- (test5 '() '((req (required? #t)))))
- )
- (with-test-prefix "specified no-value, given anyway"
- (define (test6 args specs)
- (getopt-long (cons "foo" args) specs))
- (pass-if-fatal-exception "using \"=\" syntax"
- exception:option-does-not-support-arg
- (test6 '("--maybe=yes") '((maybe))))
- )
- (with-test-prefix "specified arg required"
- (define (test7 args)
- (getopt-long (cons "foo" args) '((hmm (value #t) (single-char #\H))
- (ignore))))
- (pass-if "short opt, arg given"
- (equal? (test7 '("-H" "99"))
- '((()) (hmm . "99"))))
- (pass-if "long non-\"=\" opt, arg given"
- (equal? (test7 '("--hmm" "100"))
- '((()) (hmm . "100"))))
- (pass-if "long \"=\" opt, arg given"
- (equal? (test7 '("--hmm=101"))
- '((()) (hmm . "101"))))
- (pass-if-fatal-exception "short opt, arg not given"
- exception:option-must-have-arg
- (test7 '("-H")))
- (pass-if-fatal-exception "long non-\"=\" opt, arg not given (next arg an option)"
- exception:option-must-have-arg
- (test7 '("--hmm" "--ignore")))
- (pass-if-fatal-exception "long \"=\" opt, arg not given"
- exception:option-must-have-arg
- (test7 '("--hmm")))
- )
- (with-test-prefix "apples-blimps-catalexis example"
- (define (test8 . args)
- (equal? (sort (getopt-long (cons "foo" args)
- '((apples (single-char #\a))
- (blimps (single-char #\b) (value #t))
- (catalexis (single-char #\c) (value #t))))
- (lambda (a b)
- (cond ((null? (car a)) #t)
- ((null? (car b)) #f)
- (else (string<? (symbol->string (car a))
- (symbol->string (car b)))))))
- '((())
- (apples . #t)
- (blimps . "bang")
- (catalexis . "couth"))))
- (pass-if "normal 1" (test8 "-a" "-b" "bang" "-c" "couth"))
- (pass-if "normal 2" (test8 "-ab" "bang" "-c" "couth"))
- (pass-if "normal 3" (test8 "-ac" "couth" "-b" "bang"))
- (pass-if-fatal-exception "bad ordering causes missing option"
- exception:option-must-have-arg
- (test8 "-abc" "couth" "bang"))
- )
- (with-test-prefix "multiple occurrences"
- (define (test9 . args)
- (equal? (getopt-long (cons "foo" args)
- '((inc (single-char #\I) (value #t))
- (foo (single-char #\f))))
- '((()) (inc . "2") (foo . #t) (inc . "1"))))
- ;; terminology:
- ;; sf -- single-char free
- ;; sa -- single-char abutted
- ;; lf -- long free
- ;; la -- long abutted (using "=")
- (pass-if "sf/sf" (test9 "-I" "1" "-f" "-I" "2"))
- (pass-if "sa/sa" (test9 "-I1" "-f" "-I2"))
- (pass-if "sf/sa" (test9 "-I" "1" "-f" "-I2"))
- (pass-if "sa/sf" (test9 "-I1" "-f" "-I" "2"))
- (pass-if "lf/lf" (test9 "--inc" "1" "-f" "--inc" "2"))
- (pass-if "la/la" (test9 "--inc=1" "-f" "--inc=2"))
- (pass-if "lf/la" (test9 "--inc" "1" "-f" "--inc=2"))
- (pass-if "la/lf" (test9 "--inc=1" "-f" "--inc" "2"))
- (pass-if "sf/lf" (test9 "-I" "1" "-f" "--inc" "2"))
- (pass-if "lf/sf" (test9 "--inc" "1" "-f" "-I" "2"))
- (pass-if "sf/la" (test9 "-I" "1" "-f" "--inc=2"))
- (pass-if "la/sf" (test9 "--inc=1" "-f" "-I" "2"))
- (pass-if "sa/lf" (test9 "-I1" "-f" "--inc" "2"))
- (pass-if "lf/sa" (test9 "--inc" "1" "-f" "-I2"))
- (pass-if "sa/la" (test9 "-I1" "-f" "--inc=2"))
- (pass-if "la/sa" (test9 "--inc=1" "-f" "-I2"))
- )
- (with-test-prefix "stop-at-first-non-option"
- (pass-if "guile-tools compile example"
- (equal? (getopt-long '("guile-tools" "compile" "-Wformat" "eval.scm" "-o" "eval.go")
- '((help (single-char #\h))
- (version (single-char #\v)))
- #:stop-at-first-non-option #t)
- '((() "compile" "-Wformat" "eval.scm" "-o" "eval.go"))))
- )
- ;;; getopt-long.test ends here
|