470 lines
18 KiB
EmacsLisp
470 lines
18 KiB
EmacsLisp
|
;;; flycheck-ert.el --- Flycheck: ERT extensions -*- lexical-binding: t; -*-
|
|||
|
|
|||
|
;; Copyright (C) 2013-2015 Sebastian Wiesner <swiesner@lunaryorn.com>
|
|||
|
|
|||
|
;; Author: Sebastian Wiesner <swiesner@lunaryorn.com>
|
|||
|
;; URL: https://github.com/flycheck/flycheck
|
|||
|
|
|||
|
;; This file is not part of GNU Emacs.
|
|||
|
|
|||
|
;; This program 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.
|
|||
|
|
|||
|
;; This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
|||
|
|
|||
|
;;; Commentary:
|
|||
|
|
|||
|
;; Unit testing library for Flycheck, the modern on-the-fly syntax checking
|
|||
|
;; extension for GNU Emacs.
|
|||
|
|
|||
|
;; Provide various utility functions and unit test helpers to test Flycheck and
|
|||
|
;; Flycheck extensions.
|
|||
|
|
|||
|
;;; Code:
|
|||
|
|
|||
|
(require 'flycheck)
|
|||
|
(require 'ert)
|
|||
|
(require 'macroexp) ; For macro utilities
|
|||
|
|
|||
|
|
|||
|
;;; Compatibility
|
|||
|
|
|||
|
(eval-and-compile
|
|||
|
;; Provide `ert-skip' and friends for Emacs 24.3
|
|||
|
(defconst flycheck-ert-ert-can-skip (fboundp 'ert-skip)
|
|||
|
"Whether ERT supports test skipping.")
|
|||
|
|
|||
|
(unless flycheck-ert-ert-can-skip
|
|||
|
;; Fake skipping
|
|||
|
|
|||
|
(put 'flycheck-ert-skipped 'error-message "Test skipped")
|
|||
|
(put 'flycheck-ert-skipped 'error-conditions '(error))
|
|||
|
|
|||
|
(defun ert-skip (data)
|
|||
|
(signal 'flycheck-ert-skipped data))
|
|||
|
|
|||
|
(defmacro skip-unless (form)
|
|||
|
`(unless (ignore-errors ,form)
|
|||
|
(signal 'flycheck-ert-skipped ',form)))
|
|||
|
|
|||
|
(defun ert-test-skipped-p (result)
|
|||
|
(and (ert-test-failed-p result)
|
|||
|
(eq (car (ert-test-failed-condition result))
|
|||
|
'flycheck-ert-skipped)))))
|
|||
|
|
|||
|
|
|||
|
;;; Internal variables
|
|||
|
|
|||
|
(defvar flycheck-ert--resource-directory nil
|
|||
|
"The directory to get resources from in this test suite.")
|
|||
|
|
|||
|
|
|||
|
;;; Resource management macros
|
|||
|
|
|||
|
(defmacro flycheck-ert-with-temp-buffer (&rest body)
|
|||
|
"Eval BODY within a temporary buffer.
|
|||
|
|
|||
|
Like `with-temp-buffer', but resets the modification state of the
|
|||
|
temporary buffer to make sure that it is properly killed even if
|
|||
|
it has a backing file and is modified."
|
|||
|
(declare (indent 0))
|
|||
|
`(with-temp-buffer
|
|||
|
(unwind-protect
|
|||
|
,(macroexp-progn body)
|
|||
|
;; Reset modification state of the buffer, and unlink it from its backing
|
|||
|
;; file, if any, because Emacs refuses to kill modified buffers with
|
|||
|
;; backing files, even if they are temporary.
|
|||
|
(set-buffer-modified-p nil)
|
|||
|
(set-visited-file-name nil 'no-query))))
|
|||
|
|
|||
|
(defmacro flycheck-ert-with-file-buffer (file-name &rest body)
|
|||
|
"Create a buffer from FILE-NAME and eval BODY.
|
|||
|
|
|||
|
BODY is evaluated with `current-buffer' being a buffer with the
|
|||
|
contents FILE-NAME."
|
|||
|
(declare (indent 1))
|
|||
|
`(let ((file-name ,file-name))
|
|||
|
(unless (file-exists-p file-name)
|
|||
|
(error "%s does not exist" file-name))
|
|||
|
(flycheck-ert-with-temp-buffer
|
|||
|
(insert-file-contents file-name 'visit)
|
|||
|
(set-visited-file-name file-name 'no-query)
|
|||
|
(cd (file-name-directory file-name))
|
|||
|
;; Mark the buffer as not modified, because we just loaded the file up to
|
|||
|
;; now.
|
|||
|
(set-buffer-modified-p nil)
|
|||
|
,@body)))
|
|||
|
|
|||
|
(defmacro flycheck-ert-with-help-buffer (&rest body)
|
|||
|
"Execute BODY and kill the help buffer afterwards.
|
|||
|
|
|||
|
Use this macro to test functions that create a Help buffer."
|
|||
|
(declare (indent 0))
|
|||
|
`(unwind-protect
|
|||
|
,(macroexp-progn body)
|
|||
|
(when (buffer-live-p (get-buffer (help-buffer)))
|
|||
|
(kill-buffer (help-buffer)))))
|
|||
|
|
|||
|
(defmacro flycheck-ert-with-global-mode (&rest body)
|
|||
|
"Execute BODY with Global Flycheck Mode enabled.
|
|||
|
|
|||
|
After BODY, restore the old state of Global Flycheck Mode."
|
|||
|
(declare (indent 0))
|
|||
|
`(let ((old-state global-flycheck-mode))
|
|||
|
(unwind-protect
|
|||
|
(progn
|
|||
|
(global-flycheck-mode 1)
|
|||
|
,@body)
|
|||
|
(global-flycheck-mode (if old-state 1 -1)))))
|
|||
|
|
|||
|
(defmacro flycheck-ert-with-env (env &rest body)
|
|||
|
"Add ENV to `process-environment' in BODY.
|
|||
|
|
|||
|
Execute BODY with a `process-environment' with contains all
|
|||
|
variables from ENV added.
|
|||
|
|
|||
|
ENV is an alist, where each cons cell `(VAR . VALUE)' is a
|
|||
|
environment variable VAR to be added to `process-environment'
|
|||
|
with VALUE."
|
|||
|
(declare (indent 1))
|
|||
|
`(let ((process-environment (copy-sequence process-environment)))
|
|||
|
(pcase-dolist (`(,var . ,value) ,env)
|
|||
|
(setenv var value))
|
|||
|
,@body))
|
|||
|
|
|||
|
|
|||
|
;;; Test resources
|
|||
|
(defun flycheck-ert-resource-filename (resource-file)
|
|||
|
"Determine the absolute file name of a RESOURCE-FILE.
|
|||
|
|
|||
|
Relative file names are expanded against
|
|||
|
`flycheck-ert-resources-directory'."
|
|||
|
(expand-file-name resource-file flycheck-ert--resource-directory))
|
|||
|
|
|||
|
(defmacro flycheck-ert-with-resource-buffer (resource-file &rest body)
|
|||
|
"Create a temp buffer from a RESOURCE-FILE and execute BODY.
|
|||
|
|
|||
|
The absolute file name of RESOURCE-FILE is determined with
|
|||
|
`flycheck-ert-resource-filename'."
|
|||
|
(declare (indent 1))
|
|||
|
`(flycheck-ert-with-file-buffer
|
|||
|
(flycheck-ert-resource-filename ,resource-file)
|
|||
|
,@body))
|
|||
|
|
|||
|
(defun flycheck-ert-locate-config-file (filename _checker)
|
|||
|
"Find a configuration FILENAME within unit tests.
|
|||
|
|
|||
|
_CHECKER is ignored."
|
|||
|
(let* ((directory (flycheck-ert-resource-filename "config-files"))
|
|||
|
(filepath (expand-file-name filename directory)))
|
|||
|
(when (file-exists-p filepath)
|
|||
|
filepath)))
|
|||
|
|
|||
|
|
|||
|
;;; Test suite initialization
|
|||
|
|
|||
|
(defun flycheck-ert-initialize (resource-dir)
|
|||
|
"Initialize a test suite with RESOURCE-DIR.
|
|||
|
|
|||
|
RESOURCE-DIR is the directory, `flycheck-ert-resource-filename'
|
|||
|
should use to lookup resource files."
|
|||
|
(when flycheck-ert--resource-directory
|
|||
|
(error "Test suite already initialized"))
|
|||
|
(let ((tests (ert-select-tests t t)))
|
|||
|
;; Select all tests
|
|||
|
(unless tests
|
|||
|
(error "No tests defined. Call `flycheck-ert-initialize' after defining all tests!"))
|
|||
|
|
|||
|
(setq flycheck-ert--resource-directory resource-dir)
|
|||
|
|
|||
|
;; Emacs 24.3 don't support skipped tests, so we add poor man's test
|
|||
|
;; skipping: We mark skipped tests as expected failures by adjusting the
|
|||
|
;; expected result of all test cases. Not particularly pretty, but works :)
|
|||
|
(unless flycheck-ert-ert-can-skip
|
|||
|
(dolist (test tests)
|
|||
|
(let ((result (ert-test-expected-result-type test)))
|
|||
|
(setf (ert-test-expected-result-type test)
|
|||
|
`(or ,result (satisfies ert-test-skipped-p))))))))
|
|||
|
|
|||
|
|
|||
|
;;; Environment and version information
|
|||
|
|
|||
|
(defconst flycheck-ert-user-error-type
|
|||
|
(if (version< emacs-version "24.2")
|
|||
|
'error
|
|||
|
'user-error)
|
|||
|
"The `user-error' type used by Flycheck.")
|
|||
|
|
|||
|
(defun flycheck-ert-travis-ci-p ()
|
|||
|
"Determine whether we are running on Travis CI."
|
|||
|
(string= (getenv "TRAVIS") "true"))
|
|||
|
|
|||
|
(defun flycheck-ert-check-gpg ()
|
|||
|
"Check whether GPG is available."
|
|||
|
(or (epg-check-configuration (epg-configuration)) t))
|
|||
|
|
|||
|
(defun flycheck-ert-extract-version-command (re executable &rest args)
|
|||
|
"Use RE to extract the version from EXECUTABLE with ARGS.
|
|||
|
|
|||
|
Run EXECUTABLE with ARGS, catch the output, and apply RE to find
|
|||
|
the version number. Return the text captured by the first group
|
|||
|
in RE, or nil, if EXECUTABLE is missing, or if RE failed to
|
|||
|
match."
|
|||
|
(-when-let (executable (executable-find executable))
|
|||
|
(with-temp-buffer
|
|||
|
(apply #'call-process executable nil t nil args)
|
|||
|
(goto-char (point-min))
|
|||
|
(when (re-search-forward re nil 'no-error)
|
|||
|
(match-string 1)))))
|
|||
|
|
|||
|
|
|||
|
;;; Test case definitions
|
|||
|
(defmacro flycheck-ert-def-checker-test (checker language name
|
|||
|
&rest keys-and-body)
|
|||
|
"Define a test case for a syntax CHECKER for LANGUAGE.
|
|||
|
|
|||
|
CHECKER is a symbol or a list of symbols denoting syntax checkers
|
|||
|
being tested by the test. The test case is skipped, if any of
|
|||
|
these checkers cannot be used. LANGUAGE is a symbol or a list of
|
|||
|
symbols denoting the programming languages supported by the
|
|||
|
syntax checkers. This is currently only used for tagging the
|
|||
|
test appropriately.
|
|||
|
|
|||
|
NAME is a symbol denoting the local name of the test. The test
|
|||
|
itself is ultimately named
|
|||
|
`flycheck-define-checker/CHECKER/NAME'. If CHECKER is a list,
|
|||
|
the first checker in the list is used for naming the test.
|
|||
|
|
|||
|
Optionally, the keyword arguments `:tags' and `:expected-result'
|
|||
|
may be given. They have the same meaning as in `ert-deftest.',
|
|||
|
and are added to the tags and result expectations set up by this
|
|||
|
macro.
|
|||
|
|
|||
|
The remaining forms denote the body of the test case, including
|
|||
|
assertions and setup code."
|
|||
|
(declare (indent 3))
|
|||
|
(unless checker
|
|||
|
(error "No syntax checkers specified."))
|
|||
|
(unless language
|
|||
|
(error "No languages specified"))
|
|||
|
(let* ((checkers (if (symbolp checker) (list checker) checker))
|
|||
|
(checker (car checkers))
|
|||
|
(languages (if (symbolp language) (list language) language))
|
|||
|
(language-tags (mapcar (lambda (l) (intern (format "language-%s" l)))
|
|||
|
languages))
|
|||
|
(checker-tags (mapcar (lambda (c) (intern (format "checker-%s" c)))
|
|||
|
checkers))
|
|||
|
(local-name (or name 'default))
|
|||
|
(full-name (intern (format "flycheck-define-checker/%s/%s"
|
|||
|
checker local-name)))
|
|||
|
(keys-and-body (ert--parse-keys-and-body keys-and-body))
|
|||
|
(body (cadr keys-and-body))
|
|||
|
(keys (car keys-and-body))
|
|||
|
(default-tags '(syntax-checker external-tool)))
|
|||
|
`(ert-deftest ,full-name ()
|
|||
|
:expected-result
|
|||
|
(list 'or
|
|||
|
'(satisfies flycheck-ert-syntax-check-timed-out-p)
|
|||
|
,(or (plist-get keys :expected-result) :passed))
|
|||
|
:tags (append ',(append default-tags language-tags checker-tags)
|
|||
|
,(plist-get keys :tags))
|
|||
|
,@(mapcar (lambda (c) `(skip-unless
|
|||
|
;; Ignore non-command checkers
|
|||
|
(or (not (get ',c 'flycheck-command))
|
|||
|
(executable-find (flycheck-checker-executable ',c)))))
|
|||
|
checkers)
|
|||
|
,@body)))
|
|||
|
|
|||
|
|
|||
|
;;; Test case results
|
|||
|
|
|||
|
(defun flycheck-ert-syntax-check-timed-out-p (result)
|
|||
|
"Whether RESULT denotes a timed-out test.
|
|||
|
|
|||
|
RESULT is an ERT test result object."
|
|||
|
(and (ert-test-failed-p result)
|
|||
|
(eq (car (ert-test-failed-condition result))
|
|||
|
'flycheck-ert-syntax-check-timed-out)))
|
|||
|
|
|||
|
|
|||
|
;;; Syntax checking in tests
|
|||
|
|
|||
|
(defvar-local flycheck-ert-syntax-checker-finished nil
|
|||
|
"Non-nil if the current checker has finished.")
|
|||
|
|
|||
|
(add-hook 'flycheck-after-syntax-check-hook
|
|||
|
(lambda () (setq flycheck-ert-syntax-checker-finished t)))
|
|||
|
|
|||
|
(defconst flycheck-ert-checker-wait-time 10
|
|||
|
"Time to wait until a checker is finished in seconds.
|
|||
|
|
|||
|
After this time has elapsed, the checker is considered to have
|
|||
|
failed, and the test aborted with failure.")
|
|||
|
|
|||
|
(put 'flycheck-ert-syntax-check-timed-out 'error-message
|
|||
|
"Syntax check timed out.")
|
|||
|
(put 'flycheck-ert-syntax-check-timed-out 'error-conditions '(error))
|
|||
|
|
|||
|
(defun flycheck-ert-wait-for-syntax-checker ()
|
|||
|
"Wait until the syntax check in the current buffer is finished."
|
|||
|
(let ((starttime (float-time)))
|
|||
|
(while (and (not flycheck-ert-syntax-checker-finished)
|
|||
|
(< (- (float-time) starttime) flycheck-ert-checker-wait-time))
|
|||
|
(sleep-for 1))
|
|||
|
(unless (< (- (float-time) starttime) flycheck-ert-checker-wait-time)
|
|||
|
(flycheck-stop)
|
|||
|
(signal 'flycheck-ert-syntax-check-timed-out nil)))
|
|||
|
(setq flycheck-ert-syntax-checker-finished nil))
|
|||
|
|
|||
|
(defun flycheck-ert-buffer-sync ()
|
|||
|
"Like `flycheck-buffer', but synchronously."
|
|||
|
(setq flycheck-ert-syntax-checker-finished nil)
|
|||
|
(should (not (flycheck-running-p)))
|
|||
|
(flycheck-mode) ; This will only start a deferred check,
|
|||
|
(flycheck-buffer) ; so we need an explicit manual check
|
|||
|
;; After starting the check, the checker should either be running now, or
|
|||
|
;; already be finished (if it was fast).
|
|||
|
(should (or flycheck-current-syntax-check
|
|||
|
flycheck-ert-syntax-checker-finished))
|
|||
|
;; Also there should be no deferred check pending anymore
|
|||
|
(should-not (flycheck-deferred-check-p))
|
|||
|
(flycheck-ert-wait-for-syntax-checker))
|
|||
|
|
|||
|
(defun flycheck-ert-ensure-clear ()
|
|||
|
"Clear the current buffer.
|
|||
|
|
|||
|
Raise an assertion error if the buffer is not clear afterwards."
|
|||
|
(flycheck-clear)
|
|||
|
(should (not flycheck-current-errors))
|
|||
|
(should (not (-any? (lambda (ov) (overlay-get ov 'flycheck-overlay))
|
|||
|
(overlays-in (point-min) (point-max))))))
|
|||
|
|
|||
|
|
|||
|
;;; Test assertions
|
|||
|
|
|||
|
(defun flycheck-ert-should-overlay (error)
|
|||
|
"Test that ERROR has a proper overlay in the current buffer.
|
|||
|
|
|||
|
ERROR is a Flycheck error object."
|
|||
|
(let* ((overlay (-first (lambda (ov) (equal (overlay-get ov 'flycheck-error)
|
|||
|
error))
|
|||
|
(flycheck-overlays-in 0 (+ 1 (buffer-size)))))
|
|||
|
(region (flycheck-error-region-for-mode error 'symbols))
|
|||
|
(message (flycheck-error-message error))
|
|||
|
(level (flycheck-error-level error))
|
|||
|
(category (flycheck-error-level-overlay-category level))
|
|||
|
(face (get category 'face))
|
|||
|
(fringe-bitmap (flycheck-error-level-fringe-bitmap level))
|
|||
|
(fringe-face (flycheck-error-level-fringe-face level))
|
|||
|
(fringe-icon (list 'left-fringe fringe-bitmap fringe-face)))
|
|||
|
(should overlay)
|
|||
|
(should (overlay-get overlay 'flycheck-overlay))
|
|||
|
(should (= (overlay-start overlay) (car region)))
|
|||
|
(should (= (overlay-end overlay) (cdr region)))
|
|||
|
(should (eq (overlay-get overlay 'face) face))
|
|||
|
(should (equal (get-char-property 0 'display
|
|||
|
(overlay-get overlay 'before-string))
|
|||
|
fringe-icon))
|
|||
|
(should (eq (overlay-get overlay 'category) category))
|
|||
|
(should (equal (overlay-get overlay 'flycheck-error) error))
|
|||
|
(should (string= (overlay-get overlay 'help-echo) message))))
|
|||
|
|
|||
|
(defun flycheck-ert-should-errors (&rest errors)
|
|||
|
"Test that the current buffers has ERRORS.
|
|||
|
|
|||
|
ERRORS is a list of errors expected to be present in the current
|
|||
|
buffer. Each error is given as a list of arguments to
|
|||
|
`flycheck-error-new-at'.
|
|||
|
|
|||
|
If ERRORS are omitted, test that there are no errors at all in
|
|||
|
the current buffer.
|
|||
|
|
|||
|
With ERRORS, test that each error in ERRORS is present in the
|
|||
|
current buffer, and that the number of errors in the current
|
|||
|
buffer is equal to the number of given ERRORS. In other words,
|
|||
|
check that the buffer has all ERRORS, and no other errors."
|
|||
|
(let ((expected (mapcar (apply-partially #'apply #'flycheck-error-new-at)
|
|||
|
errors)))
|
|||
|
(should (equal expected flycheck-current-errors))
|
|||
|
(mapc #'flycheck-ert-should-overlay expected))
|
|||
|
(should (= (length errors)
|
|||
|
(length (flycheck-overlays-in (point-min) (point-max))))))
|
|||
|
|
|||
|
(defun flycheck-ert-should-syntax-check (resource-file modes &rest errors)
|
|||
|
"Test a syntax check in RESOURCE-FILE with MODES.
|
|||
|
|
|||
|
RESOURCE-FILE is the file to check. MODES is a single major mode
|
|||
|
symbol or a list thereof, specifying the major modes to syntax
|
|||
|
check with. If more than one major mode is specified, the test
|
|||
|
is run for each mode separately, so if you give three major
|
|||
|
modes, the entire test will run three times. ERRORS is the list
|
|||
|
of expected errors, as in `flycheck-ert-should-errors'. If
|
|||
|
omitted, the syntax check must not emit any errors. The errors
|
|||
|
are cleared after each test.
|
|||
|
|
|||
|
The syntax checker is selected via standard syntax checker
|
|||
|
selection. To test a specific checker, you need to set
|
|||
|
`flycheck-checker' or `flycheck-disabled-checkers' accordingly
|
|||
|
before using this predicate, depending on whether you want to use
|
|||
|
manual or automatic checker selection.
|
|||
|
|
|||
|
During the syntax check, configuration files of syntax checkers
|
|||
|
are also searched in the `config-files' sub-directory of the
|
|||
|
resource directory."
|
|||
|
(when (symbolp modes)
|
|||
|
(setq modes (list modes)))
|
|||
|
(dolist (mode modes)
|
|||
|
(unless (fboundp mode)
|
|||
|
(ert-skip (format "%S missing" mode)))
|
|||
|
(flycheck-ert-with-resource-buffer resource-file
|
|||
|
(funcall mode)
|
|||
|
;; Configure config file locating for unit tests
|
|||
|
(dolist (fn '(flycheck-locate-config-file-absolute-path
|
|||
|
flycheck-ert-locate-config-file))
|
|||
|
(add-hook 'flycheck-locate-config-file-functions fn 'append 'local))
|
|||
|
(let ((process-hook-called 0))
|
|||
|
(add-hook 'flycheck-process-error-functions
|
|||
|
(lambda (_err)
|
|||
|
(setq process-hook-called (1+ process-hook-called))
|
|||
|
nil)
|
|||
|
nil :local)
|
|||
|
(flycheck-ert-buffer-sync)
|
|||
|
(apply #'flycheck-ert-should-errors errors)
|
|||
|
(should (= process-hook-called (length errors))))
|
|||
|
(flycheck-ert-ensure-clear))))
|
|||
|
|
|||
|
(defun flycheck-ert-at-nth-error (n)
|
|||
|
"Determine whether point is at the N'th Flycheck error.
|
|||
|
|
|||
|
Return non-nil if the point is at the N'th Flycheck error in the
|
|||
|
current buffer. Otherwise return nil."
|
|||
|
(let* ((error (nth (1- n) flycheck-current-errors))
|
|||
|
(mode flycheck-highlighting-mode)
|
|||
|
(region (flycheck-error-region-for-mode error mode)))
|
|||
|
(and (member error (flycheck-overlay-errors-at (point)))
|
|||
|
(= (point) (car region)))))
|
|||
|
|
|||
|
(defun flycheck-ert-explain--at-nth-error (n)
|
|||
|
(let ((errors (flycheck-overlay-errors-at (point))))
|
|||
|
(if (null errors)
|
|||
|
(format "Expected to be at error %s, but no error at point %s"
|
|||
|
n (point))
|
|||
|
(let ((pos (cl-position (car errors) flycheck-current-errors)))
|
|||
|
(format "Expected to be at error %s, but point %s is at error %s"
|
|||
|
n (point) (1+ pos))))))
|
|||
|
|
|||
|
(put 'flycheck-ert-at-nth-error 'ert-explainer
|
|||
|
'flycheck-ert-explain--at-nth-error)
|
|||
|
|
|||
|
(provide 'flycheck-ert)
|
|||
|
|
|||
|
;;; flycheck-ert.el ends here
|