command-line now prints detailed message on "--help"

This commit is contained in:
Abdulaziz Ghuloum 2008-12-29 00:38:50 -05:00
parent 120a6dab52
commit e72effdb73
1 changed files with 146 additions and 140 deletions

View File

@ -11,23 +11,12 @@
(import (ikarus)) (import (ikarus))
(define (dispatch-opts arguments data* proc*) (define (dispatch-opts arguments data* proc*)
(define-condition-type &help &condition
make-help-condition help-condition?
(extended? help-extended?))
(define-condition-type &unmatched &condition
make-unmatched-condition unmatched-condition?)
(define (help x)
(raise (make-help-condition x)))
(define (unmatched)
(raise (make-unmatched-condition)))
(define (print-usage detailed?) (define (print-usage detailed?)
(define-record-type f (fields id char type def)) (define-record-type f (fields id char type def))
(define (mkf x id) (define (mkf x id)
(make-f id (car x) (cadr x) (cddr x))) (make-f id (car x) (cadr x) (cddr x)))
(define (get type ls) (define (get type ls)
(let-values ([(ls _) (partition (lambda (x) (eq? (f-type x) type)) ls)]) (filter (lambda (x) (eq? (f-type x) type)) ls))
ls))
(define (fmt-req x) (define (fmt-req x)
(format " -~a <~a>" (f-char x) (f-id x))) (format " -~a <~a>" (f-char x) (f-id x)))
(define (fmt-req-no-value x) (define (fmt-req-no-value x)
@ -74,7 +63,8 @@
(lambda (x) (lambda (x)
(printf " -~a defaults to ~a\n" (f-char x) (printf " -~a defaults to ~a\n" (f-char x)
(f-def x))) (f-def x)))
def*)))))) def*)))
(newline))))
(printf "\nUsage:\n") (printf "\nUsage:\n")
(for-each (lambda (x) (apply print-usage-line x)) data*) (for-each (lambda (x) (apply print-usage-line x)) data*)
(print-usage-line "Display this help message" (print-usage-line "Display this help message"
@ -84,19 +74,33 @@
#f #f
#f) #f)
#f) #f)
(define (arguments-match)
(define-condition-type &help &condition
make-help-condition help-condition?
(extended? help-extended?))
(define-condition-type &unmatched &condition
make-unmatched-condition unmatched-condition?)
(define (help x)
(raise (make-help-condition x)))
(define (unmatched)
(raise (make-unmatched-condition)))
(define (option? x) (define (option? x)
(or (equal? x "--help") ;;; hack
(and (string? x) (and (string? x)
(>= (string-length x) 2) (>= (string-length x) 2)
(char=? (string-ref x 0) #\-) (char=? (string-ref x 0) #\-)
(not (char=? (string-ref x 1) #\-)))) (not (char=? (string-ref x 1) #\-)))))
(define (fill-char-opt c ls fields) (define (fill-char-opt c ls fields)
;;; field = [c required ] ;;; field = [c required0 . _] ; requires 0 args
;;; | [c flag . default] ;;; | [c required1 . _] ; requires 1 arg
;;; | [c zero-plus . list] ;;; | [c flag . default] ; toggles default to #t
;;; | [c one-plus . list] ;;; | [c zero-plus . reversed-list]
;;; | [c optional . str] ;;; | [c one-plus . reversed-list]
;;; | [c optional . default] ; overridden by value
;;; | [c ok . value] ; already used, not on input
(let f ([fields fields]) (let f ([fields fields])
(when (null? fields) (unmatched)) (when (null? fields) (unmatched))
(let ([field (car fields)]) (let ([field (car fields)])
@ -121,6 +125,7 @@
(values (cons field fields) ls)))))) (values (cons field fields) ls))))))
(define (fill-option a ls fields) (define (fill-option a ls fields)
(when (string=? a "--help") (help #t))
(if (= (string-length a) 2) (if (= (string-length a) 2)
(let ([char (string-ref a 1)]) (let ([char (string-ref a 1)])
(when (char=? char #\h) (help #f)) (when (char=? char #\h) (help #f))
@ -199,9 +204,9 @@
[(unmatched-condition? con) [(unmatched-condition? con)
(f (cdr data*) (cdr proc*))]) (f (cdr data*) (cdr proc*))])
(apply (car proc*) (apply match (car data*)))))))) (apply (car proc*) (apply match (car data*))))))))
(arguments-match))
(define-syntax command-line-interface
(define-syntax parse-command-line
(lambda (stx) (lambda (stx)
(define (parse-format stx) (define (parse-format stx)
(define (err str x) (define (err str x)
@ -229,7 +234,7 @@
(free-identifier=? x #'(... ...)))) (free-identifier=? x #'(... ...))))
(define (id? x) (define (id? x)
(and (identifier? x) (not (dots? x)))) (and (identifier? x) (not (dots? x))))
(define (parse-command-line ls) (define (command-line-interface ls)
(define (str? x) (define (str? x)
(let ([d (syntax->datum x)]) (let ([d (syntax->datum x)])
(and (string? d) (not (string=? d "--"))))) (and (string? d) (not (string=? d "--")))))
@ -280,7 +285,7 @@
(f (car ls) (cdr ls))))) (f (car ls) (cdr ls)))))
ls)) ls))
(let-values ([(prog opt-strs opt-ids args args-rest dash-rest) (let-values ([(prog opt-strs opt-ids args args-rest dash-rest)
(parse-command-line stx)]) (command-line-interface stx)])
(list (get-fmls prog opt-ids args args-rest dash-rest) (list (get-fmls prog opt-ids args args-rest dash-rest)
opt-strs opt-ids args args-rest dash-rest))) opt-strs opt-ids args args-rest dash-rest)))
(define (parse-clause stx) (define (parse-clause stx)
@ -312,7 +317,7 @@
(printf "OK\n")))])) (printf "OK\n")))]))
(define (command1 ls) (define (command1 ls)
(parse-command-line ls (command-line-interface ls
[(p) "Help0" (list 0 p)] [(p) "Help0" (list 0 p)]
[(p p1) "Help1" (list 1 p p1)] [(p p1) "Help1" (list 1 p p1)]
[(p p1 p2) "Help2" (list 2 p p1 p2)] [(p p1 p2) "Help2" (list 2 p p1 p2)]
@ -325,7 +330,7 @@
(test command1 ("./prog" "p1" "p2" "p3" "p4") #f) (test command1 ("./prog" "p1" "p2" "p3" "p4") #f)
(define (command2 ls) (define (command2 ls)
(parse-command-line ls (command-line-interface ls
[(p p1 p2 p3) "Help3" (list 3 p p1 p2 p3)] [(p p1 p2 p3) "Help3" (list 3 p p1 p2 p3)]
[(p p1 p2 ps ...) "Help2" (list 2 p p1 p2 ps)] [(p p1 p2 ps ...) "Help2" (list 2 p p1 p2 ps)]
[(p p1 ps ...) "Help1" (list 1 p p1 ps)] [(p p1 ps ...) "Help1" (list 1 p p1 ps)]
@ -339,7 +344,7 @@
(test command2 ("./prog" "-h") #f) (test command2 ("./prog" "-h") #f)
(define (command3 ls) (define (command3 ls)
(parse-command-line ls (command-line-interface ls
[(p "-X" xopt "-Y" yopt) (list 'xy p xopt yopt)] [(p "-X" xopt "-Y" yopt) (list 'xy p xopt yopt)]
[(p "-X" xopt) (list 'x p xopt)] [(p "-X" xopt) (list 'x p xopt)]
[(p "-Y" yopt) (list 'y p yopt)])) [(p "-Y" yopt) (list 'y p yopt)]))
@ -352,7 +357,7 @@
(test command3 ("./prog" "-h") #f) (test command3 ("./prog" "-h") #f)
(define (command4 ls) (define (command4 ls)
(parse-command-line ls (command-line-interface ls
[(p "-X?" xopt "-Y?" yopt) (list p xopt yopt)] [(p "-X?" xopt "-Y?" yopt) (list p xopt yopt)]
[(p "-X?" xopt rest ...) (list p xopt rest)])) [(p "-X?" xopt rest ...) (list p xopt rest)]))
@ -366,7 +371,7 @@
(test command4 ("./prog" "-h") #f) (test command4 ("./prog" "-h") #f)
(define (command5 ls) (define (command5 ls)
(parse-command-line ls (command-line-interface ls
[(p "-X=default" xopt) (list p xopt)])) [(p "-X=default" xopt) (list p xopt)]))
(test command5 ("p") ("p" "default")) (test command5 ("p") ("p" "default"))
@ -374,7 +379,7 @@
(test command5 ("./prog" "-h") #f) (test command5 ("./prog" "-h") #f)
(define (command6 ls) (define (command6 ls)
(parse-command-line ls (command-line-interface ls
[(p "-X*" xopts) (list p xopts)] [(p "-X*" xopts) (list p xopts)]
[(p "-X*" xopts "-Y*" yopts) (list p xopts yopts)])) [(p "-X*" xopts "-Y*" yopts) (list p xopts yopts)]))
@ -388,7 +393,7 @@
(test command6 ("./prog" "-h") #f) (test command6 ("./prog" "-h") #f)
(define (command7 ls) (define (command7 ls)
(parse-command-line ls (command-line-interface ls
[(p "-X+" xopts) (list p xopts)] [(p "-X+" xopts) (list p xopts)]
[(p "-X*" xopts "-Y+" yopts) (list p xopts yopts)])) [(p "-X*" xopts "-Y+" yopts) (list p xopts yopts)]))
@ -402,17 +407,18 @@
(test command7 ("./prog" "-h") #f) (test command7 ("./prog" "-h") #f)
(define (command8 ls) (define (command8 ls)
(parse-command-line ls (command-line-interface ls
[(p "-Q=foobar" q "-R=blabla" r "-X?" xopts "-Y?" yopt "-L*" libs "-f" file file* ...) [(p "-Q=foobar" q "-R=blabla" r "-X?" xopts "-Y?" yopt "-L*" libs "-f" file file* ...)
"Does something nice" "Does something nice"
#t])) #t]))
(test command8 ("./prog") #f) (test command8 ("./prog") #f)
(test command8 ("./prog" "-h") #f) (test command8 ("./prog" "-h") #f)
(test command8 ("./prog" "--help") #f)
(define (ls-command ls) (define (ls-command ls)
(parse-command-line ls (command-line-interface ls
[(ls "-A?" A "-B?" B "-C?" C "-F?" F "-G?" G "-H?" H "-L?" L [(ls "-A?" A "-B?" B "-C?" C "-F?" F "-G?" G "-H?" H "-L?" L
"-P?" P "-R?" R "-S?" S "-T?" T "-W?" W "-Z?" Z "-a?" a "-P?" P "-R?" R "-S?" S "-T?" T "-W?" W "-Z?" Z "-a?" a
"-b?" b "-c?" c "-d?" d "-e?" e "-f?" f "-g?" g "-i?" i "-b?" b "-c?" c "-d?" d "-e?" e "-f?" f "-g?" g "-i?" i