summaryrefslogtreecommitdiff
path: root/lisp/emacs-lisp/checkdoc.el
diff options
context:
space:
mode:
Diffstat (limited to 'lisp/emacs-lisp/checkdoc.el')
-rw-r--r--lisp/emacs-lisp/checkdoc.el2849
1 files changed, 2849 insertions, 0 deletions
diff --git a/lisp/emacs-lisp/checkdoc.el b/lisp/emacs-lisp/checkdoc.el
new file mode 100644
index 00000000000..3f9bc28e0b0
--- /dev/null
+++ b/lisp/emacs-lisp/checkdoc.el
@@ -0,0 +1,2849 @@
+;;; checkdoc.el --- check documentation strings for style requirements -*- lexical-binding:t -*-
+
+;; Copyright (C) 1997-2022 Free Software Foundation, Inc.
+
+;; Author: Eric M. Ludlam <zappo@gnu.org>
+;; Old-Version: 0.6.2
+;; Keywords: docs, maint, lisp
+
+;; 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 3 of the License, 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. If not, see <https://www.gnu.org/licenses/>.
+
+;;; Commentary:
+;;
+;; The Emacs Lisp manual has a nice chapter on how to write
+;; documentation strings. Many stylistic suggestions are fairly
+;; deterministic and easy to check for syntactically, but also easy
+;; to forget. The main checkdoc engine will perform the stylistic
+;; checks needed to make sure these styles are remembered.
+;;
+;; There are three ways to use checkdoc:
+;; 1) Use `flymake-mode'.
+;; 2) Periodically use `checkdoc' or `checkdoc-current-buffer'.
+;; `checkdoc' is a more interactive version of
+;; `checkdoc-current-buffer'
+;; 3) Use `checkdoc-minor-mode' to automatically check your
+;; documentation whenever you evaluate Lisp code with C-M-x
+;; or [menu-bar emacs-lisp eval-buffer]. Additional key-bindings
+;; are also provided under C-c ? KEY
+;; (add-hook 'emacs-lisp-mode-hook 'checkdoc-minor-mode)
+;;
+;; Using `checkdoc':
+;;
+;; The commands `checkdoc' and `checkdoc-ispell' are the top-level
+;; entry points to all of the different checks that are available. It
+;; breaks examination of your Lisp file into four sections (comments,
+;; documentation, messages, and spacing) and indicates its current
+;; state in a status buffer.
+;;
+;; The Comments check examines your headers, footers, and
+;; various tags (such as "Code:") to make sure that your code is ready
+;; for easy integration into existing systems.
+;;
+;; The Documentation check deals with documentation strings
+;; and their elements that help make Emacs easier to use.
+;;
+;; The Messages check ensures that the strings displayed in the
+;; minibuffer by some commands (such as `error' and `y-or-n-p')
+;; are consistent with the Emacs environment.
+;;
+;; The Spacing check cleans up white-space at the end of lines.
+;;
+;; The interface while working with documentation and messages is
+;; slightly different when being run in the interactive mode. The
+;; interface offers several options, including the ability to skip to
+;; the next error, or back up to previous errors. Auto-fixing is
+;; turned off at this stage, but you can use the `f' or `F' key to fix
+;; a given error (if the fix is available.)
+;;
+;; Auto-fixing:
+;;
+;; There are four classifications of style errors in terms of how
+;; easy they are to fix. They are simple, complex, really complex,
+;; and impossible. (Impossible really means that checkdoc does not
+;; have a fixing routine yet.) Typically white-space errors are
+;; classified as simple, and are auto-fixed by default. Typographic
+;; changes are considered complex, and the user is asked if they want
+;; the problem fixed before checkdoc makes the change. These changes
+;; can be done without asking if `checkdoc-autofix-flag' is properly
+;; set. Potentially redundant changes are considered really complex,
+;; and the user is always asked before a change is inserted. The
+;; variable `checkdoc-autofix-flag' controls how these types of errors
+;; are fixed.
+;;
+;; Spell checking text:
+;;
+;; The variable `checkdoc-spellcheck-documentation-flag' can be set
+;; to customize how spell checking is to be done. Since spell
+;; checking can be quite slow, you can optimize how best you want your
+;; checking done. The default is `defun', which spell checks each time
+;; `checkdoc-defun' or `checkdoc-eval-defun' is used. Setting to nil
+;; prevents spell checking during normal usage.
+;; Setting this variable to nil does not mean you cannot take
+;; advantage of the spell checking. You can instead use the
+;; interactive functions `checkdoc-ispell-*' to check the spelling of
+;; your documentation.
+;; There is a list of Lisp-specific words which checkdoc will
+;; install into Ispell on the fly, but only if Ispell is not already
+;; running. Use `ispell-kill-ispell' to make checkdoc restart it with
+;; these words enabled.
+;;
+;; Checking parameters:
+;;
+;; You might not always want a function to have its parameters listed
+;; in order. When this is the case, put the following comment just in
+;; front of the documentation string: "; checkdoc-order: nil" This
+;; overrides the value of `checkdoc-arguments-in-order-flag'.
+;;
+;; If you specifically wish to avoid mentioning a parameter of a
+;; function in the doc string (such as a hidden parameter, or a
+;; parameter which is very obvious like events), you can have checkdoc
+;; skip looking for it by putting the following comment just in front
+;; of the documentation string: "; checkdoc-params: (args go here)"
+;;
+;; Checking message strings:
+;;
+;; The text that follows the `error' and `y-or-n-p' commands is
+;; also checked. The documentation for `error' clearly states some
+;; simple style rules to follow which checkdoc will auto-fix for you.
+;; `y-or-n-p' and `yes-or-no-p' should also end in "?".
+;;
+;; Adding your own checks:
+;;
+;; You can experiment with adding your own checks by setting the
+;; hooks `checkdoc-style-functions' and `checkdoc-comment-style-functions'.
+;; Return a string which is the error you wish to report. The cursor
+;; position should be preserved.
+;;
+;; Error errors:
+;;
+;; Checkdoc does not always flag errors correctly. There are a
+;; couple ways you can coax your file into passing all of checkdoc's
+;; tests through buffer local variables.
+;;
+;; The variable `checkdoc-verb-check-experimental-flag' can be used
+;; to turn off the check for verb-voice in case you use words that are
+;; not semantically verbs, but are still in the incomplete list.
+;;
+;; The variable `checkdoc-symbol-words' can be a list of words that
+;; happen to also be symbols. This is not a problem for one-word
+;; symbols, but if you use a hyphenated word that is also a symbol,
+;; then you may need this.
+;;
+;; The symbol `checkdoc-force-docstrings-flag' can be set to nil if
+;; you have many undocumented functions you don't wish to document.
+;;
+;; See the above section "Checking Parameters" for details about
+;; parameter checking.
+
+;;; TO DO:
+;; Hook into the byte compiler on a defun/defvar level to generate
+;; warnings in the byte-compiler's warning/error buffer.
+;; Better ways to override more typical `eval' functions. Advice
+;; might be good but hard to turn on/off as a minor mode.
+;;
+;;; Maybe Do:
+;; Code sweep checks for "forbidden functions", proper use of hooks,
+;; proper keybindings, and other items from the manual that are
+;; not specifically docstring related. Would this even be useful?
+
+;;; Code:
+
+(require 'bytecomp) ;; for byte-compile-docstring-max-column
+(require 'cl-lib)
+(require 'help-mode) ;; for help-xref-info-regexp
+(require 'thingatpt) ;; for handy thing-at-point-looking-at
+(require 'lisp-mode) ;; for lisp-mode-symbol regexp
+(eval-when-compile (require 'dired)) ;; for dired-map-over-marks
+(require 'lisp-mnt)
+
+(defvar compilation-error-regexp-alist)
+(defvar compilation-mode-font-lock-keywords)
+
+(defgroup checkdoc nil
+ "Support for doc string checking in Emacs Lisp."
+ :prefix "checkdoc"
+ :group 'lisp
+ :version "20.3")
+
+(defcustom checkdoc-minor-mode-string " CDoc"
+ "String to display in mode line when Checkdoc mode is enabled; nil for none."
+ :type '(choice string (const :tag "None" nil))
+ :version "23.1")
+
+(defcustom checkdoc-autofix-flag 'semiautomatic
+ "Non-nil means attempt auto-fixing of doc strings.
+If this value is the symbol `query', then the user is queried before
+any change is made. If the value is `automatic', then all changes are
+made without asking unless the change is very complex. If the value
+is `semiautomatic' or any other value, then simple fixes are made
+without asking, and complex changes are made by asking the user first.
+The value `never' is the same as nil, never ask or change anything."
+ :type '(choice (const automatic)
+ (const query)
+ (const never)
+ (other :tag "semiautomatic" semiautomatic)))
+
+(defcustom checkdoc-bouncy-flag t
+ "Non-nil means to \"bounce\" to auto-fix locations.
+Setting this to nil will silently make fixes that require no user
+interaction. See `checkdoc-autofix-flag' for auto-fixing details."
+ :type 'boolean)
+
+(defcustom checkdoc-force-docstrings-flag t
+ "Non-nil means that all checkable definitions should have documentation.
+Style guide dictates that interactive functions MUST have documentation,
+and that it's good but not required practice to make non user visible items
+have doc strings."
+ :type 'boolean)
+;;;###autoload(put 'checkdoc-force-docstrings-flag 'safe-local-variable #'booleanp)
+
+(defcustom checkdoc-force-history-flag nil
+ "Non-nil means that files should have a History section or ChangeLog file.
+This helps document the evolution of, and recent changes to, the package."
+ :type 'boolean)
+;;;###autoload(put 'checkdoc-force-history-flag 'safe-local-variable #'booleanp)
+
+(defcustom checkdoc-permit-comma-termination-flag nil
+ "Non-nil means the first line of a docstring may end with a comma.
+Ordinarily, a full sentence is required. This may be misleading when
+there is a substantial caveat to the one-line description -- the comma
+should be used when the first part could stand alone as a sentence, but
+it indicates that a modifying clause follows."
+ :type 'boolean)
+;;;###autoload(put 'checkdoc-permit-comma-termination-flag 'safe-local-variable #'booleanp)
+
+(defcustom checkdoc-spellcheck-documentation-flag nil
+ "Non-nil means run Ispell on text based on value.
+This is automatically set to nil if Ispell does not exist on your
+system. Possible values are:
+
+ nil - Don't spell-check during basic style checks.
+ defun - Spell-check when style checking a single defun.
+ buffer - Spell-check when style checking the whole buffer.
+ interactive - Spell-check during any interactive check.
+ t - Always spell-check.
+
+There is a list of Lisp-specific words which checkdoc will
+install into Ispell on the fly, but only if Ispell is not already
+running. Use `ispell-kill-ispell' to make checkdoc restart it
+with these words enabled."
+ :type '(choice (const nil)
+ (const defun)
+ (const buffer)
+ (const interactive)
+ (const t)))
+;;;###autoload(put 'checkdoc-spellcheck-documentation-flag 'safe-local-variable #'booleanp)
+
+(defvar checkdoc-ispell-lisp-words
+ '("alist" "emacs" "etags" "keymap" "paren" "regexp" "sexp")
+ "List of words that are correct when spell-checking Lisp documentation.")
+;;;###autoload(put 'checkdoc-ispell-list-words 'safe-local-variable #'list-of-strings-p)
+
+(defcustom checkdoc-max-keyref-before-warn nil
+ "If non-nil, number of \\\\=[command-to-keystroke] tokens allowed in a doc string.
+Any more than this and a warning is generated suggesting that the construct
+\\\\={mapvar} be used instead. If the value is nil, never warn.
+
+It used to not be practical to use `\\\\=[...]' very many times,
+because display of the documentation string would become slow.
+This is not an issue on modern machines, unless you have
+thousands of substitutions."
+ :type '(choice (const nil)
+ integer)
+ :version "28.1")
+
+(defcustom checkdoc-arguments-in-order-flag nil
+ "Non-nil means warn if arguments appear out of order.
+Setting this to nil will mean only checking that all the arguments
+appear in the proper form in the documentation, not that they are in
+the same order as they appear in the argument list. No mention is
+made in the style guide relating to order."
+ :version "26.1"
+ :type 'boolean)
+;;;###autoload(put 'checkdoc-arguments-in-order-flag 'safe-local-variable #'booleanp)
+
+(defcustom checkdoc-package-keywords-flag nil
+ "Non-nil means warn if this file's package keywords are not recognized.
+Currently, all recognized keywords must be on `finder-known-keywords'."
+ :version "25.1"
+ :type 'boolean)
+
+(defvar checkdoc-style-functions nil
+ "Hook run after the standard style check is completed.
+All functions must return nil or a string representing the error found.
+Useful for adding new user implemented commands.
+
+Each hook is called with two parameters, (DEFUNINFO ENDPOINT).
+DEFUNINFO is the return value of `checkdoc-defun-info'. ENDPOINT is the
+location of end of the documentation string.")
+
+(defvar checkdoc-comment-style-functions nil
+ "Hook run after the standard comment style check is completed.
+Must return nil if no errors are found, or a string describing the
+problem discovered. This is useful for adding additional checks.")
+
+(defvar checkdoc-diagnostic-buffer "*Style Warnings*"
+ "Name of warning message buffer.")
+
+(defcustom checkdoc-verb-check-experimental-flag t
+ "Non-nil means to attempt to check the voice of the doc string.
+This check keys off some words which are commonly misused. See the
+variable `checkdoc-common-verbs-wrong-voice' if you wish to add your own."
+ :type 'boolean)
+;;;###autoload(put 'checkdoc-verb-check-experimental-flag 'safe-local-variable #'booleanp)
+
+(defvar checkdoc-generate-compile-warnings-flag nil
+ "Non-nil means generate warnings in a buffer for browsing.
+Do not set this by hand, use a function like `checkdoc-current-buffer'
+with a universal argument.")
+
+(defcustom checkdoc-symbol-words
+ '("beginning-of-buffer" "beginning-of-line" "byte-code"
+ "byte-compile" "command-line" "end-of-buffer" "end-of-line"
+ "major-mode" "point-max" "point-min" "syntax-table"
+ "top-level" "user-error" "version-control" "window-system")
+ "A list of symbol names (strings) which also happen to make good words.
+These words are ignored when unquoted symbols are searched for.
+This should be set in an Emacs Lisp file's local variables."
+ :type '(repeat (string :tag "Word"))
+ :version "28.1")
+;;;###autoload(put 'checkdoc-symbol-words 'safe-local-variable #'list-of-strings-p)
+
+(defcustom checkdoc-column-zero-backslash-before-paren t
+ "Non-nil means to warn if there is no \"\\\" before \"(\" in column zero.
+This backslash is no longer needed on Emacs 27.1 or later.
+
+See Info node `(elisp) Documentation Tips' for background."
+ :type 'boolean
+ :version "28.1")
+
+;; This is how you can use checkdoc to make mass fixes on the Emacs
+;; source tree:
+;;
+;; (setq checkdoc--argument-missing-flag nil) ; optional
+;; (setq checkdoc--disambiguate-symbol-flag nil) ; optional
+;; (setq checkdoc--interactive-docstring-flag nil) ; optional
+;; (setq checkdoc-verb-check-experimental-flag nil)
+;; Then use `M-x find-dired' ("-name '*.el'") and `M-x checkdoc-dired'
+
+(defvar checkdoc--argument-missing-flag t
+ "Non-nil means warn if arguments are missing from docstring.
+This variable is intended for use on Emacs itself, where the
+large number of libraries means it is impractical to fix all
+of these warnings en masse. In almost any other case, setting
+this to anything but t is likely to be counter-productive.")
+
+(defvar checkdoc--disambiguate-symbol-flag t
+ "Non-nil means ask to disambiguate Lisp symbol.
+This variable is intended for use on Emacs itself, where the
+large number of libraries means it is impractical to fix all
+of these warnings masse. In almost any other case, setting
+this to anything but t is likely to be counter-productive.")
+
+(defvar checkdoc--interactive-docstring-flag t
+ "Non-nil means warn if interactive function has no docstring.
+This variable is intended for use on Emacs itself, where the
+large number of libraries means it is impractical to fix all
+of these warnings masse. In almost any other case, setting
+this to anything but t is likely to be counter-productive.")
+
+(defun checkdoc-list-of-strings-p (obj)
+ "Return t when OBJ is a list of strings."
+ (declare (obsolete list-of-strings-p "29.1"))
+ ;; this is a function so it might be shared by checkdoc-proper-noun-list
+ ;; and/or checkdoc-ispell-lisp-words in the future
+ (and (listp obj)
+ (not (memq nil (mapcar #'stringp obj)))))
+
+(defvar checkdoc-proper-noun-list
+ '("emacs" "lisp" "dired")
+ "List of words (not capitalized) which should be capitalized.")
+
+(defvar checkdoc-proper-noun-regexp
+ ;; "[.!?]" is for noun at end of a sentence, since those chars
+ ;; are symbol syntax in emacs-lisp-mode and so don't match \\_>.
+ ;; The \" allows it to be the last sentence in a docstring too.
+ (concat "\\_<"
+ (regexp-opt checkdoc-proper-noun-list t)
+ "\\(\\_>\\|[.!?][ \t\n\"]\\)")
+ "Regular expression derived from `checkdoc-proper-noun-regexp'.")
+;;;###autoload(put 'checkdoc-proper-noun-regexp 'safe-local-variable 'stringp)
+
+(defvar checkdoc-common-verbs-regexp nil
+ "Regular expression derived from `checkdoc-common-verbs-regexp'.")
+;;;###autoload(put 'checkdoc-common-verbs-regexp 'safe-local-variable 'stringp)
+
+(defvar checkdoc-common-verbs-wrong-voice
+ '(("adds" . "add")
+ ("allows" . "allow")
+ ("appends" . "append")
+ ("applies" . "apply")
+ ("arranges" . "arrange")
+ ("brings" . "bring")
+ ("calls" . "call")
+ ("catches" . "catch")
+ ("changes" . "change")
+ ("checks" . "check")
+ ("contains" . "contain")
+ ("converts" . "convert")
+ ("creates" . "create")
+ ("defines" . "define")
+ ("destroys" . "destroy")
+ ("determines" . "determine")
+ ("disables" . "disable")
+ ("echoes" . "echo")
+ ("executes" . "execute")
+ ("extends" . "extend")
+ ("evals" . "evaluate")
+ ("evaluates" . "evaluate")
+ ("finds" . "find")
+ ("forces" . "force")
+ ("gathers" . "gather")
+ ("generates" . "generate")
+ ("goes" . "go")
+ ("guesses" . "guess")
+ ("highlights" . "highlight")
+ ("holds" . "hold")
+ ("ignores" . "ignore")
+ ("indents" . "indent")
+ ("initializes" . "initialize")
+ ("inserts" . "insert")
+ ("installs" . "install")
+ ("investigates" . "investigate")
+ ("keeps" . "keep")
+ ("kills" . "kill")
+ ("leaves" . "leave")
+ ("lets" . "let")
+ ("loads" . "load")
+ ("looks" . "look")
+ ("makes" . "make")
+ ("marks" . "mark")
+ ;;("matches" . "match") ; Leads to almost only false positives.
+ ("moves" . "move")
+ ("notifies" . "notify")
+ ("offers" . "offer")
+ ("parses" . "parse")
+ ("performs" . "perform")
+ ("prepares" . "prepare")
+ ("prepends" . "prepend")
+ ("prompts" . "prompt")
+ ("reads" . "read")
+ ("raises" . "raise")
+ ("removes" . "remove")
+ ("replaces" . "replace")
+ ("resets" . "reset")
+ ("restores" . "restore")
+ ("returns" . "return")
+ ("runs" . "run")
+ ("saves" . "save")
+ ("says" . "say")
+ ("searches" . "search")
+ ("selects" . "select")
+ ("sets" . "set")
+ ("sex" . "s*x")
+ ("shows" . "show")
+ ("signifies" . "signify")
+ ("sorts" . "sort")
+ ("starts" . "start")
+ ("steps" . "step")
+ ("stores" . "store")
+ ("switches" . "switch")
+ ("tells" . "tell")
+ ("tests" . "test")
+ ("toggles" . "toggle")
+ ("tries" . "try")
+ ("turns" . "turn")
+ ("undoes" . "undo")
+ ("unloads" . "unload")
+ ("unmarks" . "unmark")
+ ("updates" . "update")
+ ("uses" . "use")
+ ("yanks" . "yank")
+ )
+ "Alist of common words in the wrong voice and what should be used instead.
+Set `checkdoc-verb-check-experimental-flag' to nil to avoid this costly
+and experimental check. Do not modify this list without setting
+the value of `checkdoc-common-verbs-regexp' to nil which cause it to
+be re-created.")
+
+(defvar checkdoc-syntax-table
+ (let ((st (make-syntax-table emacs-lisp-mode-syntax-table)))
+ ;; When dealing with syntax in doc strings, make sure that - are
+ ;; encompassed in words so we can use cheap \\> to get the end of a symbol,
+ ;; not the end of a word in a conglomerate.
+ (modify-syntax-entry ?- "w" st)
+ st)
+ "Syntax table used by checkdoc in document strings.")
+
+(defconst checkdoc--help-buffer "*Checkdoc Help*"
+ "Name of buffer used for Checkdoc Help.")
+
+(defvar checkdoc-commentary-header-string "\n;;; Commentary:\n;; \n\n"
+ "String inserted as commentary marker in `checkdoc-file-comments-engine'.")
+
+;;; User level commands
+;;
+;;;###autoload
+(defun checkdoc ()
+ "Interactively check the entire buffer for style errors.
+The current status of the check will be displayed in a buffer which
+the users will view as each check is completed."
+ (interactive nil emacs-lisp-mode)
+ (let ((status (list "Checking..." "-" "-" "-"))
+ (checkdoc-spellcheck-documentation-flag
+ (car (memq checkdoc-spellcheck-documentation-flag
+ '(buffer interactive t))))
+ ;; if the user set autofix to never, then that breaks the
+ ;; obviously requested asking implied by using this function.
+ ;; Set it to paranoia level.
+ (checkdoc-autofix-flag (if (or (not checkdoc-autofix-flag)
+ (eq checkdoc-autofix-flag 'never))
+ 'query
+ checkdoc-autofix-flag))
+ tmp)
+ (checkdoc-display-status-buffer status)
+ ;; check the comments
+ (setf (nth 0 status)
+ (cond
+ ((not buffer-file-name) "Not checked")
+ ((checkdoc-file-comments-engine) "Errors")
+ (t "Ok")))
+ (setf (nth 1 status) "Checking...")
+ (checkdoc-display-status-buffer status)
+ ;; Check the documentation
+ (setq tmp (checkdoc-interactive nil t))
+ (setf (nth 1 status)
+ (if tmp (format "%d Errors" (length tmp)) "Ok"))
+ (setf (nth 2 status) "Checking...")
+ (checkdoc-display-status-buffer status)
+ ;; Check the message text
+ (setf (nth 2 status)
+ (if (setq tmp (checkdoc-message-interactive nil t))
+ (format "%d Errors" (length tmp))
+ "Ok"))
+ (setf (nth 3 status) "Checking...")
+ (checkdoc-display-status-buffer status)
+ ;; Rogue spacing
+ (setf (nth 3 status)
+ (if (ignore-errors (checkdoc-rogue-spaces nil t))
+ "Errors"
+ "Ok"))
+ (checkdoc-display-status-buffer status)))
+
+(defun checkdoc-display-status-buffer (check)
+ "Display and update the status buffer for the current checkdoc mode.
+CHECK is a list of four strings stating the current status of each
+test; the nth string describes the status of the nth test."
+ (let (temp-buffer-setup-hook)
+ (with-output-to-temp-buffer "*Checkdoc Status*"
+ (mapc #'princ
+ (list "Buffer comments and tags: " (nth 0 check)
+ "\nDocumentation style: " (nth 1 check)
+ "\nMessage/Query text style: " (nth 2 check)
+ "\nUnwanted Spaces: " (nth 3 check)))))
+ (shrink-window-if-larger-than-buffer
+ (get-buffer-window "*Checkdoc Status*"))
+ (message nil)
+ (sit-for 0))
+
+;;;###autoload
+(defun checkdoc-interactive (&optional start-here showstatus)
+ "Interactively check the current buffer for doc string errors.
+Prefix argument START-HERE will start the checking from the current
+point, otherwise the check starts at the beginning of the current
+buffer. Allows navigation forward and backwards through document
+errors. Does not check for comment or space warnings.
+Optional argument SHOWSTATUS indicates that we should update the
+checkdoc status window instead of the usual behavior."
+ (interactive "P" emacs-lisp-mode)
+ (let ((checkdoc-spellcheck-documentation-flag
+ (car (memq checkdoc-spellcheck-documentation-flag
+ '(interactive t)))))
+ (prog1
+ ;; Due to a design flaw, this will never spell check
+ ;; docstrings.
+ (checkdoc-interactive-loop start-here showstatus
+ #'checkdoc-next-error)
+ ;; This is a workaround to perform spell checking.
+ (checkdoc-interactive-ispell-loop start-here))))
+
+;;;###autoload
+(defun checkdoc-message-interactive (&optional start-here showstatus)
+ "Interactively check the current buffer for message string errors.
+Prefix argument START-HERE will start the checking from the current
+point, otherwise the check starts at the beginning of the current
+buffer. Allows navigation forward and backwards through document
+errors. Does not check for comment or space warnings.
+Optional argument SHOWSTATUS indicates that we should update the
+checkdoc status window instead of the usual behavior."
+ (interactive "P" emacs-lisp-mode)
+ (let ((checkdoc-spellcheck-documentation-flag
+ (car (memq checkdoc-spellcheck-documentation-flag
+ '(interactive t)))))
+ (prog1
+ ;; Due to a design flaw, this will never spell check messages.
+ (checkdoc-interactive-loop start-here showstatus
+ #'checkdoc-next-message-error)
+ ;; This is a workaround to perform spell checking.
+ (checkdoc-message-interactive-ispell-loop start-here))))
+
+(defun checkdoc-interactive-loop (start-here showstatus findfunc)
+ "Interactively loop over all errors that can be found by a given method.
+
+If START-HERE is nil, searching starts at the beginning of the current
+buffer, otherwise searching starts at START-HERE. SHOWSTATUS
+expresses the verbosity of the search, and whether ending the search
+will auto-exit this function.
+
+FINDFUNC is a symbol representing a function that will position the
+cursor, and return error message text to present to the user. It is
+assumed that the cursor will stop just before a major sexp, which will
+be highlighted to present the user with feedback as to the offending
+style."
+ ;; Determine where to start the test
+ (let* ((begin (prog1 (point)
+ (if (not start-here) (goto-char (point-min)))))
+ ;; Assign a flag to spellcheck flag
+ (checkdoc-spellcheck-documentation-flag
+ (car (memq checkdoc-spellcheck-documentation-flag
+ '(buffer interactive t))))
+ ;; Fetch the error list
+ (err-list (list (funcall findfunc nil)))
+ (cdo nil)
+ (returnme nil)
+ c)
+ (save-window-excursion
+ (if (not (car err-list)) (setq err-list nil))
+ ;; Include whatever function point is in for good measure.
+ (beginning-of-defun)
+ (while err-list
+ (goto-char (cdr (car err-list)))
+ ;; The cursor should be just in front of the offending doc string
+ (setq cdo (if (stringp (car (car err-list)))
+ (save-excursion (make-overlay
+ (point) (progn (forward-sexp 1)
+ (point))))
+ (make-overlay
+ (checkdoc-error-start (car (car err-list)))
+ (checkdoc-error-end (car (car err-list))))))
+ (unwind-protect
+ (progn
+ (overlay-put cdo 'face 'highlight)
+ ;; Make sure the whole doc string is visible if possible.
+ (sit-for 0)
+ (if (and (= (following-char) ?\")
+ (not (pos-visible-in-window-p
+ (save-excursion (forward-sexp 1) (point))
+ (selected-window))))
+ (let ((l (count-lines (point)
+ (save-excursion
+ (forward-sexp 1) (point)))))
+ (if (> l (window-height))
+ (recenter 1)
+ (recenter (/ (- (window-height) l) 2))))
+ (recenter))
+ (message "%s (C-h,%se,n,p,q)" (checkdoc-error-text
+ (car (car err-list)))
+ (if (checkdoc-error-unfixable (car (car err-list)))
+ "" "f,"))
+ (save-excursion
+ (goto-char (checkdoc-error-start (car (car err-list))))
+ (if (not (pos-visible-in-window-p))
+ (recenter (- (window-height) 2)))
+ (setq c (read-event)))
+ (if (not (integerp c)) (setq c ??))
+ (cond
+ ;; Exit condition
+ ((eq c ?\C-g) (signal 'quit nil))
+ ;; Request an auto-fix
+ ((memq c '(?y ?f))
+ (delete-overlay cdo)
+ (setq cdo nil)
+ (goto-char (cdr (car err-list)))
+ ;; `automatic-then-never' tells the autofix function
+ ;; to only allow one fix to be automatic. The autofix
+ ;; function will then set the flag to `never', allowing
+ ;; the checker to return a different error.
+ (let ((checkdoc-autofix-flag 'automatic-then-never)
+ (fixed nil))
+ (funcall findfunc t)
+ (setq fixed (not (eq checkdoc-autofix-flag
+ 'automatic-then-never)))
+ (if (not fixed)
+ (progn
+ (message "A Fix was not available.")
+ (sit-for 2))
+ (setq err-list (cdr err-list))))
+ (beginning-of-defun)
+ (let ((ne (funcall findfunc nil)))
+ (if ne
+ (setq err-list (cons ne err-list))
+ (cond ((not err-list)
+ (message "No More Stylistic Errors.")
+ (sit-for 2))
+ (t
+ (message
+ "No Additional style errors. Continuing...")
+ (sit-for 2))))))
+ ;; Move to the next error (if available)
+ ((memq c '(?n ?\s))
+ (let ((ne (funcall findfunc nil)))
+ (if (not ne)
+ (if showstatus
+ (setq returnme err-list
+ err-list nil)
+ (if (not err-list)
+ (message "No More Stylistic Errors.")
+ (message "No Additional style errors. Continuing..."))
+ (sit-for 2))
+ (setq err-list (cons ne err-list)))))
+ ;; Go backwards in the list of errors
+ ((memq c '(?p ?\C-?))
+ (if (/= (length err-list) 1)
+ (progn
+ (setq err-list (cdr err-list))
+ (goto-char (cdr (car err-list)))
+ (beginning-of-defun))
+ (message "No Previous Errors.")
+ (sit-for 2)))
+ ;; Edit the buffer recursively.
+ ((eq c ?e)
+ (checkdoc-recursive-edit
+ (checkdoc-error-text (car (car err-list))))
+ (delete-overlay cdo)
+ (setq err-list (cdr err-list)) ;back up the error found.
+ (beginning-of-defun)
+ (let ((ne (funcall findfunc nil)))
+ (if (not ne)
+ (if showstatus
+ (setq returnme err-list
+ err-list nil)
+ (message "No More Stylistic Errors.")
+ (sit-for 2))
+ (setq err-list (cons ne err-list)))))
+ ;; Quit checkdoc
+ ((eq c ?q)
+ (setq returnme err-list
+ err-list nil
+ begin (point)))
+ ;; Goofy stuff
+ (t
+ (if (get-buffer-window checkdoc--help-buffer)
+ (progn
+ (delete-window (get-buffer-window checkdoc--help-buffer))
+ (kill-buffer checkdoc--help-buffer))
+ (with-output-to-temp-buffer checkdoc--help-buffer
+ (with-current-buffer standard-output
+ (insert
+ "Checkdoc Keyboard Summary:\n"
+ (if (checkdoc-error-unfixable (car (car err-list)))
+ ""
+ (concat
+ "f, y - auto Fix this warning without asking"
+ " (if available.)\n"
+ " Very complex operations will still query.\n"))
+ "e - Enter recursive Edit. Press C-M-c to exit.\n"
+ "SPC, n - skip to the Next error.\n"
+ "DEL, p - skip to the Previous error.\n"
+ "q - Quit checkdoc.\n"
+ "C-h - Toggle this help buffer.")))
+ (shrink-window-if-larger-than-buffer
+ (get-buffer-window checkdoc--help-buffer))))))
+ (if cdo (delete-overlay cdo)))))
+ (goto-char begin)
+ (if (get-buffer checkdoc--help-buffer) (kill-buffer checkdoc--help-buffer))
+ (message "Checkdoc: Done.")
+ returnme))
+
+(defun checkdoc-interactive-ispell-loop (start-here)
+ "Interactively spell check doc strings in the current buffer.
+If START-HERE is nil, searching starts at the beginning of the current
+buffer, otherwise searching starts at START-HERE."
+ (when checkdoc-spellcheck-documentation-flag
+ (save-excursion
+ ;; Move point to where we need to start.
+ (if start-here
+ ;; Include whatever function point is in for good measure.
+ (beginning-of-defun)
+ (goto-char (point-min)))
+ ;; Loop over docstrings.
+ (while (checkdoc-next-docstring)
+ (message "Searching for doc string spell error...%d%%"
+ (floor (* 100.0 (point)) (point-max)))
+ (when (= (following-char) ?\")
+ (checkdoc-ispell-docstring-engine
+ (save-excursion (forward-sexp 1) (point-marker)))))
+ (message "Checkdoc: Done."))))
+
+(defun checkdoc-message-interactive-ispell-loop (start-here)
+ "Interactively spell check messages in the current buffer.
+If START-HERE is nil, searching starts at the beginning of the current
+buffer, otherwise searching starts at START-HERE."
+ (when checkdoc-spellcheck-documentation-flag
+ (save-excursion
+ ;; Move point to where we need to start.
+ (if start-here
+ ;; Include whatever function point is in for good measure.
+ (beginning-of-defun)
+ (goto-char (point-min)))
+ ;; Loop over message strings.
+ (while (checkdoc-message-text-next-string (point-max))
+ (message "Searching for message string spell error...%d%%"
+ (floor (* 100.0 (point)) (point-max)))
+ (if (= (following-char) ?\")
+ (checkdoc-ispell-docstring-engine
+ (save-excursion (forward-sexp 1) (point-marker)))))
+ (message "Checkdoc: Done."))))
+
+
+(defun checkdoc-next-error (enable-fix)
+ "Find and return the next checkdoc error list, or nil.
+Only documentation strings are checked.
+An error list is of the form (WARNING . POSITION) where WARNING is the
+warning text, and POSITION is the point in the buffer where the error
+was found. We can use points and not markers because we promise not
+to edit the buffer before point without re-executing this check.
+Argument ENABLE-FIX will enable auto-fixing while looking for the next
+error. This argument assumes that the cursor is already positioned to
+perform the fix."
+ (if enable-fix
+ (checkdoc-this-string-valid)
+ (let ((msg nil) (p (point))
+ (checkdoc-autofix-flag nil))
+ (condition-case nil
+ (while (and (not msg) (checkdoc-next-docstring))
+ (message "Searching for doc string error...%d%%"
+ (floor (* 100.0 (point)) (point-max)))
+ (if (setq msg (checkdoc-this-string-valid))
+ (setq msg (cons msg (point)))))
+ ;; Quit.. restore position, Other errors, leave alone
+ (quit (goto-char p)))
+ msg)))
+
+(defun checkdoc-next-message-error (enable-fix)
+ "Find and return the next checkdoc message related error list, or nil.
+Only text for error and `y-or-n-p' strings are checked. See
+`checkdoc-next-error' for details on the return value.
+Argument ENABLE-FIX turns on the auto-fix feature. This argument
+assumes that the cursor is already positioned to perform the fix."
+ (if enable-fix
+ (checkdoc-message-text-engine)
+ (let ((msg nil) (p (point)) (type nil)
+ (checkdoc-autofix-flag nil))
+ (condition-case nil
+ (while (and (not msg)
+ (setq type
+ (checkdoc-message-text-next-string (point-max))))
+ (message "Searching for message string error...%d%%"
+ (floor (* 100.0 (point)) (point-max)))
+ (if (setq msg (checkdoc-message-text-engine type))
+ (setq msg (cons msg (point)))))
+ ;; Quit.. restore position, Other errors, leave alone
+ (quit (goto-char p)))
+ msg)))
+
+(defun checkdoc-recursive-edit (msg)
+ "Enter recursive edit to permit a user to fix some error checkdoc has found.
+MSG is the error that was found, which is displayed in a help buffer."
+ (with-output-to-temp-buffer checkdoc--help-buffer
+ (with-current-buffer standard-output
+ (insert "Error message:\n " msg "\n\n"
+ (substitute-command-keys
+ "Edit to fix this problem, and press \\[exit-recursive-edit] to continue."))))
+ (shrink-window-if-larger-than-buffer
+ (get-buffer-window checkdoc--help-buffer))
+ (message (substitute-command-keys
+ "When you're done editing press \\[exit-recursive-edit] to continue."))
+ (unwind-protect
+ (recursive-edit)
+ (if (get-buffer-window checkdoc--help-buffer)
+ (progn
+ (delete-window (get-buffer-window checkdoc--help-buffer))
+ (kill-buffer checkdoc--help-buffer)))))
+
+;;;###autoload
+(defun checkdoc-eval-current-buffer ()
+ "Evaluate and check documentation for the current buffer.
+Evaluation is done first because good documentation for something that
+doesn't work is just not useful. Comments, doc strings, and rogue
+spacing are all verified."
+ (interactive)
+ (eval-buffer nil)
+ (checkdoc-current-buffer t))
+
+;;;###autoload
+(defun checkdoc-current-buffer (&optional take-notes)
+ "Check current buffer for document, comment, error style, and rogue spaces.
+With a prefix argument (in Lisp, the argument TAKE-NOTES),
+store all errors found in a warnings buffer,
+otherwise stop after the first error."
+ (interactive "P" emacs-lisp-mode)
+ (if (called-interactively-p 'interactive)
+ (message "Checking buffer for style..."))
+ ;; Assign a flag to spellcheck flag
+ (let ((checkdoc-spellcheck-documentation-flag
+ (car (memq checkdoc-spellcheck-documentation-flag
+ '(buffer t))))
+ (checkdoc-autofix-flag (if take-notes 'never
+ checkdoc-autofix-flag))
+ (checkdoc-generate-compile-warnings-flag
+ (or take-notes checkdoc-generate-compile-warnings-flag)))
+ (if take-notes
+ (checkdoc-start-section "checkdoc-current-buffer"))
+ ;; every test is responsible for returning the cursor.
+ (or (and buffer-file-name ;; only check comments in a file
+ (checkdoc-comments))
+ (checkdoc-start take-notes)
+ (checkdoc-message-text)
+ (checkdoc-rogue-spaces)
+ (when checkdoc-package-keywords-flag
+ (checkdoc-package-keywords))
+ (not (called-interactively-p 'interactive))
+ (if take-notes (checkdoc-show-diagnostics))
+ (message "Checking buffer for style...Done."))))
+
+;;;###autoload
+(defun checkdoc-file (file)
+ "Check FILE for document, comment, error style, and rogue spaces."
+ (with-current-buffer (find-file-noselect file)
+ (let ((checkdoc-diagnostic-buffer "*warn*"))
+ (checkdoc-current-buffer t))))
+
+;;;###autoload
+(defun checkdoc-start (&optional take-notes)
+ "Start scanning the current buffer for documentation string style errors.
+Only documentation strings are checked.
+Use `checkdoc-continue' to continue checking if an error cannot be fixed.
+Prefix argument TAKE-NOTES means to collect all the warning messages into
+a separate buffer."
+ (interactive "P" emacs-lisp-mode)
+ (let ((p (point)))
+ (goto-char (point-min))
+ (if (and take-notes (called-interactively-p 'interactive))
+ (checkdoc-start-section "checkdoc-start"))
+ (checkdoc-continue take-notes)
+ ;; Go back since we can't be here without success above.
+ (goto-char p)
+ nil))
+
+;;;###autoload
+(defun checkdoc-continue (&optional take-notes)
+ "Find the next doc string in the current buffer which has a style error.
+Prefix argument TAKE-NOTES means to continue through the whole
+buffer and save warnings in a separate buffer."
+ (interactive "P" emacs-lisp-mode)
+ (let ((wrong nil) (msg nil)
+ ;; Assign a flag to spellcheck flag
+ (checkdoc-spellcheck-documentation-flag
+ (car (memq checkdoc-spellcheck-documentation-flag
+ '(buffer t))))
+ (checkdoc-autofix-flag (if take-notes 'never
+ checkdoc-autofix-flag))
+ (checkdoc-generate-compile-warnings-flag
+ (or take-notes checkdoc-generate-compile-warnings-flag)))
+ (save-excursion
+ ;; If we are taking notes, encompass the whole buffer, otherwise
+ ;; the user is navigating down through the buffer.
+ (while (and (not wrong) (checkdoc-next-docstring))
+ ;; OK, let's look at the doc string.
+ (setq msg (checkdoc-this-string-valid take-notes))
+ (if msg (setq wrong (point)))))
+ (if wrong
+ (progn
+ (goto-char wrong)
+ (if (not take-notes)
+ (user-error "%s" (checkdoc-error-text msg)))))
+ (checkdoc-show-diagnostics)
+ (if (called-interactively-p 'interactive)
+ (message "No style warnings."))))
+
+(defun checkdoc-next-docstring ()
+ "Move to the next doc string after point, and return t.
+Return nil if there are no more doc strings."
+ (let (found)
+ (while (and (not (setq found (checkdoc--next-docstring)))
+ (beginning-of-defun -1)))
+ found))
+
+(defun checkdoc--next-docstring ()
+ "When looking at a definition with a doc string, find it.
+Move to the next doc string after point, and return t. When not
+looking at a definition containing a doc string, return nil and
+don't move point."
+ (pcase (save-excursion (condition-case nil
+ (read (current-buffer))
+ ;; Conservatively skip syntax errors.
+ (invalid-read-syntax)
+ ;; Don't bug out if the file is empty (or a
+ ;; definition ends prematurely.
+ (end-of-file)))
+ (`(,(and (pred symbolp) def
+ (let (and doc (guard doc)) (function-get def 'doc-string-elt)))
+ ,(pred symbolp)
+ ;; Require an initializer, i.e. ignore single-argument `defvar'
+ ;; forms, which never have a doc string.
+ ,_ . ,_)
+ (down-list)
+ ;; Skip over function or macro name.
+ (forward-sexp 1)
+ ;; And now skip until the docstring.
+ (forward-sexp (1- ; We already skipped the function or macro name.
+ (cond
+ ((numberp doc) doc)
+ ((functionp doc) (funcall doc)))))
+ (skip-chars-forward " \n\t")
+ t)))
+
+;;;###autoload
+(defun checkdoc-comments (&optional take-notes)
+ "Find missing comment sections in the current Emacs Lisp file.
+Prefix argument TAKE-NOTES non-nil means to save warnings in a
+separate buffer. Otherwise print a message. This returns the error
+if there is one."
+ (interactive "P" emacs-lisp-mode)
+ (if take-notes (checkdoc-start-section "checkdoc-comments"))
+ (if (not buffer-file-name)
+ (error "Can only check comments for a file buffer"))
+ (let* ((checkdoc-spellcheck-documentation-flag
+ (car (memq checkdoc-spellcheck-documentation-flag
+ '(buffer t))))
+ (checkdoc-autofix-flag (if take-notes 'never checkdoc-autofix-flag))
+ (e (checkdoc-file-comments-engine))
+ (checkdoc-generate-compile-warnings-flag
+ (or take-notes checkdoc-generate-compile-warnings-flag)))
+ (if e (user-error "%s" (checkdoc-error-text e)))
+ (checkdoc-show-diagnostics)
+ e))
+
+;;;###autoload
+(defun checkdoc-rogue-spaces (&optional take-notes interact)
+ "Find extra spaces at the end of lines in the current file.
+Prefix argument TAKE-NOTES non-nil means to save warnings in a
+separate buffer. Otherwise print a message. This returns the error
+if there is one.
+Optional argument INTERACT permits more interactive fixing."
+ (interactive "P" emacs-lisp-mode)
+ (if take-notes (checkdoc-start-section "checkdoc-rogue-spaces"))
+ (let* ((checkdoc-autofix-flag (if take-notes 'never checkdoc-autofix-flag))
+ (e (checkdoc-rogue-space-check-engine nil nil interact))
+ (checkdoc-generate-compile-warnings-flag
+ (or take-notes checkdoc-generate-compile-warnings-flag)))
+ (if (not (called-interactively-p 'interactive))
+ e
+ (if e
+ (message "%s" (checkdoc-error-text e))
+ (checkdoc-show-diagnostics)
+ (message "Space Check: done.")))))
+
+;;;###autoload
+(defun checkdoc-message-text (&optional take-notes)
+ "Scan the buffer for occurrences of the error function, and verify text.
+Optional argument TAKE-NOTES causes all errors to be logged."
+ (interactive "P" emacs-lisp-mode)
+ (if take-notes (checkdoc-start-section "checkdoc-message-text"))
+ (let* ((p (point)) e
+ (checkdoc-autofix-flag (if take-notes 'never checkdoc-autofix-flag))
+ (checkdoc-generate-compile-warnings-flag
+ (or take-notes checkdoc-generate-compile-warnings-flag)))
+ (setq e (checkdoc-message-text-search))
+ (if (not (called-interactively-p 'interactive))
+ e
+ (if e
+ (user-error "%s" (checkdoc-error-text e))
+ (checkdoc-show-diagnostics)))
+ (goto-char p))
+ (if (called-interactively-p 'interactive)
+ (message "Checking interactive message text...done.")))
+
+;;;###autoload
+(defun checkdoc-eval-defun ()
+ "Evaluate the current form with `eval-defun' and check its documentation.
+Evaluation is done first so the form will be read before the
+documentation is checked. If there is a documentation error, then the display
+of what was evaluated will be overwritten by the diagnostic message."
+ (interactive)
+ (call-interactively #'eval-defun)
+ (checkdoc-defun))
+
+;;;###autoload
+(defun checkdoc-defun (&optional no-error)
+ "Examine the doc string of the function or variable under point.
+Call `error' if the doc string has problems. If NO-ERROR is
+non-nil, then do not call error, but call `message' instead.
+If the doc string passes the test, then check the function for rogue white
+space at the end of each line."
+ (interactive)
+ (save-excursion
+ (beginning-of-defun)
+ (when (checkdoc--next-docstring)
+ (let* ((checkdoc-spellcheck-documentation-flag
+ (car (memq checkdoc-spellcheck-documentation-flag
+ '(defun t))))
+ (beg (save-excursion (beginning-of-defun) (point)))
+ (end (save-excursion (end-of-defun) (point))))
+ (dolist (fun (list #'checkdoc-this-string-valid
+ (lambda () (checkdoc-message-text-search beg end))
+ (lambda () (checkdoc-rogue-space-check-engine beg end))))
+ (let ((msg (funcall fun)))
+ (if msg (if no-error
+ (message "%s" (checkdoc-error-text msg))
+ (user-error "%s" (checkdoc-error-text msg))))))
+ (if (called-interactively-p 'interactive)
+ (message "Checkdoc: done."))))))
+
+(defconst checkdoc--dired-skip-lines-re
+ (rx (or (seq bol
+ (or ";; Generated from Unicode data files by unidat"
+ ";; This file is automatically generated from"
+ ";; Generated by the command "))
+ ".el --- automatically extracted autoloads -*- lexical-binding: t -*-"
+ ";;; lisp/trampver.el. Generated from trampver.el.in by configure."))
+ "Regexp that when it matches tells `checkdoc-dired' to skip a file.")
+
+;;;###autoload
+(defun checkdoc-dired (files)
+ "In Dired, run `checkdoc' on marked files.
+Skip anything that doesn't have the Emacs Lisp library file
+extension (\".el\").
+When called from Lisp, FILES is a list of filenames."
+ (interactive
+ (progn
+ ;; These Dired functions must be defined since we're in a Dired buffer.
+ (declare-function dired-get-filename "dired"
+ (&optional localp no-error-if-not-filep))
+ ;; These functions are used by the expansion of `dired-map-over-marks'.
+ (declare-function dired-move-to-filename "dired"
+ (&optional raise-error eol))
+ (declare-function dired-marker-regexp "dired" ())
+ (list
+ (delq nil
+ (mapcar
+ ;; skip anything that doesn't look like an Emacs Lisp library
+ (lambda (f) (if (equal (file-name-extension f) "el") f nil))
+ (nreverse (dired-map-over-marks (dired-get-filename) nil))))))
+ dired-mode)
+ (if (null files)
+ (error "No files to run checkdoc on")
+ (save-window-excursion
+ (dolist (fil files)
+ (find-file fil)
+ (unless (and
+ (goto-char (point-min))
+ (re-search-forward checkdoc--dired-skip-lines-re nil t))
+ (checkdoc)))))
+ (message "checkdoc-dired: Successfully checked %d files" (length files)))
+
+;;; Ispell interface for forcing a spell check
+;;
+
+;;;###autoload
+(defun checkdoc-ispell ()
+ "Check the style and spelling of everything interactively.
+Calls `checkdoc' with spell-checking turned on.
+Prefix argument is the same as for `checkdoc'."
+ (interactive)
+ (let ((checkdoc-spellcheck-documentation-flag t))
+ (call-interactively #'checkdoc)))
+
+;;;###autoload
+(defun checkdoc-ispell-current-buffer ()
+ "Check the style and spelling of the current buffer.
+Calls `checkdoc-current-buffer' with spell-checking turned on.
+Prefix argument is the same as for `checkdoc-current-buffer'."
+ (interactive)
+ (let ((checkdoc-spellcheck-documentation-flag t))
+ (call-interactively #'checkdoc-current-buffer)))
+
+;;;###autoload
+(defun checkdoc-ispell-interactive ()
+ "Check the style and spelling of the current buffer interactively.
+Calls `checkdoc-interactive' with spell-checking turned on.
+Prefix argument is the same as for `checkdoc-interactive'."
+ (interactive)
+ (let ((checkdoc-spellcheck-documentation-flag t))
+ (call-interactively #'checkdoc-interactive)))
+
+;;;###autoload
+(defun checkdoc-ispell-message-interactive ()
+ "Check the style and spelling of message text interactively.
+Calls `checkdoc-message-interactive' with spell-checking turned on.
+Prefix argument is the same as for `checkdoc-message-interactive'."
+ (interactive)
+ (let ((checkdoc-spellcheck-documentation-flag t))
+ (call-interactively #'checkdoc-message-interactive
+ nil current-prefix-arg)))
+
+;;;###autoload
+(defun checkdoc-ispell-message-text ()
+ "Check the style and spelling of message text interactively.
+Calls `checkdoc-message-text' with spell-checking turned on.
+Prefix argument is the same as for `checkdoc-message-text'."
+ (interactive)
+ (let ((checkdoc-spellcheck-documentation-flag t))
+ (call-interactively #'checkdoc-message-text)))
+
+;;;###autoload
+(defun checkdoc-ispell-start ()
+ "Check the style and spelling of the current buffer.
+Calls `checkdoc-start' with spell-checking turned on.
+Prefix argument is the same as for `checkdoc-start'."
+ (interactive)
+ (let ((checkdoc-spellcheck-documentation-flag t))
+ (call-interactively #'checkdoc-start)))
+
+;;;###autoload
+(defun checkdoc-ispell-continue ()
+ "Check the style and spelling of the current buffer after point.
+Calls `checkdoc-continue' with spell-checking turned on.
+Prefix argument is the same as for `checkdoc-continue'."
+ (interactive)
+ (let ((checkdoc-spellcheck-documentation-flag t))
+ (call-interactively #'checkdoc-continue)))
+
+;;;###autoload
+(defun checkdoc-ispell-comments ()
+ "Check the style and spelling of the current buffer's comments.
+Calls `checkdoc-comments' with spell-checking turned on.
+Prefix argument is the same as for `checkdoc-comments'."
+ (interactive)
+ (let ((checkdoc-spellcheck-documentation-flag t))
+ (call-interactively #'checkdoc-comments)))
+
+;;;###autoload
+(defun checkdoc-ispell-defun ()
+ "Check the style and spelling of the current defun with Ispell.
+Calls `checkdoc-defun' with spell-checking turned on.
+Prefix argument is the same as for `checkdoc-defun'."
+ (interactive)
+ (let ((checkdoc-spellcheck-documentation-flag t))
+ (call-interactively #'checkdoc-defun)))
+
+;;; Error Management
+;;
+;; Errors returned from checkdoc functions can have various
+;; features and behaviors, so we need some ways of specifying
+;; them, and making them easier to use in the wacked-out interfaces
+;; people are requesting
+
+(cl-defstruct (checkdoc-error
+ (:constructor nil)
+ (:constructor checkdoc--create-error (text start end &optional unfixable)))
+ (text nil :read-only t)
+ (start nil :read-only t)
+ (end nil :read-only t)
+ (unfixable nil :read-only t))
+
+(defvar checkdoc-create-error-function #'checkdoc--create-error-for-checkdoc
+ "Function called when Checkdoc encounters an error.
+Should accept as arguments (TEXT START END &optional UNFIXABLE).
+
+TEXT is the descriptive text of the error. START and END define the region
+it is sensible to highlight when describing the problem.
+Optional argument UNFIXABLE means that the error has no auto-fix available.
+
+An object of type `checkdoc-error' is returned if we are not
+generating a buffered list of errors.")
+
+(defun checkdoc-create-error (text start end &optional unfixable)
+ "Used to create the return error text returned from all engines.
+TEXT, START, END and UNFIXABLE conform to
+`checkdoc-create-error-function', which see."
+ (funcall checkdoc-create-error-function text start end unfixable))
+
+(defun checkdoc--create-error-for-checkdoc (text start end &optional unfixable)
+ "Create an error for Checkdoc.
+TEXT, START, END and UNFIXABLE conform to
+`checkdoc-create-error-function', which see."
+ (if checkdoc-generate-compile-warnings-flag
+ (progn (checkdoc-error start text)
+ nil)
+ (checkdoc--create-error text start end unfixable)))
+
+;;; Minor Mode specification
+;;
+
+(defvar-keymap checkdoc-minor-mode-map
+ :doc "Keymap used to override evaluation key-bindings for documentation checking."
+ ;; Override some bindings
+ "C-M-x" #'checkdoc-eval-defun
+ "C-x `" #'checkdoc-continue
+ "<menu-bar> <emacs-lisp> <eval-buffer>" #'checkdoc-eval-current-buffer
+
+ ;; Add some new bindings under C-c ?
+ "C-c ? x" #'checkdoc-defun
+ "C-c ? X" #'checkdoc-ispell-defun
+ "C-c ? `" #'checkdoc-continue
+ "C-c ? ~" #'checkdoc-ispell-continue
+ "C-c ? s" #'checkdoc-start
+ "C-c ? S" #'checkdoc-ispell-start
+ "C-c ? d" #'checkdoc
+ "C-c ? D" #'checkdoc-ispell
+ "C-c ? b" #'checkdoc-current-buffer
+ "C-c ? B" #'checkdoc-ispell-current-buffer
+ "C-c ? e" #'checkdoc-eval-current-buffer
+ "C-c ? m" #'checkdoc-message-text
+ "C-c ? M" #'checkdoc-ispell-message-text
+ "C-c ? c" #'checkdoc-comments
+ "C-c ? C" #'checkdoc-ispell-comments
+ "C-c ? SPC" #'checkdoc-rogue-spaces)
+
+(easy-menu-define nil checkdoc-minor-mode-map
+ "Checkdoc Minor Mode Menu."
+ '("CheckDoc"
+ ["Interactive Buffer Style Check" checkdoc t]
+ ["Interactive Buffer Style and Spelling Check" checkdoc-ispell t]
+ ["Check Buffer" checkdoc-current-buffer t]
+ ["Check and Spell Buffer" checkdoc-ispell-current-buffer t]
+ "---"
+ ["Interactive Style Check" checkdoc-interactive t]
+ ["Interactive Style and Spelling Check" checkdoc-ispell-interactive t]
+ ["Find First Style Error" checkdoc-start t]
+ ["Find First Style or Spelling Error" checkdoc-ispell-start t]
+ ["Next Style Error" checkdoc-continue t]
+ ["Next Style or Spelling Error" checkdoc-ispell-continue t]
+ ["Interactive Message Text Style Check" checkdoc-message-interactive t]
+ ["Interactive Message Text Style and Spelling Check"
+ checkdoc-ispell-message-interactive t]
+ ["Check Message Text" checkdoc-message-text t]
+ ["Check and Spell Message Text" checkdoc-ispell-message-text t]
+ ["Check Comment Style" checkdoc-comments buffer-file-name]
+ ["Check Comment Style and Spelling" checkdoc-ispell-comments
+ buffer-file-name]
+ ["Check for Rogue Spaces" checkdoc-rogue-spaces t]
+ "---"
+ ["Check Defun" checkdoc-defun t]
+ ["Check and Spell Defun" checkdoc-ispell-defun t]
+ ["Check and Evaluate Defun" checkdoc-eval-defun t]
+ ["Check and Evaluate Buffer" checkdoc-eval-current-buffer t]))
+
+;;;###autoload
+(define-minor-mode checkdoc-minor-mode
+ "Toggle automatic docstring checking (Checkdoc minor mode).
+
+In Checkdoc minor mode, the usual bindings for `eval-defun' which is
+bound to \\<checkdoc-minor-mode-map>\\[checkdoc-eval-defun] and `checkdoc-eval-current-buffer' are overridden to include
+checking of documentation strings.
+
+\\{checkdoc-minor-mode-map}"
+ :lighter checkdoc-minor-mode-string
+ :group 'checkdoc)
+
+;;; Subst utils
+;;
+
+(defsubst checkdoc-create-common-verbs-regexp ()
+ "Rebuild the contents of `checkdoc-common-verbs-regexp'."
+ (or checkdoc-common-verbs-regexp
+ (setq checkdoc-common-verbs-regexp
+ (concat "\\<\\("
+ (mapconcat (lambda (e) (concat (car e)))
+ checkdoc-common-verbs-wrong-voice "\\|")
+ "\\)\\>"))))
+
+;;; Checking engines
+;;
+(defun checkdoc-this-string-valid (&optional take-notes)
+ "Return a message string if the current doc string is invalid.
+Check for style only, such as the first line always being a complete
+sentence, whitespace restrictions, and making sure there are no
+hard-coded key-codes such as C-[char] or mouse-[number] in the comment.
+See the style guide in the Emacs Lisp manual for more details.
+
+With a non-nil TAKE-NOTES, store all errors found in a warnings
+buffer, otherwise stop after the first error."
+
+ ;; Jump over comments between the last object and the doc string
+ (while (looking-at "[ \t\n]*;")
+ (forward-line 1)
+ (beginning-of-line)
+ (skip-chars-forward " \n\t"))
+
+ (let ((fp (checkdoc-defun-info))
+ (err nil))
+ (setq
+ err
+ ;; * Every command, function, or variable intended for users to know
+ ;; about should have a documentation string.
+ ;;
+ ;; * An internal variable or subroutine of a Lisp program might as well
+ ;; have a documentation string. In earlier Emacs versions, you could
+ ;; save space by using a comment instead of a documentation string,
+ ;; but that is no longer the case.
+ (if (and (not (nth 1 fp)) ; not a variable
+ (or (nth 2 fp) ; is interactive
+ checkdoc-force-docstrings-flag) ;or we always complain
+ (not (eq (following-char) ?\"))) ; no doc string
+ ;; Sometimes old code has comments where the documentation should
+ ;; be. Let's see if we can find the comment, and offer to turn it
+ ;; into documentation for them.
+ (let ((have-comment nil)
+ (comment-start ";")) ; in case it's not default
+ (condition-case nil
+ (progn
+ (forward-sexp -1)
+ (forward-sexp 1)
+ (skip-chars-forward "\n \t")
+ (setq have-comment (looking-at comment-start)))
+ (error nil))
+ (if have-comment
+ (if (or (eq checkdoc-autofix-flag
+ 'automatic-then-never)
+ (checkdoc-y-or-n-p
+ "Convert comment to documentation?"))
+ (save-excursion
+ ;; Our point is at the beginning of the comment!
+ ;; Insert a quote, then remove the comment chars.
+ (insert "\"")
+ (let ((docstring-start-point (point)))
+ (while (looking-at comment-start)
+ (while (looking-at comment-start)
+ (delete-char 1))
+ (if (looking-at "[ \t]+")
+ (delete-region (match-beginning 0) (match-end 0)))
+ (forward-line 1)
+ (beginning-of-line)
+ (skip-chars-forward " \t")
+ (if (looking-at comment-start)
+ (progn
+ (beginning-of-line)
+ (zap-to-char 1 ?\;))))
+ (beginning-of-line)
+ (forward-char -1)
+ (insert "\"")
+ (forward-char -1)
+ ;; quote any double-quote characters in the comment.
+ (while (search-backward "\"" docstring-start-point t)
+ (insert "\\"))
+ (if (eq checkdoc-autofix-flag 'automatic-then-never)
+ (setq checkdoc-autofix-flag 'never))))
+ (checkdoc-create-error
+ "You should convert this comment to documentation"
+ (point) (line-end-position)))
+ (when checkdoc--interactive-docstring-flag
+ (checkdoc-create-error
+ (if (nth 2 fp)
+ "All interactive functions should have documentation"
+ "All variables and subroutines might as well have a \
+documentation string")
+ (point) (+ (point) 1) t))))))
+ (if (and (not err) (= (following-char) ?\"))
+ (with-syntax-table checkdoc-syntax-table
+ (checkdoc-this-string-valid-engine fp take-notes))
+ err)))
+
+(defun checkdoc-this-string-valid-engine (fp &optional take-notes)
+ "Return an error list or string if the current doc string is invalid.
+Depends on `checkdoc-this-string-valid' to reset the syntax table so that
+regexp short cuts work. FP is the function defun information.
+
+With a non-nil TAKE-NOTES, store all errors found in a warnings
+buffer, otherwise stop after the first error."
+ (let ((case-fold-search nil)
+ ;; Use a marker so if an early check modifies the text,
+ ;; we won't accidentally lose our place. This could cause
+ ;; end-of doc string whitespace to also delete the " char.
+ (s (point))
+ (e (if (= (following-char) ?\")
+ (save-excursion (forward-sexp 1) (point-marker))
+ (point))))
+ (or
+ ;; * *Do not* indent subsequent lines of a documentation string so that
+ ;; the text is lined up in the source code with the text of the first
+ ;; line. This looks nice in the source code, but looks bizarre when
+ ;; users view the documentation. Remember that the indentation
+ ;; before the starting double-quote is not part of the string!
+ (save-excursion
+ (forward-line 1)
+ (beginning-of-line)
+ (if (and (< (point) e)
+ (looking-at "\\([ \t]+\\)[^ \t\n]"))
+ (if (checkdoc-autofix-ask-replace (match-beginning 1)
+ (match-end 1)
+ "Remove this whitespace?"
+ "")
+ nil
+ (checkdoc-create-error
+ "Second line should not have indentation"
+ (match-beginning 1)
+ (match-end 1)))))
+ ;; * Check for '(' in column 0.
+ (when checkdoc-column-zero-backslash-before-paren
+ (save-excursion
+ (when (re-search-forward "^(" e t)
+ (if (checkdoc-autofix-ask-replace (match-beginning 0)
+ (match-end 0)
+ (format-message "Escape this `('?")
+ "\\(")
+ nil
+ (checkdoc-create-error
+ "Open parenthesis in column 0 should be escaped"
+ (match-beginning 0) (match-end 0))))))
+ ;; * Do not start or end a documentation string with whitespace.
+ (let (start end)
+ (if (or (if (looking-at "\"\\([ \t\n]+\\)")
+ (setq start (match-beginning 1)
+ end (match-end 1)))
+ (save-excursion
+ (forward-sexp 1)
+ (forward-char -1)
+ (if (/= (skip-chars-backward " \t\n") 0)
+ (setq start (point)
+ end (1- e)))))
+ (if (checkdoc-autofix-ask-replace
+ start end "Remove this whitespace?" "")
+ nil
+ (checkdoc-create-error
+ "Documentation strings should not start or end with whitespace"
+ start end))))
+ ;; * The first line of the documentation string should consist of one
+ ;; or two complete sentences that stand on their own as a summary.
+ ;; `M-x apropos' displays just the first line, and if it doesn't
+ ;; stand on its own, the result looks bad. In particular, start the
+ ;; first line with a capital letter and end with a period.
+ (save-excursion
+ (end-of-line)
+ (skip-chars-backward " \t\n")
+ (if (> (point) e) (goto-char e)) ;of the form (defun n () "doc" nil)
+ (forward-char -1)
+ (cond
+ ((and (eq (following-char) ?\")
+ ;; A backslashed double quote at the end of a sentence
+ (not (eq (preceding-char) ?\\)))
+ ;; We might have to add a period in this case
+ (forward-char -1)
+ (if (looking-at "[.!?]")
+ nil
+ (forward-char 1)
+ (if (checkdoc-autofix-ask-replace
+ (point) (1+ (point)) "Add period to sentence?"
+ ".\"" t)
+ nil
+ (checkdoc-create-error
+ "First sentence should end with punctuation"
+ (point) (1+ (point))))))
+ ((looking-at "[\\!?;:.)]")
+ ;; These are ok
+ nil)
+ ((and checkdoc-permit-comma-termination-flag (= (following-char) ?,))
+ nil)
+ (t
+ ;; If it is not a complete sentence, let's see if we can
+ ;; predict a clever way to make it one.
+ (let ((msg "First line is not a complete sentence")
+ (e (point)))
+ (beginning-of-line)
+ (if (re-search-forward "\\. +" e t)
+ ;; Here we have found a complete sentence, but no break.
+ (if (checkdoc-autofix-ask-replace
+ (1+ (match-beginning 0)) (match-end 0)
+ "First line not a complete sentence. Add RET here?"
+ "\n" t)
+ (let (l1 l2)
+ (end-of-line 2)
+ (setq l1 (current-column)
+ l2 (save-excursion
+ (end-of-line 2)
+ (current-column)))
+ (if (> (+ l1 l2 1) 80)
+ (setq msg "Incomplete auto-fix; doc string \
+may require more formatting")
+ ;; We can merge these lines! Replace this CR
+ ;; with a space.
+ (delete-char 1) (insert " ")
+ (setq msg nil))))
+ ;; Let's see if there is enough room to draw the next
+ ;; line's sentence up here. I often get hit w/
+ ;; auto-fill moving my words around.
+ (let ((numc (progn (end-of-line) (- 80 (current-column))))
+ (p (point)))
+ (forward-line 1)
+ (beginning-of-line)
+ (if (and (re-search-forward "[.!?:\"]\\([ \t\n]+\\|\"\\)"
+ (line-end-position) t)
+ (< (current-column) numc))
+ (when (checkdoc-autofix-ask-replace
+ p (1+ p)
+ "First line not a complete sentence. Join these lines?"
+ " " t)
+ (setq msg nil)))))
+ (if msg
+ (checkdoc-create-error msg s (save-excursion
+ (goto-char s)
+ (line-end-position))))))))
+ ;; Continuation of above. Make sure our sentence is capitalized.
+ (save-excursion
+ (skip-chars-forward "\"*")
+ (if (looking-at "[a-z]")
+ (if (checkdoc-autofix-ask-replace
+ (match-beginning 0) (match-end 0)
+ "Capitalize your sentence?" (upcase (match-string 0))
+ t)
+ nil
+ (checkdoc-create-error
+ "First line should be capitalized"
+ (match-beginning 0) (match-end 0)))
+ nil))
+ ;; * Don't write key sequences directly in documentation strings.
+ ;; Instead, use the `\\[...]' construct to stand for them.
+ (save-excursion
+ (let ((f nil) (m nil) (start (point))
+ ;; Ignore the "A-" modifier: it is uncommon in practice,
+ ;; and leads to false positives in regexp ranges.
+ (re "[^`‘A-Za-z0-9_]\\([CMs]-[a-zA-Z]\\|\\(\\([CMs]-\\)?\
+mouse-[0-3]\\)\\)\\>"))
+ ;; Find the first key sequence not in a sample
+ (while (and (not f) (setq m (re-search-forward re e t)))
+ (setq f (not (checkdoc-in-sample-code-p start e))))
+ (if m
+ (checkdoc-create-error
+ (concat
+ "Keycode " (match-string 1)
+ " embedded in doc string. Use \\\\<mapvar> & \\\\[command] "
+ "instead")
+ (match-beginning 1) (match-end 1) t))))
+ ;; Optionally warn about too many command substitutions.
+ (when checkdoc-max-keyref-before-warn
+ (save-excursion
+ (if (and (re-search-forward "\\\\\\\\\\[\\w+" e t
+ (1+ checkdoc-max-keyref-before-warn))
+ (not (re-search-forward "\\\\\\\\{\\w+}" e t)))
+ (checkdoc-create-error
+ "Too many occurrences of \\[command]. Use \\{mapvar} instead"
+ s (marker-position e)))))
+ ;; Ambiguous quoted symbol. When a symbol is both bound and fbound,
+ ;; and is referred to in documentation, it should be prefixed with
+ ;; something to disambiguate it. This check must be before the
+ ;; 80 column check because it might break that.
+ (save-excursion
+ (let ((case-fold-search t)
+ (ret nil) mb me)
+ (while (and (re-search-forward
+ "[`‘]\\(\\sw\\(\\sw\\|\\s_\\)+\\)['’]" e t)
+ (not ret))
+ (let* ((ms1 (match-string 1))
+ (sym (intern-soft ms1)))
+ (setq mb (match-beginning 1)
+ me (match-end 1))
+ (if (and sym (boundp sym) (fboundp sym)
+ checkdoc--disambiguate-symbol-flag
+ ;; Mode names do not need disambiguating. (Bug#4110)
+ (not (string-match (rx "-mode" string-end)
+ (symbol-name sym)))
+ (save-excursion
+ (goto-char mb)
+ (forward-word-strictly -1)
+ (not (looking-at
+ "variable\\|option\\|function\\|command\\|symbol"))))
+ (if (checkdoc-autofix-ask-replace
+ mb me "Prefix this ambiguous symbol?" ms1 t)
+ ;; We didn't actually replace anything. Here we find
+ ;; out what special word form they wish to use as
+ ;; a prefix.
+ (let ((disambiguate
+ (completing-read
+ (format-prompt "Disambiguating Keyword"
+ "variable")
+ '(("function") ("command") ("variable")
+ ("option") ("symbol"))
+ nil t nil nil "variable")))
+ (goto-char (1- mb))
+ (insert disambiguate " ")
+ (forward-word-strictly 1))
+ (setq ret
+ (format "Disambiguate %s by preceding w/ \
+function,command,variable,option or symbol." ms1))))))
+ (if ret
+ (checkdoc-create-error ret mb me)
+ nil)))
+ ;; * Format the documentation string so that it fits in an
+ ;; Emacs window on an 80-column screen. It is a good idea
+ ;; for most lines to be no wider than 60 characters. The
+ ;; first line can be wider if necessary to fit the
+ ;; information that ought to be there.
+ (save-excursion
+ (let* ((start (point))
+ (eol nil)
+ ;; Respect this file local variable.
+ (max-column (max 80 byte-compile-docstring-max-column))
+ ;; Allow the first line to be three characters longer, to
+ ;; fit the leading ` "' while still having a docstring
+ ;; shorter than e.g. 80 characters.
+ (first t)
+ (get-max-column (lambda () (+ max-column (if first 3 0)))))
+ (while (and (< (point) e)
+ (or (progn (end-of-line) (setq eol (point))
+ (< (current-column) (funcall get-max-column)))
+ (progn (beginning-of-line)
+ (re-search-forward "\\\\\\\\[[<{]"
+ eol t))
+ (checkdoc-in-sample-code-p start e)))
+ (setq first nil)
+ (forward-line 1))
+ (end-of-line)
+ (if (and (< (point) e) (> (current-column) (funcall get-max-column)))
+ (checkdoc-create-error
+ (format "Some lines are over %d columns wide" max-column)
+ s (save-excursion (goto-char s) (line-end-position))))))
+ ;; Here we deviate to tests based on a variable or function.
+ ;; We must do this before checking for symbols in quotes because there
+ ;; is a chance that just such a symbol might really be an argument.
+ (cond ((eq (nth 1 fp) t)
+ ;; This is if we are in a variable
+ (or
+ ;; * The documentation string for a variable that is a
+ ;; yes-or-no flag should start with words such as Non-nil
+ ;; means..., to make it clear that all non-nil values are
+ ;; equivalent and indicate explicitly what nil and non-nil
+ ;; mean.
+ ;; * If a user option variable records a true-or-false
+ ;; condition, give it a name that ends in `-flag'.
+
+ ;; "True ..." should be "Non-nil ..."
+ (when (looking-at "\"\\*?\\(True\\)\\b")
+ (if (checkdoc-autofix-ask-replace
+ (match-beginning 1) (match-end 1)
+ "Say \"Non-nil\" instead of \"True\"?"
+ "Non-nil")
+ nil
+ (checkdoc-create-error
+ "\"True\" should usually be \"Non-nil\""
+ (match-beginning 1) (match-end 1))))
+
+ ;; If the variable has -flag in the name, make sure
+ (if (and (string-match "-flag$" (car fp))
+ (not (looking-at "\"\\*?Non-nil\\s-+means\\s-+")))
+ (checkdoc-create-error
+ "Flag variable doc strings should usually start: Non-nil means"
+ s (marker-position e) t))
+ ;; Don't rename variable to "foo-flag". This is unnecessary
+ ;; and such names often end up inconvenient when the variable
+ ;; is later expanded to non-boolean values. --Stef
+ ;; If the doc string starts with "Non-nil means"
+ ;; (if (and (looking-at "\"\\*?Non-nil\\s-+means\\s-+")
+ ;; (not (string-match "-flag$" (car fp))))
+ ;; (let ((newname
+ ;; (if (string-match "-p$" (car fp))
+ ;; (concat (substring (car fp) 0 -2) "-flag")
+ ;; (concat (car fp) "-flag"))))
+ ;; (if (checkdoc-y-or-n-p
+ ;; (format
+ ;; "Rename to %s and Query-Replace all occurrences?"
+ ;; newname))
+ ;; (progn
+ ;; (beginning-of-defun)
+ ;; (query-replace-regexp
+ ;; (concat "\\<" (regexp-quote (car fp)) "\\>")
+ ;; newname))
+ ;; (checkdoc-create-error
+ ;; "Flag variable names should normally end in `-flag'" s
+ ;; (marker-position e)))))
+ ;; Done with variables
+ ))
+ (t
+ ;; This if we are in a function definition
+ (or
+ ;; * When a function's documentation string mentions the value
+ ;; of an argument of the function, use the argument name in
+ ;; capital letters as if it were a name for that value. Thus,
+ ;; the documentation string of the function `/' refers to its
+ ;; second argument as `DIVISOR', because the actual argument
+ ;; name is `divisor'.
+
+ ;; Addendum: Make sure they appear in the doc in the same
+ ;; order that they are found in the arg list.
+ (let ((args (nthcdr 4 fp))
+ (last-pos 0)
+ (found 1)
+ (order (and (nth 3 fp) (car (nth 3 fp))))
+ (nocheck (append '("&optional" "&rest" "&key" "&aux"
+ "&context" "&environment" "&whole"
+ "&body" "&allow-other-keys")
+ (nth 3 fp)))
+ (inopts nil))
+ (while (and args found (> found last-pos))
+ (if (or (member (car args) nocheck)
+ (string-match "\\`_" (car args)))
+ (setq args (cdr args)
+ inopts t)
+ (setq last-pos found
+ found (save-excursion
+ (re-search-forward
+ (concat "\\<" (upcase (car args))
+ ;; Require whitespace OR
+ ;; ITEMth<space> OR
+ ;; ITEMs<space>
+ "\\(\\>\\|th\\>\\|s\\>\\|[.,;:]\\)")
+ e t)))
+ (if (not found)
+ (let ((case-fold-search t))
+ ;; If the symbol was not found, let's see if we
+ ;; can find it with a different capitalization
+ ;; and see if the user wants to capitalize it.
+ (if (save-excursion
+ (re-search-forward
+ (concat "\\<\\(" (car args)
+ ;; Require whitespace OR
+ ;; ITEMth<space> OR
+ ;; ITEMs<space>
+ "\\)\\(\\>\\|th\\>\\|s\\>\\)")
+ e t))
+ (if (checkdoc-autofix-ask-replace
+ (match-beginning 1) (match-end 1)
+ (format-message
+ "If this is the argument `%s', it should appear as %s. Fix?"
+ (car args) (upcase (car args)))
+ (upcase (car args)) t)
+ (setq found (match-beginning 1))))))
+ (if found (setq args (cdr args)))))
+ (if (not found)
+ ;; It wasn't found at all! Offer to attach this new symbol
+ ;; to the end of the documentation string.
+ (if (checkdoc-y-or-n-p
+ (format
+ "Add %s documentation to end of doc string?"
+ (upcase (car args))))
+ ;; Now do some magic and invent a doc string.
+ (save-excursion
+ (goto-char e) (forward-char -1)
+ (insert "\n"
+ (if inopts "Optional a" "A")
+ "rgument " (upcase (car args))
+ " ")
+ (insert (read-string "Describe: "))
+ (if (not (save-excursion (forward-char -1)
+ (looking-at "[.?!]")))
+ (insert "."))
+ nil)
+ (when checkdoc--argument-missing-flag
+ (checkdoc-create-error
+ (format-message
+ "Argument `%s' should appear (as %s) in the doc string"
+ (car args) (upcase (car args)))
+ s (marker-position e))))
+ (if (or (and order (eq order 'yes))
+ (and (not order) checkdoc-arguments-in-order-flag))
+ (if (< found last-pos)
+ (checkdoc-create-error
+ "Arguments occur in the doc string out of order"
+ s (marker-position e) t)))))
+ ;; * For consistency, phrase the verb in the first sentence of a
+ ;; documentation string for functions as an imperative.
+ ;; For instance, use `Return the cons of A and
+ ;; B.' in preference to `Returns the cons of A and B.'
+ ;; Usually it looks good to do likewise for the rest of the
+ ;; first paragraph. Subsequent paragraphs usually look better
+ ;; if they have proper subjects.
+ ;;
+ ;; This is the least important of the above tests. Make sure
+ ;; it occurs last.
+ (and checkdoc-verb-check-experimental-flag
+ (save-excursion
+ ;; Maybe rebuild the monster-regexp
+ (checkdoc-create-common-verbs-regexp)
+ (let ((lim (save-excursion
+ (end-of-line)
+ ;; check string-continuation
+ (if (eq (preceding-char) ?\\)
+ (line-end-position 2)
+ (point))))
+ (rs nil) replace original (case-fold-search t))
+ (while (and (not rs)
+ (re-search-forward
+ checkdoc-common-verbs-regexp
+ lim t))
+ (setq original (buffer-substring-no-properties
+ (match-beginning 1) (match-end 1))
+ rs (assoc (downcase original)
+ checkdoc-common-verbs-wrong-voice))
+ (if (not rs) (error "Verb voice alist corrupted"))
+ (setq replace (let ((case-fold-search nil))
+ (if (string-match-p "^[A-Z]" original)
+ (capitalize (cdr rs))
+ (cdr rs))))
+ (if (checkdoc-autofix-ask-replace
+ (match-beginning 1) (match-end 1)
+ (format "Use the imperative for \"%s\". \
+Replace with \"%s\"?" original replace)
+ replace t)
+ (setq rs nil)))
+ (if rs
+ ;; there was a match, but no replace
+ (checkdoc-create-error
+ (format
+ "Probably \"%s\" should be imperative \"%s\""
+ original replace)
+ (match-beginning 1) (match-end 1))))))
+ ;; "Return true ..." should be "Return non-nil ..."
+ (when (looking-at "\"Return \\(true\\)\\b")
+ (if (checkdoc-autofix-ask-replace
+ (match-beginning 1) (match-end 1)
+ "Say \"non-nil\" instead of \"true\"?"
+ "non-nil")
+ nil
+ (checkdoc-create-error
+ "\"true\" should usually be \"non-nil\""
+ (match-beginning 1) (match-end 1))))
+ ;; Done with functions
+ )))
+ ;;* When a documentation string refers to a Lisp symbol, write it as
+ ;; it would be printed (which usually means in lower case), with
+ ;; single-quotes around it. For example: ‘lambda’. There are two
+ ;; exceptions: write t and nil without single-quotes. (For
+ ;; compatibility with an older Emacs style, quoting with ` and '
+ ;; also works, e.g., `lambda' is treated like ‘lambda’.)
+ (save-excursion
+ (let ((found nil) (start (point)) (msg nil) (ms nil))
+ (while (and (not msg)
+ (re-search-forward
+ ;; Ignore manual page references like
+ ;; git-config(1).
+ "[^-([`'‘’:a-zA-Z]\\(\\w+[:-]\\(\\w\\|\\s_\\)+\\)[^]('’]"
+ e t))
+ (setq ms (match-string 1))
+ ;; A . is a \s_ char, so we must remove periods from
+ ;; sentences more carefully.
+ (when (string-match-p "\\.$" ms)
+ (setq ms (substring ms 0 (1- (length ms)))))
+ (if (and (not (checkdoc-in-sample-code-p start e))
+ (not (checkdoc-in-example-string-p start e))
+ (not (member ms checkdoc-symbol-words))
+ (setq found (intern-soft ms))
+ (or (boundp found) (fboundp found)))
+ (progn
+ (setq msg (format-message
+ "Add quotes around Lisp symbol `%s'?" ms))
+ (if (checkdoc-autofix-ask-replace
+ (match-beginning 1) (+ (match-beginning 1)
+ (length ms))
+ msg (format "`%s'" ms) t)
+ (setq msg nil)
+ (setq msg
+ (format-message
+ "Lisp symbol `%s' should appear in quotes" ms))))))
+ (if msg
+ (checkdoc-create-error msg (match-beginning 1)
+ (+ (match-beginning 1)
+ (length ms)))
+ nil)))
+ ;; t and nil case
+ (save-excursion
+ (if (re-search-forward "\\([`‘]\\(t\\|nil\\)['’]\\)" e t)
+ (if (checkdoc-autofix-ask-replace
+ (match-beginning 1) (match-end 1)
+ (format "%s should not appear in quotes. Remove?"
+ (match-string 2))
+ (match-string 2) t)
+ nil
+ (checkdoc-create-error
+ "Symbols t and nil should not appear in single quotes"
+ (match-beginning 1) (match-end 1)))))
+ ;; Here is some basic sentence formatting
+ (checkdoc-sentencespace-region-engine (point) e)
+ ;; Here are common proper nouns that should always appear capitalized.
+ (checkdoc-proper-noun-region-engine (point) e)
+ ;; Make sure the doc string has correctly spelled English words
+ ;; in it. This function is extracted due to its complexity,
+ ;; and reliance on the Ispell program.
+ (checkdoc-ispell-docstring-engine e take-notes)
+ ;; User supplied checks
+ (save-excursion (run-hook-with-args-until-success 'checkdoc-style-functions fp e)))))
+
+(defun checkdoc-defun-info nil
+ "Return a list of details about the current sexp.
+It is a list of the form:
+ (NAME VARIABLE INTERACTIVE NODOCPARAMS PARAMETERS ...)
+where NAME is the name, VARIABLE is t if this is a `defvar',
+INTERACTIVE is nil if this is not an interactive function, otherwise
+it is the position of the `interactive' call, and PARAMETERS is a
+string which is the name of each variable in the function's argument
+list. The NODOCPARAMS is a sublist of parameters specified by a checkdoc
+comment for a given defun. If the first element is not a string, then
+the token checkdoc-order: <TOKEN> exists, and TOKEN is a symbol read
+from the comment."
+ (save-excursion
+ (beginning-of-defun)
+ (let ((defun (looking-at
+ "(\\(?:cl-\\)?def\\(un\\|macro\\|subst\\|advice\\|generic\\|method\\)"))
+ (is-advice (looking-at "(defadvice"))
+ (defun-depth (ppss-depth (syntax-ppss)))
+ (lst nil)
+ (ret nil)
+ (oo (make-vector 3 0))) ;substitute obarray for `read'
+ (forward-char 1)
+ (forward-sexp 1)
+ (skip-chars-forward " \n\t")
+ (setq ret
+ (list (buffer-substring-no-properties
+ (point) (progn (forward-sexp 1) (point)))))
+ (if (not defun)
+ (setq ret (cons t ret))
+ ;; The variable spot
+ (setq ret (cons nil ret))
+ ;; Interactive
+ (save-excursion
+ (push (and (re-search-forward "^\\s-*(interactive"
+ (save-excursion
+ (end-of-defun)
+ (point))
+ t)
+ ;; Disregard `interactive' from other parts of
+ ;; the function.
+ (= (ppss-depth (syntax-ppss))
+ (+ defun-depth 2))
+ (point))
+ ret))
+ (skip-chars-forward " \t\n")
+ (let ((bss (buffer-substring (point) (save-excursion (forward-sexp 1)
+ (point))))
+ ;; Overload th main obarray so read doesn't intern the
+ ;; local symbols of the function we are checking.
+ ;; Without this we end up cluttering the symbol space w/
+ ;; useless symbols.
+ (obarray oo))
+ ;; Ok, check for checkdoc parameter comment here
+ (save-excursion
+ (setq ret
+ (cons
+ (let ((sl1 nil))
+ (if (re-search-forward ";\\s-+checkdoc-order:\\s-+"
+ (save-excursion (end-of-defun)
+ (point))
+ t)
+ (setq sl1 (list (cond ((looking-at "nil") 'no)
+ ((looking-at "t") 'yes)))))
+ (if (re-search-forward ";\\s-+checkdoc-params:\\s-+"
+ (save-excursion (end-of-defun)
+ (point))
+ t)
+ (let ((sl nil))
+ (goto-char (match-end 0))
+ (condition-case nil
+ (setq lst (read (current-buffer)))
+ (error (setq lst nil))) ; error in text
+ (if (not (listp lst)) ; not a list of args
+ (setq lst (list lst)))
+ (if (and lst (not (symbolp (car lst)))) ;weird arg
+ (setq lst nil))
+ (while lst
+ (setq sl (cons (symbol-name (car lst)) sl)
+ lst (cdr lst)))
+ (setq sl1 (append sl1 sl))))
+ sl1)
+ ret)))
+ ;; Read the list of parameters, but do not put the symbols in
+ ;; the standard obarray.
+ (setq lst (read bss)))
+ ;; This is because read will intern nil if it doesn't into the
+ ;; new obarray.
+ (if (not (listp lst)) (setq lst nil))
+ (unless is-advice
+ ;; (car lst) can be something like ((foo bar) baz) from
+ ;; cl-lib methods; flatten it:
+ (while lst
+ (setq ret (cons (symbol-name (car (flatten-tree (car lst)))) ret)
+ lst (cdr lst)))))
+ (nreverse ret))))
+
+(defun checkdoc-in-sample-code-p (start limit)
+ "Return non-nil if the current point is in a code fragment.
+A code fragment is identified by an open parenthesis followed by a
+symbol which is a valid function or a word in all CAPS, or a parenthesis
+that is quoted with the \\=' character. Only the region from START to LIMIT
+is allowed while searching for the bounding parenthesis."
+ (save-match-data
+ (save-restriction
+ (narrow-to-region start limit)
+ (save-excursion
+ (and (condition-case nil (progn (up-list 1) t) (error nil))
+ (condition-case nil (progn (forward-list -1) t) (error nil))
+ (or (save-excursion (forward-char -1) (looking-at "'("))
+ (and (looking-at "(\\(\\(\\w\\|[-:_]\\)+\\)[ \t\n;]")
+ (let ((ms (buffer-substring-no-properties
+ (match-beginning 1) (match-end 1))))
+ ;; if this string is function bound, we are in
+ ;; sample code. If it has a - or : character in
+ ;; the name, then it is probably supposed to be bound
+ ;; but isn't yet.
+ (or (fboundp (intern-soft ms))
+ (let ((case-fold-search nil))
+ (string-match "^[A-Z-]+$" ms))
+ (string-match "\\w[-:_]+\\w" ms))))))))))
+
+(defun checkdoc-in-example-string-p (start limit)
+ "Return non-nil if the current point is in an \"example string\".
+This string is identified by the characters \\\" surrounding the text.
+The text checked is between START and LIMIT."
+ (save-match-data
+ (save-excursion
+ (let ((p (point))
+ (c 0))
+ (goto-char start)
+ (while (and (< (point) p) (re-search-forward "\\\\\"" limit t))
+ (setq c (1+ c)))
+ (and (< 0 c) (= (% c 2) 0))))))
+
+(defun checkdoc-in-abbreviation-p (begin)
+ "Return non-nil if point is at an abbreviation.
+Examples of recognized abbreviations: \"e.g.\", \"i.e.\", \"cf.\"."
+ (save-excursion
+ (goto-char begin)
+ (condition-case nil
+ (let (single-letter)
+ (forward-word -1)
+ ;; Skip over all dots backwards, as `forward-word' will only
+ ;; go one dot at a time in a string like "e.g.".
+ (while (save-excursion (forward-char -1)
+ (looking-at (rx ".")))
+ (forward-word -1))
+ (when (= (point) (1- begin))
+ (setq single-letter t))
+ ;; Piece of an abbreviation.
+ (looking-at
+ (if single-letter
+ ;; Handle a single letter, as in "a.", as this might be
+ ;; a part of a list.
+ (rx letter ".")
+ (rx (or
+ ;; The abbreviations (a trailing dot is added below).
+ (seq (any "cC") "f") ; cf.
+ (seq (any "eE") ".g") ; e.g.
+ (seq (any "iI") "." (any "eE")) ; i.e.
+ "a.k.a" "etc" "vs" "N.B"
+ ;; Some non-standard or less common ones that we
+ ;; might as well accept.
+ "Inc" "Univ" "misc" "resp")
+ "."))))
+ (error t))))
+
+(defun checkdoc-proper-noun-region-engine (begin end)
+ "Check all text between BEGIN and END for lower case proper nouns.
+These are Emacs centric proper nouns which should be capitalized for
+consistency. Return an error list if any are not fixed, but
+internally skip over no answers.
+If the offending word is in a piece of quoted text, then it is skipped."
+ (save-excursion
+ (let ((case-fold-search nil)
+ (errtxt nil) bb be)
+ (with-syntax-table checkdoc-syntax-table
+ (goto-char begin)
+ (while (re-search-forward checkdoc-proper-noun-regexp end t)
+ (let ((text (match-string 1))
+ (b (match-beginning 1))
+ (e (match-end 1)))
+ (if (and (not (save-excursion
+ (goto-char b)
+ (forward-char -1)
+ (looking-at "[`\".‘]\\|\\\\")))
+ ;; surrounded by /, as in a URL or filename: /emacs/
+ (not (and (= ?/ (char-after e))
+ (= ?/ (char-before b))))
+ (not (checkdoc-in-example-string-p begin end))
+ ;; info or url links left alone
+ (not (thing-at-point-looking-at
+ help-xref-info-regexp))
+ (not (thing-at-point-looking-at
+ help-xref-url-regexp)))
+ (if (checkdoc-autofix-ask-replace
+ b e (format "Text %s should be capitalized. Fix?"
+ text)
+ (capitalize text) t)
+ nil
+ (if errtxt
+ ;; If there is already an error, then generate
+ ;; the warning output if applicable
+ (if checkdoc-generate-compile-warnings-flag
+ (checkdoc-create-error
+ (format
+ "Name %s should appear capitalized as %s"
+ text (capitalize text))
+ b e))
+ (setq errtxt
+ (format
+ "Name %s should appear capitalized as %s"
+ text (capitalize text))
+ bb b be e)))))))
+ (if errtxt (checkdoc-create-error errtxt bb be)))))
+
+(defun checkdoc-sentencespace-region-engine (begin end)
+ "Make sure all sentences have double spaces between BEGIN and END."
+ (if sentence-end-double-space
+ (save-excursion
+ (let ((case-fold-search nil)
+ (errtxt nil) bb be)
+ (with-syntax-table checkdoc-syntax-table
+ (goto-char begin)
+ (while (re-search-forward "[^ .0-9]\\(\\. \\)[^ \n]" end t)
+ (let ((b (match-beginning 1))
+ (e (match-end 1)))
+ (unless (or (checkdoc-in-sample-code-p begin end)
+ (checkdoc-in-example-string-p begin end)
+ (checkdoc-in-abbreviation-p b))
+ (if (checkdoc-autofix-ask-replace
+ b e
+ "There should be two spaces after a period. Fix?"
+ ". ")
+ nil
+ (if errtxt
+ ;; If there is already an error, then generate
+ ;; the warning output if applicable
+ (if checkdoc-generate-compile-warnings-flag
+ (checkdoc-create-error
+ "There should be two spaces after a period"
+ b e))
+ (setq errtxt
+ "There should be two spaces after a period"
+ bb b be e)))))))
+ (if errtxt (checkdoc-create-error errtxt bb be))))))
+
+;;; Ispell engine
+;;
+(defvar ispell-process)
+(declare-function ispell-buffer-local-words "ispell" ())
+(declare-function ispell-correct-p "ispell" (&optional following))
+(declare-function ispell-set-spellchecker-params "ispell" ())
+(declare-function ispell-accept-buffer-local-defs "ispell" ())
+(declare-function ispell-error-checking-word "ispell" (word))
+
+(defun checkdoc-ispell-init ()
+ "Initialize Ispell process (default version) with Lisp words.
+The words used are from `checkdoc-ispell-lisp-words'. If `ispell'
+cannot be loaded, then set `checkdoc-spellcheck-documentation-flag' to
+nil."
+ (require 'ispell)
+ (unless ispell-process
+ (condition-case nil
+ (progn
+ (ispell-set-spellchecker-params) ; Initialize variables and dict alists.
+ (ispell-accept-buffer-local-defs) ; Use the correct dictionary.
+ (dolist (w checkdoc-ispell-lisp-words)
+ (process-send-string ispell-process (concat "@" w "\n"))))
+ (error (setq checkdoc-spellcheck-documentation-flag nil)))))
+
+(defun checkdoc-ispell-docstring-engine (end &optional take-notes)
+ "Run the Ispell tools on the doc string between point and END.
+Since Ispell isn't Lisp-smart, we must pre-process the doc string
+before using the Ispell engine on it.
+
+With a non-nil TAKE-NOTES, store all errors found in a warnings
+buffer, otherwise stop after the first error."
+ (when (and checkdoc-spellcheck-documentation-flag
+ ;; If the user wants no questions or fixing, then we must
+ ;; disable spell checking as not useful.
+ (or take-notes
+ (and checkdoc-autofix-flag
+ (not (eq checkdoc-autofix-flag 'never)))))
+ (checkdoc-ispell-init)
+ (unless checkdoc-spellcheck-documentation-flag
+ ;; this happens when (checkdoc-ispell-init) can't start `ispell-program-name'
+ (user-error "No spellchecker installed: check the variable `ispell-program-name'"))
+ (save-excursion
+ (skip-chars-forward "^a-zA-Z")
+ (let (word sym case-fold-search word-beginning word-end) ;; err
+ (while (and (< (point) end)) ;; (not err)
+ (if (save-excursion (forward-char -1) (looking-at "[('`]"))
+ ;; Skip lists describing meta-syntax, or bound variables
+ (forward-sexp 1)
+ (setq word-beginning (point)
+ word-end (progn
+ (skip-chars-forward "a-zA-Z-")
+ (point))
+ word (buffer-substring-no-properties word-beginning word-end)
+ sym (intern-soft word))
+ (unless (and sym (or (boundp sym) (fboundp sym)))
+ ;; Find out how we spell-check this word.
+ (unless (or
+ ;; All caps w/ option th, or s tacked on the end
+ ;; for pluralization or number.
+ (string-match "^[A-Z][A-Z]+\\(s\\|th\\)?$" word)
+ (looking-at "}") ; a keymap expression
+ )
+ (save-excursion
+ (let ((lk last-input-event))
+ (if take-notes
+ (progn
+ (unless (ispell-correct-p)
+ (checkdoc-create-error
+ (ispell-error-checking-word word)
+ word-beginning word-end)))
+ (ispell-word nil t))
+ (if (not (equal last-input-event lk))
+ (progn
+ (sit-for 0)
+ (message "Continuing..."))))))))
+ (skip-chars-forward "^a-zA-Z"))
+ nil)))) ;; err
+
+;;; Rogue space checking engine
+;;
+(defun checkdoc-rogue-space-check-engine (&optional start end interact)
+ "Return a message list if there is a line with white space at the end.
+If `checkdoc-autofix-flag' permits, delete that whitespace instead.
+If optional arguments START and END are non-nil, bound the check to
+this region.
+Optional argument INTERACT may permit the user to fix problems on the fly."
+ (let ((p (point))
+ (msg nil) s e (f nil))
+ (if (not start) (setq start (point-min)))
+ ;; If end is nil, it means end of buffer to search anyway
+ (or
+ ;; Check for an error if `? ' or `?\ ' is used at the end of a line.
+ ;; (It's dangerous)
+ (progn
+ (goto-char start)
+ (while (and (not msg) (re-search-forward "\\?\\\\?[ \t][ \t]*$" end t))
+ (setq msg
+ "Don't use `? ' at the end of a line. \
+News agents may remove it"
+ s (match-beginning 0) e (match-end 0) f t)
+ ;; If interactive is passed down, give them a chance to fix things.
+ (if (and interact (y-or-n-p (concat msg ". Fix?")))
+ (progn
+ (checkdoc-recursive-edit msg)
+ (setq msg nil)
+ (goto-char s)
+ (beginning-of-line)))))
+ ;; Check for, and potentially remove whitespace appearing at the
+ ;; end of different lines.
+ (progn
+ (goto-char start)
+ ;; There is no documentation in the Emacs Lisp manual about this check,
+ ;; it is intended to help clean up messy code and reduce the file size.
+ (while (and (not msg) (re-search-forward "[^ \t\n;]\\([ \t]+\\)$" end t))
+ ;; This is not a complex activity
+ (if (checkdoc-autofix-ask-replace
+ (match-beginning 1) (match-end 1)
+ "White space at end of line. Remove?" "")
+ nil
+ (setq msg "White space found at end of line"
+ s (match-beginning 1) e (match-end 1))))))
+ ;; Return an error and leave the cursor at that spot, or restore
+ ;; the cursor.
+ (if msg
+ (checkdoc-create-error msg s e f)
+ (goto-char p)
+ nil)))
+
+;;; Comment checking engine
+;;
+(defun checkdoc-file-comments-engine ()
+ "Return a message list if this file does not match the Emacs standard.
+This checks for style only, such as the first line, Commentary:,
+Code:, and others referenced in the style guide."
+ (save-excursion
+ (let* ((f1 (file-name-nondirectory (buffer-file-name)))
+ (fn (file-name-sans-extension f1))
+ (fe (substring f1 (length fn)))
+ (err nil))
+ (goto-char (point-min))
+ ;; This file has been set up where ERR is a variable. Each check is
+ ;; asked, and the function will make sure that if the user does not
+ ;; auto-fix some error, that we still move on to the next auto-fix,
+ ;; AND we remember the past errors.
+ (setq
+ err
+ ;; Lisp Maintenance checks first
+ ;; Was: (lm-verify) -> not flexible enough for some people
+ ;; * Summary at the beginning of the file:
+ (if (not (lm-summary))
+ ;; This certifies as very complex so always ask unless
+ ;; it's set to never
+ (if (checkdoc-y-or-n-p "There is no first line summary! Add one?")
+ (progn
+ (goto-char (point-min))
+ (insert ";;; " fn fe " --- " (read-string "Summary: ") "\n"))
+ (checkdoc-create-error
+ "The first line should be of the form: \";;; package --- Summary\""
+ (point-min) (save-excursion (goto-char (point-min))
+ (line-end-position))))
+ nil))
+ (setq
+ err
+ (or
+ ;; * Commentary Section
+ (if (and (not (lm-commentary-mark))
+ ;; No need for a commentary section in test files.
+ (not (string-match
+ (rx (or (seq (or "-test.el" "-tests.el") string-end)
+ "/test/" "/tests/"))
+ (buffer-file-name))))
+ (progn
+ (goto-char (point-min))
+ (cond
+ ((re-search-forward
+ "write\\s-+to\\s-+the\\s-+Free Software Foundation, Inc."
+ nil t)
+ (re-search-forward "^;;\\s-*\n\\|^\n" nil t))
+ ((or (re-search-forward "^;;; History" nil t)
+ (re-search-forward "^;;; Code" nil t)
+ (re-search-forward "^(require" nil t)
+ (re-search-forward "^(" nil t))
+ (beginning-of-line))
+ ((not (re-search-forward ";;; .* --- .*\n" nil t))
+ (checkdoc-create-error
+ "You should have a summary line (\";;; .* --- .*\")"
+ nil nil t)))
+ (if (checkdoc-y-or-n-p
+ "You should have a \";;; Commentary:\", add one?")
+ (insert checkdoc-commentary-header-string)
+ (checkdoc-create-error
+ "You should have a section marked \";;; Commentary:\""
+ nil nil t)))
+ nil)
+ err))
+ (setq
+ err
+ (or
+ ;; * History section. Say nothing if there is a file ChangeLog
+ (if (or (not checkdoc-force-history-flag)
+ (file-exists-p "ChangeLog")
+ (file-exists-p "../ChangeLog")
+ (lm-history-mark))
+ nil
+ (progn
+ (goto-char (or (lm-commentary-mark) (point-min)))
+ (cond
+ ((re-search-forward
+ "write\\s-+to\\s-+the\\s-+Free Software Foundation, Inc."
+ nil t)
+ (re-search-forward "^;;\\s-*\n\\|^\n" nil t))
+ ((or (re-search-forward "^;;; Code" nil t)
+ (re-search-forward "^(require" nil t)
+ (re-search-forward "^(" nil t))
+ (beginning-of-line)))
+ (if (checkdoc-y-or-n-p
+ "You should have a \";;; History:\", add one?")
+ (insert "\n;;; History:\n;; \n\n")
+ (checkdoc-create-error
+ "You should have a section marked \";;; History:\" or use a ChangeLog"
+ (point) nil))))
+ err))
+ (setq
+ err
+ (or
+ ;; * Code section
+ (if (not (lm-code-mark))
+ (let ((cont t)
+ pos)
+ (goto-char (point-min))
+ ;; match ";;;###autoload" cookie to keep it with the form
+ (while (and cont (re-search-forward
+ (concat "^\\(" lisp-mode-autoload-regexp
+ "\n\\)?"
+ "(")
+ nil t))
+ (setq pos (match-beginning 0)
+ cont (looking-at "require\\s-+")))
+ (if (and (not cont)
+ (checkdoc-y-or-n-p
+ "There is no ;;; Code: marker. Insert one?"))
+ (progn (goto-char pos)
+ (insert ";;; Code:\n\n")
+ nil)
+ (checkdoc-create-error
+ "You should have a section marked \";;; Code:\""
+ (point) nil)))
+ nil)
+ err))
+ (setq
+ err
+ (or
+ ;; * A footer. Not compartmentalized from lm-verify: too bad.
+ ;; The following is partially clipped from lm-verify
+ (save-excursion
+ (goto-char (point-max))
+ (if (not (re-search-backward
+ ;; This should match the requirement in
+ ;; `package-buffer-info'.
+ (concat "^;;; " (regexp-quote (concat fn fe)) " ends here")
+ nil t))
+ (if (checkdoc-y-or-n-p "No identifiable footer! Add one?")
+ (progn
+ (goto-char (point-max))
+ (insert "\n(provide '" fn ")\n\n;;; " fn fe " ends here\n"))
+ (checkdoc-create-error
+ (format "The footer should be: (provide '%s)\\n;;; %s%s ends here"
+ fn fn fe)
+ ;; The buffer may be empty.
+ (max (point-min) (1- (point-max)))
+ (point-max)))))
+ err))
+ ;; The below checks will not return errors if the user says NO
+
+ ;; Let's spellcheck the commentary section. This is the only
+ ;; section that is easy to pick out, and it is also the most
+ ;; visible section (with the finder).
+ (let ((cm (lm-commentary-mark)))
+ (when cm
+ (save-excursion
+ (goto-char cm)
+ (let ((e (copy-marker (lm-commentary-end))))
+ ;; Since the comments talk about Lisp, use the
+ ;; specialized spell-checker we also used for doc
+ ;; strings.
+ (checkdoc-sentencespace-region-engine (point) e)
+ (checkdoc-proper-noun-region-engine (point) e)
+ (checkdoc-ispell-docstring-engine e)))))
+ (setq
+ err
+ (or
+ ;; Generic Full-file checks (should be comment related)
+ (run-hook-with-args-until-success 'checkdoc-comment-style-functions)
+ err))
+ ;; Done with full file comment checks
+ err)))
+
+(defun checkdoc-outside-major-sexp ()
+ "Return t if point is outside the bounds of a valid sexp."
+ (save-match-data
+ (save-excursion
+ (let ((p (point)))
+ (or (progn (beginning-of-defun) (bobp))
+ (progn (end-of-defun) (< (point) p)))))))
+
+;;; `error' and `message' text verifier.
+;;
+(defun checkdoc-message-text-search (&optional beg end)
+ "Search between BEG and END for a style error with message text.
+Optional arguments BEG and END represent the boundary of the check.
+The default boundary is the entire buffer."
+ (let ((e nil)
+ (type nil))
+ (if (not (or beg end)) (setq beg (point-min) end (point-max)))
+ (goto-char beg)
+ (while (setq type (checkdoc-message-text-next-string end))
+ (setq e (checkdoc-message-text-engine type)))
+ e))
+
+(defun checkdoc-message-text-next-string (end)
+ "Move cursor to the next checkable message string after point.
+Return the message classification.
+Argument END is the maximum bounds to search in."
+ (let ((return nil))
+ (while (and (not return)
+ (re-search-forward
+ (rx "("
+ (* (syntax whitespace))
+ (group
+ (or (seq (* (group (or wordchar (syntax symbol))))
+ "error")
+ (seq (* (group (or wordchar (syntax symbol))))
+ (or "y-or-n-p" "yes-or-no-p")
+ (? (group "-with-timeout")))
+ "checkdoc-autofix-ask-replace"))
+ (+ (any "\n\t ")))
+ end t))
+ (let* ((fn (match-string 1))
+ (type (cond ((string-match "error" fn)
+ 'error)
+ (t 'y-or-n-p))))
+ (if (string-match "checkdoc-autofix-ask-replace" fn)
+ (progn (forward-sexp 2)
+ (skip-chars-forward " \t\n")))
+ (if (and (eq type 'y-or-n-p)
+ (looking-at (rx "(format" (? "-message") (+ (any " \t\n")))))
+ (goto-char (match-end 0)))
+ (skip-chars-forward " \t\n")
+ (if (not (looking-at "\""))
+ nil
+ (setq return type))))
+ return))
+
+(defun checkdoc--error-bad-format-p ()
+ "Return non-nil if the start of error message at point has the wrong format.
+The correct format is \"Foo\" or \"some-symbol: Foo\". See also
+`error' and Info node `(elisp) Documentation Tips'."
+ (save-excursion
+ ;; Skip the first quote character in string.
+ (forward-char 1)
+ ;; A capital letter is always okay.
+ (unless (let ((case-fold-search nil))
+ (looking-at (rx (or upper-case "%s"))))
+ ;; A defined Lisp symbol is always okay.
+ (unless (and (looking-at (rx (group lisp-mode-symbol)))
+ (or (fboundp (intern (match-string 1)))
+ (boundp (intern (match-string 1)))))
+ ;; Other Lisp symbols are sometimes okay.
+ (rx-let ((c (? "\\\n"))) ; `c' is for a continued line
+ (let ((case-fold-search nil)
+ (some-symbol (rx lisp-mode-symbol
+ c ":" c (+ (any " \t\n"))))
+ (lowercase-str (rx c (group (any "a-z") (+ wordchar)))))
+ (if (looking-at some-symbol)
+ (looking-at (concat some-symbol lowercase-str))
+ (looking-at lowercase-str))))))))
+
+(defun checkdoc--fix-y-or-n-p ()
+ "Fix `y-or-n-p' prompt to end with \"?\" or \"? \".
+The space is technically redundant, but also more compatible with
+Emacs versions before Emacs 24.1. In the future, we might treat
+a space as a style error."
+ (when (and (save-excursion (forward-sexp 1)
+ (forward-char -3)
+ (not (looking-at "\\? ")))
+ (save-excursion (forward-sexp 1)
+ (forward-char -2)
+ (not (looking-at "\\?"))))
+ (if (and
+ (save-excursion (forward-sexp 1)
+ (forward-char -1)
+ (looking-at "\""))
+ (checkdoc-autofix-ask-replace
+ (match-beginning 0) (match-end 0)
+ (format-message
+ "`y-or-n-p' argument should end with \"?\". Fix?")
+ "?\"" t))
+ nil
+ (checkdoc-create-error
+ "`y-or-n-p' argument should end with \"?\""
+ (match-beginning 0) (match-end 0)))))
+
+(defun checkdoc-message-text-engine (&optional type)
+ "Return or fix errors found in strings passed to a message display function.
+According to the documentation for the function `error', the error list
+should not end with a period, and should start with a capital letter.
+The function `y-or-n-p' has similar constraints.
+Argument TYPE specifies the type of question, such as `error' or `y-or-n-p'."
+ ;; If type is nil, then attempt to derive it.
+ (if (not type)
+ (save-excursion
+ (up-list -1)
+ (if (looking-at "(format")
+ (up-list -1))
+ (setq type
+ (cond ((looking-at "(error")
+ 'error)
+ (t 'y-or-n-p)))))
+ (let ((case-fold-search nil))
+ (or
+ ;; From the documentation of the symbol `error':
+ ;; In Emacs, the convention is that error messages start with a capital
+ ;; letter but *do not* end with a period. Please follow this convention
+ ;; for the sake of consistency.
+ (if (and (checkdoc--error-bad-format-p)
+ (not (checkdoc-autofix-ask-replace
+ (match-beginning 1) (match-end 1)
+ "Capitalize your message text?"
+ (capitalize (match-string 1))
+ t)))
+ (checkdoc-create-error "Messages should start with a capital letter"
+ (match-beginning 1) (match-end 1))
+ nil)
+ ;; In general, sentences should have two spaces after the period.
+ (checkdoc-sentencespace-region-engine (point)
+ (save-excursion (forward-sexp 1)
+ (point)))
+ ;; Look for proper nouns in this region too.
+ (checkdoc-proper-noun-region-engine (point)
+ (save-excursion (forward-sexp 1)
+ (point)))
+ ;; Here are message type specific questions.
+ (if (and (eq type 'error)
+ (save-excursion (forward-sexp 1)
+ (forward-char -2)
+ (looking-at "\\."))
+ (not (checkdoc-autofix-ask-replace (match-beginning 0)
+ (match-end 0)
+ "Remove period from error?"
+ ""
+ t)))
+ (checkdoc-create-error
+ "Error messages should *not* end with a period"
+ (match-beginning 0) (match-end 0))
+ nil)
+ ;; From `(elisp) Programming Tips': "A question asked in the
+ ;; minibuffer with `yes-or-no-p' or `y-or-n-p' should start with
+ ;; a capital letter and end with '?'."
+ (when (eq type 'y-or-n-p)
+ (checkdoc--fix-y-or-n-p))
+ ;; Now, let's just run the spell checker on this guy.
+ (checkdoc-ispell-docstring-engine (save-excursion (forward-sexp 1)
+ (point))))))
+
+;;; Auto-fix helper functions
+;;
+(defun checkdoc-y-or-n-p (question)
+ "Like `y-or-n-p', but pays attention to `checkdoc-autofix-flag'.
+Argument QUESTION is the prompt passed to `y-or-n-p'."
+ (prog1
+ (if (or (not checkdoc-autofix-flag)
+ (eq checkdoc-autofix-flag 'never))
+ nil
+ (y-or-n-p question))
+ (if (eq checkdoc-autofix-flag 'automatic-then-never)
+ (setq checkdoc-autofix-flag 'never))))
+
+(defun checkdoc-autofix-ask-replace (start end question replacewith
+ &optional complex)
+ "Highlight between START and END and queries the user with QUESTION.
+If the user says yes, or if `checkdoc-autofix-flag' permits, replace
+the region marked by START and END with REPLACEWITH. If optional flag
+COMPLEX is non-nil, then we may ask the user a question. See the
+documentation for `checkdoc-autofix-flag' for details.
+
+If a section is auto-replaced without asking the user, this function
+will pause near the fixed code so the user will briefly see what
+happened.
+
+This function returns non-nil if the text was replaced.
+
+This function will not modify `match-data'."
+ (if (and checkdoc-autofix-flag
+ (not (eq checkdoc-autofix-flag 'never)))
+ (let ((o (make-overlay start end))
+ (ret nil)
+ (md (match-data)))
+ (unwind-protect
+ (progn
+ (overlay-put o 'face 'highlight)
+ (if (or (eq checkdoc-autofix-flag 'automatic)
+ (eq checkdoc-autofix-flag 'automatic-then-never)
+ (and (eq checkdoc-autofix-flag 'semiautomatic)
+ (not complex))
+ (and (or (eq checkdoc-autofix-flag 'query) complex)
+ (y-or-n-p question)))
+ (save-excursion
+ (goto-char start)
+ ;; On the off chance this is automatic, display
+ ;; the question anyway so the user knows what's
+ ;; going on.
+ (if checkdoc-bouncy-flag (message "%s -> done" question))
+ (delete-region start end)
+ (insert-before-markers replacewith)
+ (if checkdoc-bouncy-flag (sit-for 0))
+ (setq ret t)))
+ (delete-overlay o)
+ (set-match-data md))
+ (delete-overlay o)
+ (set-match-data md))
+ (if (eq checkdoc-autofix-flag 'automatic-then-never)
+ (setq checkdoc-autofix-flag 'never))
+ ret)))
+
+;;; Warning management
+;;
+(defvar checkdoc-output-font-lock-keywords
+ '(("^\\*\\*\\* \\(.+\\.el\\): \\([^ \n]+\\)"
+ (1 font-lock-function-name-face)
+ (2 font-lock-comment-face)))
+ "Keywords used to highlight a checkdoc diagnostic buffer.")
+
+(defvar checkdoc-output-error-regex-alist
+ '(("^\\(.+\\.el\\):\\([0-9]+\\): " 1 2)))
+
+(defvar checkdoc-pending-errors nil
+ "Non-nil when there are errors that have not been displayed yet.")
+
+(define-derived-mode checkdoc-output-mode compilation-mode "Checkdoc"
+ "Set up the major mode for the buffer containing the list of errors."
+ (setq-local compilation-error-regexp-alist
+ checkdoc-output-error-regex-alist)
+ (setq-local compilation-mode-font-lock-keywords
+ checkdoc-output-font-lock-keywords))
+
+(defun checkdoc-buffer-label ()
+ "The name to use for a checkdoc buffer in the error list."
+ (if (buffer-file-name)
+ (file-relative-name (buffer-file-name))
+ (concat "#<buffer "(buffer-name) ">")))
+
+(defun checkdoc-start-section (check-type)
+ "Initialize the checkdoc diagnostic buffer for a pass.
+Create the header so that the string CHECK-TYPE is displayed as the
+function called to create the messages."
+ (let ((dir default-directory)
+ (label (checkdoc-buffer-label)))
+ (with-current-buffer (get-buffer-create checkdoc-diagnostic-buffer)
+ (checkdoc-output-mode)
+ (setq default-directory dir)
+ (goto-char (point-max))
+ (let ((inhibit-read-only t))
+ (insert "\n\n\C-l\n*** " label ": "
+ check-type)))))
+
+(defun checkdoc-error (point msg)
+ "Store POINT and MSG as errors in the checkdoc diagnostic buffer."
+ (setq checkdoc-pending-errors t)
+ (let ((text (list "\n" (checkdoc-buffer-label) ":"
+ (int-to-string
+ (count-lines (point-min) (or point (point-min))))
+ ": " msg)))
+ (if (string= checkdoc-diagnostic-buffer "*warn*")
+ (warn (apply #'concat text))
+ (with-current-buffer (get-buffer checkdoc-diagnostic-buffer)
+ (let ((inhibit-read-only t)
+ (pt (point-max)))
+ (goto-char pt)
+ (apply #'insert text))))))
+
+(defun checkdoc-show-diagnostics ()
+ "Display the checkdoc diagnostic buffer in a temporary window."
+ (if checkdoc-pending-errors
+ (let* ((b (get-buffer checkdoc-diagnostic-buffer))
+ (win (if b (display-buffer b))))
+ (when win
+ (with-selected-window win
+ (goto-char (point-max))
+ (re-search-backward "\C-l" nil t)
+ (beginning-of-line)
+ (forward-line 1)
+ (recenter 0)))
+ (setq checkdoc-pending-errors nil)
+ nil)))
+
+(defun checkdoc-get-keywords ()
+ "Return a list of package keywords for the current file."
+ (save-excursion
+ (goto-char (point-min))
+ (when (re-search-forward "^;; Keywords: \\(.*\\)$" nil t)
+ (split-string (match-string-no-properties 1) ", " t))))
+
+(defvar finder-known-keywords)
+
+;;;###autoload
+(defun checkdoc-package-keywords ()
+ "Find package keywords that aren't in `finder-known-keywords'."
+ (interactive)
+ (require 'finder)
+ (let ((unrecognized-keys
+ (cl-remove-if
+ (lambda (x) (assoc (intern-soft x) finder-known-keywords))
+ (checkdoc-get-keywords))))
+ (if unrecognized-keys
+ (let* ((checkdoc-autofix-flag 'never)
+ (checkdoc-generate-compile-warnings-flag t))
+ (save-excursion
+ (goto-char (point-min))
+ (re-search-forward "^;; Keywords: \\(.*\\)$" nil t)
+ (checkdoc-start-section "checkdoc-package-keywords")
+ (checkdoc-create-error
+ (concat "Unrecognized keywords: "
+ (mapconcat #'identity unrecognized-keys ", "))
+ (match-beginning 1) (match-end 1)))
+ (checkdoc-show-diagnostics))
+ (when (called-interactively-p 'any)
+ (message "No Package Keyword Errors.")))))
+
+(custom-add-option 'emacs-lisp-mode-hook 'checkdoc-minor-mode)
+
+(define-obsolete-function-alias 'checkdoc-run-hooks
+ #'run-hook-with-args-until-success "28.1")
+(defvar checkdoc-version "0.6.2"
+ "Release version of checkdoc you are currently running.")
+(make-obsolete-variable 'checkdoc-version 'emacs-version "28.1")
+
+(provide 'checkdoc)
+
+;;; checkdoc.el ends here