From 318e2976570e4d2fd6859ff99946ab66fdd5944e Mon Sep 17 00:00:00 2001 From: "Kim F. Storm" Date: Sun, 23 Nov 2003 20:56:10 +0000 Subject: [PATCH] * progmodes/grep.el: New file with grep code from compile.el. (grep): New defcustom group. (grep-window-height): New defcustom, like compilation-window-height. (grep-auto-highlight): New defcustom, like compile-auto-highlight. (grep-scroll-output): New defcustom, like compilation-scroll-output. (grep-command, grep-use-null-device, grep-find-command) (grep-tree-files-aliases, grep-tree-ignore-case) (grep-tree-ignore-CVS-directories): Move to grep custom group. (grep-setup-hook): New hook variable. (grep-mode-map): New keymap for grep commands. Add Grep menu. (grep-use-compilation-buffer): New defcustom. (grep-last-buffer): New defvar, override compilation-last-buffer. (grep): Add optional arg HIGHLIGHT-REGEXP. Doc fix. Call compile-internal with args highlight-regexp and grep-mode-map. --- lisp/progmodes/grep.el | 534 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 534 insertions(+) create mode 100644 lisp/progmodes/grep.el diff --git a/lisp/progmodes/grep.el b/lisp/progmodes/grep.el new file mode 100644 index 00000000000..27932806872 --- /dev/null +++ b/lisp/progmodes/grep.el @@ -0,0 +1,534 @@ +;;; grep.el --- run compiler as inferior of Emacs, parse error messages + +;; Copyright (C) 1985, 86, 87, 93, 94, 95, 96, 97, 98, 1999, 2001, 2002 +;; Free Software Foundation, Inc. + +;; Author: Roland McGrath +;; Maintainer: FSF +;; Keywords: tools, processes + +;; 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. + +;;; Commentary: + +;; This package provides the grep facilities documented in the Emacs +;; user's manual. + +;;; Code: + +(require 'compile) + +(defgroup grep nil + "Run compiler as inferior of Emacs, parse error messages." + :group 'tools + :group 'processes) + + +;;;###autoload +(defcustom grep-window-height nil + "*Number of lines in a grep window. If nil, use `compilation-window-height'." + :type '(choice (const :tag "Default" nil) + integer) + :version "21.4" + :group 'grep) + +(defcustom grep-auto-highlight t + "*Specify how many grep matches to highlight (and parse) initially. +\(Highlighting applies to an grep match when the mouse is over it.) +If this is a number N, all grep matches in the first N lines +are highlighted and parsed as soon as they arrive in Emacs. +If t, highlight and parse the whole grep output as soon as it arrives. +If nil, don't highlight or parse any of the grep buffer until you try to +move to the error messages. + +Those grep matches which are not parsed and highlighted initially +will be parsed and highlighted as soon as you try to move to them." + :type '(choice (const :tag "All" t) + (const :tag "None" nil) + (integer :tag "First N lines")) + :version "21.4" + :group 'grep) + +(defcustom grep-scroll-output nil + "*Non-nil to scroll the *grep* buffer window as output appears. + +Setting it causes the grep commands to put point at the end of their +output window so that the end of the output is always visible rather +than the begining." + :type 'boolean + :version "21.4" + :group 'grep) + +(defcustom grep-command nil + "The default grep command for \\[grep]. +If the grep program used supports an option to always include file names +in its output (such as the `-H' option to GNU grep), it's a good idea to +include it when specifying `grep-command'. + +The default value of this variable is set up by `grep-compute-defaults'; +call that function before using this variable in your program." + :type '(choice string + (const :tag "Not Set" nil)) + :group 'grep) + +(defcustom grep-use-null-device 'auto-detect + "If t, append the value of `null-device' to `grep' commands. +This is done to ensure that the output of grep includes the filename of +any match in the case where only a single file is searched, and is not +necessary if the grep program used supports the `-H' option. + +The default value of this variable is set up by `grep-compute-defaults'; +call that function before using this variable in your program." + :type 'boolean + :type '(choice (const :tag "Do Not Append Null Device" nil) + (const :tag "Append Null Device" t) + (other :tag "Not Set" auto-detect)) + :group 'grep) + +(defcustom grep-find-command nil + "The default find command for \\[grep-find]. +The default value of this variable is set up by `grep-compute-defaults'; +call that function before using this variable in your program." + :type '(choice string + (const :tag "Not Set" nil)) + :group 'grep) + +(defcustom grep-tree-command nil + "The default find command for \\[grep-tree]. +The default value of this variable is set up by `grep-compute-defaults'; +call that function before using this variable in your program. +The following place holders should be present in the string: + - base directory for find + - find options to restrict or expand the directory list + - find options to limit the files matched + - place to put -i if case insensitive grep + - the regular expression searched for." + :type '(choice string + (const :tag "Not Set" nil)) + :version "21.4" + :group 'grep) + +(defcustom grep-tree-files-aliases '( + ("ch" . "*.[ch]") + ("c" . "*.c") + ("h" . "*.h") + ("m" . "[Mm]akefile*") + ("asm" . "*.[sS]") + ("all" . "*") + ("el" . "*.el") + ) + "*Alist of aliases for the FILES argument to `grep-tree'." + :type 'alist + :group 'grep) + +(defcustom grep-tree-ignore-case t + "*If non-nil, `grep-tree' ignores case in matches." + :type 'boolean + :group 'grep) + +(defcustom grep-tree-ignore-CVS-directories t + "*If non-nil, `grep-tree' does no recurse into CVS directories." + :type 'boolean + :group 'grep) + +;;;###autoload +(defcustom grep-setup-hook nil + "List of hook functions run by `grep-process-setup' (see `run-hooks')." + :type 'hook + :group 'grep) + +(defvar grep-mode-map + (let ((map (cons 'keymap compilation-minor-mode-map))) + (define-key map " " 'scroll-up) + (define-key map "\^?" 'scroll-down) + + (define-key map [remap next-line] 'compilation-next-error) + (define-key map [remap previous-line] 'compilation-previous-error) + + (define-key map "\r" 'compile-goto-error) ;; ? + (define-key map "n" 'next-error-no-select) + (define-key map "p" 'previous-error-no-select) + (define-key map "{" 'compilation-previous-file) + (define-key map "}" 'compilation-next-file) + (define-key map "\t" 'compilation-next-file) + + ;; Set up the menu-bar + (define-key map [menu-bar grep] + (cons "Grep" (make-sparse-keymap "Grep"))) + + (define-key map [menu-bar grep compilation-kill-compilation] + '("Kill Grep" . kill-compilation)) + (define-key map [menu-bar grep compilation-separator2] + '("----" . nil)) + (define-key map [menu-bar grep compilation-compile] + '("Compile..." . compile)) + (define-key map [menu-bar grep compilation-grep] + '("Another grep" . grep)) + (define-key map [menu-bar grep compilation-recompile] + '("Repeat grep" . recompile)) + (define-key map [menu-bar grep compilation-separator2] + '("----" . nil)) + (define-key map [menu-bar grep compilation-first-error] + '("First Match" . first-error)) + (define-key map [menu-bar grep compilation-previous-error] + '("Previous Match" . previous-error)) + (define-key map [menu-bar grep compilation-next-error] + '("Next Match" . next-error)) + map) + "Keymap for grep buffers. +`compilation-minor-mode-map' is a cdr of this.") + +;;;; TODO --- refine this!! + +(defcustom grep-use-compilation-buffer t + "When non-nil, grep specific commands update `compilation-last-buffer'. +This means that standard compile commands like \\[next-error] and \\[compile-goto-error] +can be used to navigate between grep matches (the default). +Otherwise, the grep specific commands like \\[grep-next-match] must +be used to navigate between grep matches." + :type 'boolean + :group 'grep) + +;; override compilation-last-buffer +(defvar grep-last-buffer nil + "The most recent grep buffer. +A grep buffer becomes most recent when its process is started +or when it is used with \\[grep-next-match]. +Notice that using \\[next-error] or \\[compile-goto-error] modifies +`complation-last-buffer' rather than `grep-last-buffer'.") + +;; Note: the character class after the optional drive letter does not +;; include a space to support file names with blanks. +(defvar grep-regexp-alist + '(("\\([a-zA-Z]?:?[^:(\t\n]+\\)[:( \t]+\\([0-9]+\\)[:) \t]" 1 2)) + "Regexp used to match grep hits. See `compilation-error-regexp-alist'.") + +(defvar grep-program + ;; Currently zgrep has trouble. It runs egrep instead of grep, + ;; and it doesn't pass along long options right. + "grep" + ;; (if (equal (condition-case nil ; in case "zgrep" isn't in exec-path + ;; (call-process "zgrep" nil nil nil + ;; "foo" null-device) + ;; (error nil)) + ;; 1) + ;; "zgrep" + ;; "grep") + "The default grep program for `grep-command' and `grep-find-command'. +This variable's value takes effect when `grep-compute-defaults' is called.") + +(defvar find-program "find" + "The default find program for `grep-find-command'. +This variable's value takes effect when `grep-compute-defaults' is called.") + +(defvar grep-find-use-xargs nil + "Whether \\[grep-find] uses the `xargs' utility by default. + +If nil, it uses `find -exec'; if `gnu', it uses `find -print0' and `xargs -0'; +if not nil and not `gnu', it uses `find -print' and `xargs'. + +This variable's value takes effect when `grep-compute-defaults' is called.") + +;; History of grep commands. +(defvar grep-history nil) +(defvar grep-find-history nil) + +(defun grep-process-setup () + "Setup compilation variables and buffer for `grep'. +Set up `compilation-exit-message-function' and `compilation-window-height'. +Sets `grep-last-buffer' and runs `grep-setup-hook'." + (setq grep-last-buffer (current-buffer)) + (set (make-local-variable 'compilation-exit-message-function) + (lambda (status code msg) + (if (eq status 'exit) + (cond ((zerop code) + '("finished (matches found)\n" . "matched")) + ((= code 1) + '("finished with no matches found\n" . "no match")) + (t + (cons msg code))) + (cons msg code)))) + (if grep-window-height + (set (make-local-variable 'compilation-window-height) + grep-window-height)) + (set (make-local-variable 'compile-auto-highlight) + grep-auto-highlight) + (set (make-local-variable 'compilation-scroll-output) + grep-scroll-output) + (run-hooks 'grep-setup-hook)) + +(defun grep-compute-defaults () + (unless (or (not grep-use-null-device) (eq grep-use-null-device t)) + (setq grep-use-null-device + (with-temp-buffer + (let ((hello-file (expand-file-name "HELLO" data-directory))) + (not + (and (equal (condition-case nil + (if grep-command + ;; `grep-command' is already set, so + ;; use that for testing. + (call-process-shell-command + grep-command nil t nil + "^English" hello-file) + ;; otherwise use `grep-program' + (call-process grep-program nil t nil + "-nH" "^English" hello-file)) + (error nil)) + 0) + (progn + (goto-char (point-min)) + (looking-at + (concat (regexp-quote hello-file) + ":[0-9]+:English"))))))))) + (unless grep-command + (setq grep-command + (let ((required-options (if grep-use-null-device "-n" "-nH"))) + (if (equal (condition-case nil ; in case "grep" isn't in exec-path + (call-process grep-program nil nil nil + "-e" "foo" null-device) + (error nil)) + 1) + (format "%s %s -e " grep-program required-options) + (format "%s %s " grep-program required-options))))) + (unless grep-find-use-xargs + (setq grep-find-use-xargs + (if (and + (equal (call-process "find" nil nil nil + null-device "-print0") + 0) + (equal (call-process "xargs" nil nil nil + "-0" "-e" "echo") + 0)) + 'gnu))) + (unless grep-find-command + (setq grep-find-command + (cond ((eq grep-find-use-xargs 'gnu) + (format "%s . -type f -print0 | xargs -0 -e %s" + find-program grep-command)) + (grep-find-use-xargs + (format "%s . -type f -print | xargs %s" + find-program grep-command)) + (t (cons (format "%s . -type f -exec %s {} %s \\;" + find-program grep-command null-device) + (+ 22 (length grep-command))))))) + (unless grep-tree-command + (setq grep-tree-command + (let* ((glen (length grep-program)) + (gcmd (concat grep-program " " (substring grep-command glen)))) + (cond ((eq grep-find-use-xargs 'gnu) + (format "%s -type f -print0 | xargs -0 -e %s " + find-program gcmd)) + (grep-find-use-xargs + (format "%s -type f -print | xargs %s " + find-program gcmd)) + (t (format "%s -type f -exec %s {} %s \\;" + find-program gcmd null-device))))))) + +(defun grep-default-command () + (let ((tag-default + (funcall (or find-tag-default-function + (get major-mode 'find-tag-default-function) + ;; We use grep-tag-default instead of + ;; find-tag-default, to avoid loading etags. + 'grep-tag-default))) + (sh-arg-re "\\(\\(?:\"\\(?:[^\"]\\|\\\\\"\\)+\"\\|'[^']+'\\|[^\"' \t\n]\\)+\\)") + (grep-default (or (car grep-history) grep-command))) + ;; Replace the thing matching for with that around cursor. + (when (or (string-match + (concat "[^ ]+\\s +\\(?:-[^ ]+\\s +\\)*" + sh-arg-re "\\(\\s +\\(\\S +\\)\\)?") + grep-default) + ;; If the string is not yet complete. + (string-match "\\(\\)\\'" grep-default)) + (unless (or (not (stringp buffer-file-name)) + (when (match-beginning 2) + (save-match-data + (string-match + (wildcard-to-regexp + (file-name-nondirectory + (match-string 3 grep-default))) + (file-name-nondirectory buffer-file-name))))) + (setq grep-default (concat (substring grep-default + 0 (match-beginning 2)) + " *." + (file-name-extension buffer-file-name)))) + (replace-match (or tag-default "") t t grep-default 1)))) + +;;;###autoload +(defun grep (command-args &optional highlight-regexp) + "Run grep, with user-specified args, and collect output in a buffer. +While grep runs asynchronously, you can use \\[next-error] (M-x next-error), +or \\\\[compile-goto-error] in the grep \ +output buffer, to go to the lines +where grep found matches. + +This command uses a special history list for its COMMAND-ARGS, so you can +easily repeat a grep command. + +A prefix argument says to default the argument based upon the current +tag the cursor is over, substituting it into the last grep command +in the grep command history (or into `grep-command' +if that history list is empty). + +If specified, optional second arg HIGHLIGHT-REGEXP is the regexp to +temporarily highlight in visited source lines." + (interactive + (progn + (unless (and grep-command + (or (not grep-use-null-device) (eq grep-use-null-device t))) + (grep-compute-defaults)) + (let ((default (grep-default-command))) + (list (read-from-minibuffer "Run grep (like this): " + (if current-prefix-arg + default grep-command) + nil nil 'grep-history + (if current-prefix-arg nil default)))))) + + ;; Setting process-setup-function makes exit-message-function work + ;; even when async processes aren't supported. + (let* ((compilation-process-setup-function 'grep-process-setup) + (buf (compile-internal (if (and grep-use-null-device null-device) + (concat command-args " " null-device) + command-args) + "No more grep hits" "grep" + ;; Give it a simpler regexp to match. + nil grep-regexp-alist + nil nil nil nil nil nil + highlight-regexp grep-mode-map))))) + +;; This is a copy of find-tag-default from etags.el. +(defun grep-tag-default () + (save-excursion + (while (looking-at "\\sw\\|\\s_") + (forward-char 1)) + (when (or (re-search-backward "\\sw\\|\\s_" + (save-excursion (beginning-of-line) (point)) + t) + (re-search-forward "\\(\\sw\\|\\s_\\)+" + (save-excursion (end-of-line) (point)) + t)) + (goto-char (match-end 0)) + (buffer-substring (point) + (progn (forward-sexp -1) + (while (looking-at "\\s'") + (forward-char 1)) + (point)))))) + +;;;###autoload +(defun grep-find (command-args) + "Run grep via find, with user-specified args COMMAND-ARGS. +Collect output in a buffer. +While find runs asynchronously, you can use the \\[next-error] command +to find the text that grep hits refer to. + +This command uses a special history list for its arguments, so you can +easily repeat a find command." + (interactive + (progn + (unless grep-find-command + (grep-compute-defaults)) + (list (read-from-minibuffer "Run find (like this): " + grep-find-command nil nil + 'grep-find-history)))) + (let ((null-device nil)) ; see grep + (grep command-args))) + +(defun grep-expand-command-macros (command &optional regexp files dir excl case-fold) + "Patch grep COMMAND replacing , etc." + (setq command + (replace-regexp-in-string "" + (or dir ".") command t t)) + (setq command + (replace-regexp-in-string "" + (or excl "") command t t)) + (setq command + (replace-regexp-in-string "" + (or files "") command t t)) + (setq command + (replace-regexp-in-string "" + (if case-fold "-i" "") command t t)) + (setq command + (replace-regexp-in-string "" + (or regexp "") command t t)) + command) + +(defvar grep-tree-last-regexp "") +(defvar grep-tree-last-files (car (car grep-tree-files-aliases))) + +;;;###autoload +(defun grep-tree (regexp files dir &optional subdirs) + "Grep for REGEXP in FILES in directory tree rooted at DIR. +Collect output in a buffer. +Interactively, prompt separately for each search parameter. +With prefix arg, reuse previous REGEXP. +The search is limited to file names matching shell pattern FILES. +FILES may use abbreviations defined in `grep-tree-files-aliases', e.g. +entering `ch' is equivalent to `*.[ch]'. + +While find runs asynchronously, you can use the \\[next-error] command +to find the text that grep hits refer to. + +This command uses a special history list for its arguments, so you can +easily repeat a find command. + +When used non-interactively, optional arg SUBDIRS limits the search to +those sub directories of DIR." + (interactive + (let* ((regexp + (if current-prefix-arg + grep-tree-last-regexp + (let* ((default (current-word)) + (spec (read-string + (concat "Search for" + (if (and default (> (length default) 0)) + (format " (default %s): " default) ": "))))) + (if (equal spec "") default spec)))) + (files + (read-string (concat "Search for \"" regexp "\" in files (default " grep-tree-last-files "): "))) + (dir + (read-directory-name "Base directory: " nil default-directory t))) + (list regexp files dir))) + (unless grep-tree-command + (grep-compute-defaults)) + (unless (and (stringp files) (> (length files) 0)) + (setq files grep-tree-last-files)) + (when files + (setq grep-tree-last-files files) + (let ((mf (assoc files grep-tree-files-aliases))) + (if mf + (setq files (cdr mf))))) + (let ((command-args (grep-expand-command-macros + grep-tree-command + (setq grep-tree-last-regexp regexp) + (and files (concat "-name '" files "'")) + (if subdirs + (if (stringp subdirs) + subdirs + (mapconcat 'identity subdirs " ")) + nil) ;; we change default-directory to dir + (and grep-tree-ignore-CVS-directories "-path '*/CVS' -prune -o ") + grep-tree-ignore-case)) + (default-directory dir) + (null-device nil)) ; see grep + (grep command-args regexp))) + + +(provide 'grep) + +;;; grep.el ends here + -- 2.39.5