123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554 |
- ;;; cus-test.el --- tests for custom types and load problems
- ;; Copyright (C) 1998, 2000, 2002-2015 Free Software Foundation, Inc.
- ;; Author: Markus Rost <markus.rost@mathematik.uni-regensburg.de>
- ;; Maintainer: Markus Rost <rost@math.ohio-state.edu>
- ;; Created: 13 Sep 1998
- ;; Keywords: maint
- ;; 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 <http://www.gnu.org/licenses/>.
- ;;; Commentary:
- ;; This file provides simple tests to detect custom options with
- ;; incorrect customization types and load problems for custom and
- ;; autoload dependencies.
- ;;
- ;; The basic tests can be run in batch mode. Invoke them with
- ;;
- ;; src/emacs -batch -l admin/cus-test.el -f cus-test-opts [all]
- ;;
- ;; src/emacs -batch -l admin/cus-test.el -f cus-test-deps
- ;;
- ;; src/emacs -batch -l admin/cus-test.el -f cus-test-libs [all]
- ;;
- ;; src/emacs -batch -l admin/cus-test.el -f cus-test-noloads
- ;;
- ;; in the emacs source directory.
- ;;
- ;; For interactive use: Load this file. Then
- ;;
- ;; M-x cus-test-apropos REGEXP RET
- ;;
- ;; checks the options matching REGEXP. In particular
- ;;
- ;; M-x cus-test-apropos RET
- ;;
- ;; checks all options. The detected options are stored in the
- ;; variable `cus-test-errors'.
- ;;
- ;; Only those options are checked which have been already loaded.
- ;; Therefore `cus-test-apropos' is more efficient after loading many
- ;; libraries.
- ;;
- ;; M-x cus-test-load-custom-loads
- ;;
- ;; loads all (!) custom dependencies and
- ;;
- ;; M-x cus-test-load-libs
- ;;
- ;; loads all (!) libraries with autoloads.
- ;;
- ;; Options with a custom-get property, usually defined by a :get
- ;; declaration, are stored in the variable
- ;;
- ;; `cus-test-vars-with-custom-get'
- ;;
- ;; Options with a state of 'changed ("changed outside the customize
- ;; buffer") are stored in the variable
- ;;
- ;; `cus-test-vars-with-changed-state'
- ;;
- ;; These lists are prepared just in case one wants to investigate
- ;; those options further.
- ;;
- ;; The command `cus-test-opts' tests many (all?) custom options.
- ;;
- ;; The command `cus-test-deps' is like `cus-test-load-custom-loads'
- ;; but reports about load errors.
- ;;
- ;; The command `cus-test-libs' runs for all libraries with autoloads
- ;; separate emacs processes of the form "emacs -batch -l LIB".
- ;;
- ;; The command `cus-test-noloads' returns a list of variables which
- ;; are somewhere declared as custom options, but not loaded by
- ;; `custom-load-symbol'.
- ;;; Code:
- ;;; Workarounds. For a smooth run and to avoid some side effects.
- (defvar cus-test-after-load-libs-hook nil
- "Used to switch off undesired side effects of loading libraries.")
- (defvar cus-test-skip-list nil
- "List of variables to disregard by `cus-test-apropos'.")
- (defvar cus-test-libs-noloads
- ;; Loading dunnet in batch mode leads to a Dead end.
- ;; blessmail writes a file.
- ;; characters cannot be loaded twice ("Category `a' is already defined").
- '("play/dunnet.el" "emulation/edt-mapper.el"
- "loadup.el" "mail/blessmail.el" "international/characters.el"
- "cedet/ede/loaddefs.el" "cedet/semantic/loaddefs.el"
- "net/tramp-loaddefs.el")
- "List of files not to load by `cus-test-load-libs'.
- Names should be as they appear in loaddefs.el.")
- ;; This avoids a hang of `cus-test-apropos' in 21.2.
- ;; (add-to-list 'cus-test-skip-list 'sh-alias-alist)
- (or noninteractive
- ;; Never Viperize.
- (setq viper-mode nil))
- ;; Don't create a file `save-place-file'.
- (eval-after-load "saveplace"
- '(remove-hook 'kill-emacs-hook 'save-place-kill-emacs-hook))
- ;; Don't create a file `abbrev-file-name'.
- (setq save-abbrevs nil)
- ;; Avoid compile logs from adviced functions.
- (eval-after-load "bytecomp"
- '(setq ad-default-compilation-action 'never))
- ;;; Main code:
- ;; We want to log all messages.
- (setq message-log-max t)
- (require 'cus-edit)
- (require 'cus-load)
- (defvar cus-test-errors nil
- "List of problematic variables found by `cus-test-apropos'.")
- (defvar cus-test-tested-variables nil
- "List of options tested by last call of `cus-test-apropos'.")
- ;; I haven't understood this :get stuff. The symbols with a
- ;; custom-get property are stored here.
- (defvar cus-test-vars-with-custom-get nil
- "Set by `cus-test-apropos' to a list of options with :get property.")
- (defvar cus-test-vars-with-changed-state nil
- "Set by `cus-test-apropos' to a list of options with state 'changed.")
- (defvar cus-test-deps-errors nil
- "List of require/load problems found by `cus-test-deps'.")
- (defvar cus-test-deps-required nil
- "List of dependencies required by `cus-test-deps'.
- Only unloaded features will be require'd.")
- (defvar cus-test-deps-loaded nil
- "List of dependencies loaded by `cus-test-deps'.")
- (defvar cus-test-libs-errors nil
- "List of load problems found by `cus-test-load-libs' or `cus-test-libs'.")
- (defvar cus-test-libs-loaded nil
- "List of files loaded by `cus-test-load-libs' or `cus-test-libs'.")
- (defvar cus-test-vars-not-cus-loaded nil
- "A list of options not loaded by `custom-load-symbol'.
- Set by `cus-test-noloads'.")
- ;; (defvar cus-test-vars-cus-loaded nil
- ;; "A list of options loaded by `custom-load-symbol'.")
- (defun cus-test-apropos (regexp)
- "Check the options matching REGEXP.
- The detected problematic options are stored in `cus-test-errors'."
- (interactive "sVariable regexp: ")
- (setq cus-test-errors nil)
- (setq cus-test-tested-variables nil)
- (mapc
- (lambda (symbol)
- (push symbol cus-test-tested-variables)
- ;; Be verbose in case we hang.
- (message "Cus Test running...%s %s"
- (length cus-test-tested-variables) symbol)
- (condition-case alpha
- ;; FIXME This defaults to 'sexp if no type was specified.
- ;; Always report such instances as a type mismatch.
- ;; Currently abusing cusver-scan to do that.
- (let* ((type (custom-variable-type symbol))
- (conv (widget-convert type))
- (get (or (get symbol 'custom-get) 'default-value))
- values
- mismatch)
- (when (default-boundp symbol)
- (push (funcall get symbol) values)
- (push (eval (car (get symbol 'standard-value))) values))
- (if (boundp symbol)
- (push (symbol-value symbol) values))
- ;; That does not work.
- ;; (push (widget-get conv :value) values)
- ;; Check the values
- (mapc (lambda (value)
- ;; TODO for booleans, check for values that can be
- ;; evaluated and are not t or nil. Usually a bug.
- (unless (widget-apply conv :match value)
- (setq mismatch 'mismatch)))
- values)
- ;; Store symbols with a custom-get property.
- (when (get symbol 'custom-get)
- (add-to-list 'cus-test-vars-with-custom-get symbol))
- ;; Changed outside the customize buffer?
- ;; This routine is not very much tested.
- (let ((c-value
- (or (get symbol 'customized-value)
- (get symbol 'saved-value)
- (get symbol 'standard-value))))
- (and (consp c-value)
- (boundp symbol)
- (not (equal (eval (car c-value)) (symbol-value symbol)))
- (add-to-list 'cus-test-vars-with-changed-state symbol)))
- (if mismatch
- (push symbol cus-test-errors)))
- (error
- (push symbol cus-test-errors)
- (message "Error for %s: %s" symbol alpha))))
- (cus-test-get-options regexp))
- (message "%s options tested"
- (length cus-test-tested-variables))
- (cus-test-errors-display))
- (defun cus-test-cus-load-groups (&optional cus-load)
- "Return a list of current custom groups.
- If CUS-LOAD is non-nil, include groups from cus-load.el."
- (append (mapcar 'cdr custom-current-group-alist)
- (if cus-load
- (with-temp-buffer
- (insert-file-contents (locate-library "cus-load.el"))
- (search-forward "(put '")
- (beginning-of-line)
- (let (res)
- (while (and (looking-at "^(put '\\(\\S-+\\)")
- (zerop (forward-line 1)))
- (push (intern (match-string 1)) res))
- res)))))
- (defun cus-test-get-options (regexp &optional group)
- "Return a list of custom options matching REGEXP.
- If GROUP is non-nil, return groups rather than options.
- If GROUP is `cus-load', include groups listed in cus-loads as well as
- currently defined groups."
- (let ((groups (if group (cus-test-cus-load-groups (eq group 'cus-load))))
- found)
- (mapatoms
- (lambda (symbol)
- (and
- (if group
- (memq symbol groups)
- (or
- ;; (user-variable-p symbol)
- (get symbol 'standard-value)
- ;; (get symbol 'saved-value)
- (get symbol 'custom-type)))
- (string-match regexp (symbol-name symbol))
- (not (member symbol cus-test-skip-list))
- (push symbol found))))
- found))
- (defun cus-test-errors-display ()
- "Report about the errors found by cus-test."
- (with-output-to-temp-buffer "*cus-test-errors*"
- (set-buffer standard-output)
- (insert (format "Cus Test tested %s variables.\
- See `cus-test-tested-variables'.\n\n"
- (length cus-test-tested-variables)))
- (if (not cus-test-errors)
- (insert "No errors found by cus-test.")
- (insert "The following variables seem to have problems:\n\n")
- (dolist (e cus-test-errors)
- (insert (symbol-name e) "\n")))))
- (defun cus-test-load-custom-loads ()
- "Call `custom-load-symbol' on all atoms."
- (interactive)
- (if noninteractive (let (noninteractive) (require 'dunnet)))
- (mapatoms 'custom-load-symbol)
- (run-hooks 'cus-test-after-load-libs-hook))
- (defmacro cus-test-load-1 (&rest body)
- `(progn
- (setq cus-test-libs-errors nil
- cus-test-libs-loaded nil)
- ,@body
- (message "%s libraries loaded successfully"
- (length cus-test-libs-loaded))
- (if (not cus-test-libs-errors)
- (message "No load problems encountered")
- (message "The following load problems appeared:")
- (cus-test-message cus-test-libs-errors))
- (run-hooks 'cus-test-after-load-libs-hook)))
- ;; This is just cus-test-libs, but loading in the current Emacs process.
- (defun cus-test-load-libs (&optional more)
- "Load the libraries with autoloads.
- Don't load libraries in `cus-test-libs-noloads'.
- If optional argument MORE is \"defcustom\", load all files with defcustoms.
- If it is \"all\", load all Lisp files."
- (interactive)
- (cus-test-load-1
- (let ((lispdir (file-name-directory (locate-library "loaddefs"))))
- (mapc
- (lambda (file)
- (condition-case alpha
- (unless (member file cus-test-libs-noloads)
- (load (file-name-sans-extension (expand-file-name file lispdir)))
- (push file cus-test-libs-loaded))
- (error
- (push (cons file alpha) cus-test-libs-errors)
- (message "Error for %s: %s" file alpha))))
- (if more
- (cus-test-get-lisp-files (equal more "all"))
- (cus-test-get-autoload-deps))))))
- (defun cus-test-get-autoload-deps ()
- "Return the list of files with autoloads."
- (with-temp-buffer
- (insert-file-contents (locate-library "loaddefs"))
- (let (files)
- (while (search-forward "\n;;; Generated autoloads from " nil t)
- (push (buffer-substring (match-end 0) (line-end-position)) files))
- files)))
- (defun cus-test-get-lisp-files (&optional all)
- "Return list of all Lisp files with defcustoms.
- Optional argument ALL non-nil means list all (non-obsolete) Lisp files."
- (let ((default-directory (expand-file-name "lisp/" source-directory))
- (msg "Finding files..."))
- (message "%s" msg)
- (prog1
- ;; Hack to remove leading "./".
- (mapcar (lambda (e) (substring e 2))
- (apply 'process-lines find-program
- "-name" "obsolete" "-prune" "-o"
- "-name" "[^.]*.el" ; ignore .dir-locals.el
- (if all
- '("-print")
- (list "-exec" grep-program
- "-l" "^[ \t]*(defcustom" "{}" "+"))))
- (message "%sdone" msg))))
- (defun cus-test-message (list)
- "Print the members of LIST line by line."
- (dolist (m list) (message "%s" m)))
- ;;; The routines for batch mode:
- (defun cus-test-opts (&optional all)
- "Test custom options.
- This function is suitable for batch mode. E.g., invoke
- src/emacs -batch -l admin/cus-test.el -f cus-test-opts
- in the Emacs source directory.
- Normally only tests options belonging to files in loaddefs.el.
- If optional argument ALL is non-nil, test all files with defcustoms."
- (interactive)
- (and noninteractive
- command-line-args-left
- (setq all (pop command-line-args-left)))
- (message "Running %s" 'cus-test-load-libs)
- (cus-test-load-libs (if all "defcustom"))
- (message "Running %s" 'cus-test-load-custom-loads)
- (cus-test-load-custom-loads)
- (message "Running %s" 'cus-test-apropos)
- (cus-test-apropos "")
- (if (not cus-test-errors)
- (message "No problems found")
- (message "The following options might have problems:")
- (cus-test-message cus-test-errors)))
- (defun cus-test-deps ()
- "Run a verbose version of `custom-load-symbol' on all atoms.
- This function is suitable for batch mode. E.g., invoke
- src/emacs -batch -l admin/cus-test.el -f cus-test-deps
- in the Emacs source directory."
- (interactive)
- (setq cus-test-deps-errors nil)
- (setq cus-test-deps-required nil)
- (setq cus-test-deps-loaded nil)
- (mapatoms
- ;; This code is mainly from `custom-load-symbol'.
- (lambda (symbol)
- (let ((custom-load-recursion t))
- (dolist (load (get symbol 'custom-loads))
- (cond
- ((symbolp load)
- ;; (condition-case nil (require load) (error nil))
- (condition-case alpha
- (unless (or (featurep load)
- (and noninteractive (eq load 'dunnet)))
- (require load)
- (push (list symbol load) cus-test-deps-required))
- (error
- (push (list symbol load alpha) cus-test-deps-errors)
- (message "Require problem: %s %s %s" symbol load alpha))))
- ((equal load "loaddefs")
- (push
- (message "Symbol %s has loaddefs as custom dependency" symbol)
- cus-test-deps-errors))
- ;; This is subsumed by the test below, but it's much
- ;; faster.
- ((assoc load load-history))
- ;; This was just
- ;; (assoc (locate-library load) load-history)
- ;; but has been optimized not to load locate-library
- ;; if not necessary.
- ((let ((regexp (concat "\\(\\`\\|/\\)" (regexp-quote load)
- "\\(\\'\\|\\.\\)"))
- (found nil))
- (dolist (loaded load-history)
- (and (stringp (car loaded))
- (string-match regexp (car loaded))
- (setq found t)))
- found))
- ;; Without this, we would load cus-edit recursively.
- ;; We are still loading it when we call this,
- ;; and it is not in load-history yet.
- ((equal load "cus-edit"))
- ;; This would ignore load problems with files in
- ;; lisp/term/
- ;; ((locate-library (concat term-file-prefix load)))
- (t
- ;; (condition-case nil (load load) (error nil))
- (condition-case alpha
- (progn
- (load load)
- (push (list symbol load) cus-test-deps-loaded))
- (error
- (push (list symbol load alpha) cus-test-deps-errors)
- (message "Load Problem: %s %s %s" symbol load alpha))))
- )))))
- (message "%s features required"
- (length cus-test-deps-required))
- (message "%s files loaded"
- (length cus-test-deps-loaded))
- (if (not cus-test-deps-errors)
- (message "No load problems encountered")
- (message "The following load problems appeared:")
- (cus-test-message cus-test-deps-errors))
- (run-hooks 'cus-test-after-load-libs-hook))
- (defun cus-test-libs (&optional more)
- "Load the libraries with autoloads in separate processes.
- This function is useful to detect load problems of libraries.
- It is suitable for batch mode. E.g., invoke
- ./src/emacs -batch -l admin/cus-test.el -f cus-test-libs
- in the Emacs source directory.
- If optional argument MORE is \"defcustom\", load all files with defcustoms.
- If it is \"all\", load all Lisp files."
- (interactive)
- (and noninteractive
- command-line-args-left
- (setq more (pop command-line-args-left)))
- (cus-test-load-1
- (let* ((default-directory source-directory)
- (emacs (expand-file-name "src/emacs"))
- skipped)
- (or (file-executable-p emacs)
- (error "No such executable `%s'" emacs))
- (mapc
- (lambda (file)
- (if (member file cus-test-libs-noloads)
- (push file skipped)
- (condition-case alpha
- (let* ((fn (expand-file-name file "lisp/"))
- (elc (concat fn "c"))
- status)
- (if (file-readable-p elc) ; load compiled if present (faster)
- (setq fn elc)
- (or (file-readable-p fn)
- (error "Library %s not found" file)))
- (if (equal 0 (setq status (call-process emacs nil nil nil
- "-batch" "-l" fn)))
- (message "%s" file)
- (error "%s" status))
- (push file cus-test-libs-loaded))
- (error
- (push (cons file alpha) cus-test-libs-errors)
- (message "Error for %s: %s" file alpha)))))
- (if more
- (cus-test-get-lisp-files (equal more "all"))
- (cus-test-get-autoload-deps)))
- (message "Default directory: %s" default-directory)
- (when skipped
- (message "The following libraries were skipped:")
- (cus-test-message skipped)))))
- (defun cus-test-noloads ()
- "Find custom options not loaded by `custom-load-symbol'.
- Calling this function after `cus-test-load-libs' is not meaningful.
- It is suitable for batch mode. E.g., invoke
- src/emacs -batch -l admin/cus-test.el -f cus-test-noloads
- in the Emacs source directory."
- (interactive)
- (let ((groups-loaded (cus-test-get-options "" 'cus-load))
- cus-loaded groups-not-loaded)
- (message "Running %s" 'cus-test-load-custom-loads)
- (cus-test-load-custom-loads)
- (setq cus-loaded (cus-test-get-options ""))
- (message "Running %s" 'cus-test-load-libs)
- (cus-test-load-libs "all")
- (setq cus-test-vars-not-cus-loaded (cus-test-get-options "")
- groups-not-loaded (cus-test-get-options "" t))
- (dolist (o cus-loaded)
- (setq cus-test-vars-not-cus-loaded
- (delete o cus-test-vars-not-cus-loaded)))
- (if (not cus-test-vars-not-cus-loaded)
- (message "No options not loaded by custom-load-symbol found")
- (message "The following options were not loaded by custom-load-symbol:")
- (cus-test-message
- (sort cus-test-vars-not-cus-loaded 'string<)))
- (dolist (o groups-loaded)
- (setq groups-not-loaded (delete o groups-not-loaded)))
- (if (not groups-not-loaded)
- (message "No groups not in cus-load.el found")
- (message "The following groups are not in cus-load.el:")
- (cus-test-message (sort groups-not-loaded 'string<)))))
- (provide 'cus-test)
- ;;; cus-test.el ends here
|