summaryrefslogtreecommitdiff
path: root/lisp/eshell/esh-opt.el
diff options
context:
space:
mode:
authorGerd Moellmann <gerd@gnu.org>2000-06-23 05:24:10 +0000
committerGerd Moellmann <gerd@gnu.org>2000-06-23 05:24:10 +0000
commitaffbf6477576c38d98111b55fbb1eb5b13d1a735 (patch)
treee7cccedd38944fc20cf2d20a3949246d8d558bf7 /lisp/eshell/esh-opt.el
parent022499fab948938bb763c2a33a8c5ba0c5969fcd (diff)
downloademacs-affbf6477576c38d98111b55fbb1eb5b13d1a735.tar.gz
*** empty log message ***
Diffstat (limited to 'lisp/eshell/esh-opt.el')
-rw-r--r--lisp/eshell/esh-opt.el226
1 files changed, 226 insertions, 0 deletions
diff --git a/lisp/eshell/esh-opt.el b/lisp/eshell/esh-opt.el
new file mode 100644
index 00000000000..9665bc8cc72
--- /dev/null
+++ b/lisp/eshell/esh-opt.el
@@ -0,0 +1,226 @@
+;;; esh-opt --- command options processing
+
+;; Copyright (C) 1999, 2000 Free Sofware Foundation
+
+;; This file is part of GNU Emacs.
+
+;; GNU Emacs is free software; you can redistribute it and/or modify
+;; it under the terms of the GNU General Public License as published by
+;; the Free Software Foundation; either version 2, or (at your option)
+;; any later version.
+
+;; GNU Emacs 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 General Public License for more details.
+
+;; You should have received a copy of the GNU General Public License
+;; along with GNU Emacs; see the file COPYING. If not, write to the
+;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+;; Boston, MA 02111-1307, USA.
+
+(provide 'esh-opt)
+
+(eval-when-compile (require 'esh-maint))
+
+(defgroup eshell-opt nil
+ "The options processing code handles command argument parsing for
+Eshell commands implemented in Lisp."
+ :tag "Command options processing"
+ :group 'eshell)
+
+;;; Commentary:
+
+;;; User Functions:
+
+(defmacro eshell-eval-using-options (name macro-args
+ options &rest body-forms)
+ "Process NAME's MACRO-ARGS using a set of command line OPTIONS.
+After doing so, settings will be stored in local symbols as declared
+by OPTIONS; FORMS will then be evaluated -- assuming all was OK.
+
+The syntax of OPTIONS is:
+
+ '((?C nil nil multi-column \"multi-column display\")
+ (nil \"help\" nil nil \"show this usage display\")
+ (?r \"reverse\" nil reverse-list \"reverse order while sorting\")
+ :external \"ls\"
+ :usage \"[OPTION]... [FILE]...
+ List information about the FILEs (the current directory by default).
+ Sort entries alphabetically across.\")
+
+`eshell-eval-using-options' returns the value of the last form in
+BODY-FORMS. If instead an external command is run, the tag
+`eshell-external' will be thrown with the new process for its value.
+
+Lastly, any remaining arguments will be available in a locally
+interned variable `args' (created using a `let' form)."
+ `(let ((temp-args
+ ,(if (memq ':preserve-args (cadr options))
+ macro-args
+ (list 'eshell-stringify-list
+ (list 'eshell-flatten-list macro-args)))))
+ (let ,(append (mapcar (function
+ (lambda (opt)
+ (or (and (listp opt) (nth 3 opt))
+ 'eshell-option-stub)))
+ (cadr options))
+ '(usage-msg last-value ext-command args))
+ (eshell-do-opt ,name ,options (quote ,body-forms)))))
+
+;;; Internal Functions:
+
+(eval-when-compile
+ (defvar temp-args)
+ (defvar last-value)
+ (defvar usage-msg)
+ (defvar ext-command)
+ (defvar args))
+
+(defun eshell-do-opt (name options body-forms)
+ "Helper function for `eshell-eval-using-options'.
+This code doesn't really need to be macro expanded everywhere."
+ (setq args temp-args)
+ (if (setq
+ ext-command
+ (catch 'eshell-ext-command
+ (when (setq
+ usage-msg
+ (catch 'eshell-usage
+ (setq last-value nil)
+ (if (and (= (length args) 0)
+ (memq ':show-usage options))
+ (throw 'eshell-usage
+ (eshell-show-usage name options)))
+ (setq args (eshell-process-args name args options)
+ last-value (eval (append (list 'progn)
+ body-forms)))
+ nil))
+ (error usage-msg))))
+ (throw 'eshell-external
+ (eshell-external-command ext-command args))
+ last-value))
+
+(defun eshell-show-usage (name options)
+ "Display the usage message for NAME, using OPTIONS."
+ (let ((usage (format "usage: %s %s\n\n" name
+ (cadr (memq ':usage options))))
+ (extcmd (memq ':external options))
+ (post-usage (memq ':post-usage options))
+ had-option)
+ (while options
+ (when (listp (car options))
+ (let ((opt (car options)))
+ (setq had-option t)
+ (cond ((and (nth 0 opt)
+ (nth 1 opt))
+ (setq usage
+ (concat usage
+ (format " %-20s %s\n"
+ (format "-%c, --%s" (nth 0 opt)
+ (nth 1 opt))
+ (nth 4 opt)))))
+ ((nth 0 opt)
+ (setq usage
+ (concat usage
+ (format " %-20s %s\n"
+ (format "-%c" (nth 0 opt))
+ (nth 4 opt)))))
+ ((nth 1 opt)
+ (setq usage
+ (concat usage
+ (format " %-20s %s\n"
+ (format " --%s" (nth 1 opt))
+ (nth 4 opt)))))
+ (t (setq had-option nil)))))
+ (setq options (cdr options)))
+ (if post-usage
+ (setq usage (concat usage (and had-option "\n")
+ (cadr post-usage))))
+ (when extcmd
+ (setq extcmd (eshell-search-path (cadr extcmd)))
+ (if extcmd
+ (setq usage
+ (concat usage
+ (format "
+This command is implemented in Lisp. If an unrecognized option is
+passed to this command, the external version '%s'
+will be called instead." extcmd)))))
+ (throw 'eshell-usage usage)))
+
+(defun eshell-set-option (name ai opt options)
+ "Using NAME's remaining args (index AI), set the OPT within OPTIONS.
+If the option consumes an argument for its value, the argument list
+will be modified."
+ (if (not (nth 3 opt))
+ (eshell-show-usage name options)
+ (if (eq (nth 2 opt) t)
+ (if (> ai (length args))
+ (error "%s: missing option argument" name)
+ (set (nth 3 opt) (nth ai args))
+ (if (> ai 0)
+ (setcdr (nthcdr (1- ai) args) (nthcdr (1+ ai) args))
+ (setq args (cdr args))))
+ (set (nth 3 opt) (or (nth 2 opt) t)))))
+
+(defun eshell-process-option (name switch kind ai options)
+ "For NAME, process SWITCH (of type KIND), from args at index AI.
+The SWITCH will be looked up in the set of OPTIONS.
+
+SWITCH should be either a string or character. KIND should be the
+integer 0 if it's a character, or 1 if it's a string.
+
+The SWITCH is then be matched against OPTIONS. If no matching handler
+is found, and an :external command is defined (and available), it will
+be called; otherwise, an error will be triggered to say that the
+switch is unrecognized."
+ (let* ((opts options)
+ found)
+ (while opts
+ (if (and (listp (car opts))
+ (nth kind (car opts))
+ (if (= kind 0)
+ (eq switch (nth kind (car opts)))
+ (string= switch (nth kind (car opts)))))
+ (progn
+ (eshell-set-option name ai (car opts) options)
+ (setq found t opts nil))
+ (setq opts (cdr opts))))
+ (unless found
+ (let ((extcmd (memq ':external options)))
+ (when extcmd
+ (setq extcmd (eshell-search-path (cadr extcmd)))
+ (if extcmd
+ (throw 'eshell-ext-command extcmd)
+ (if (char-valid-p switch)
+ (error "%s: unrecognized option -%c" name switch)
+ (error "%s: unrecognized option --%s" name switch))))))))
+
+(defun eshell-process-args (name args options)
+ "Process the given ARGS using OPTIONS.
+This assumes that symbols have been intern'd by `eshell-with-options'."
+ (let ((ai 0) arg)
+ (while (< ai (length args))
+ (setq arg (nth ai args))
+ (if (not (and (stringp arg)
+ (string-match "^-\\(-\\)?\\(.*\\)" arg)))
+ (setq ai (1+ ai))
+ (let* ((dash (match-string 1 arg))
+ (switch (match-string 2 arg)))
+ (if (= ai 0)
+ (setq args (cdr args))
+ (setcdr (nthcdr (1- ai) args) (nthcdr (1+ ai) args)))
+ (if dash
+ (if (> (length switch) 0)
+ (eshell-process-option name switch 1 ai options)
+ (setq ai (length args)))
+ (let ((len (length switch))
+ (index 0))
+ (while (< index len)
+ (eshell-process-option name (aref switch index) 0 ai options)
+ (setq index (1+ index)))))))))
+ args)
+
+;;; Code:
+
+;;; esh-opt.el ends here