org-test.el 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349
  1. ;;;; org-test.el --- Tests for Org-mode
  2. ;; Copyright (c) 2010 Sebastian Rose, Eric Schulte
  3. ;; Authors:
  4. ;; Sebastian Rose, Hannover, Germany, sebastian_rose gmx de
  5. ;; Eric Schulte, Santa Fe, New Mexico, USA, schulte.eric gmail com
  6. ;; David Maus, Brunswick, Germany, dmaus ictsoc de
  7. ;; Released under the GNU General Public License version 3
  8. ;; see: http://www.gnu.org/licenses/gpl-3.0.html
  9. ;; Definition of `special-mode' copied from Emacs23's simple.el to be
  10. ;; provide a testing environment for Emacs22.
  11. ;;;; Comments:
  12. ;; Interactive testing for Org mode.
  13. ;; The heart of all this is the commands `org-test-current-defun'. If
  14. ;; called while in a `defun' all ert tests with names matching the
  15. ;; name of the function are run.
  16. ;;; Test Development
  17. ;; For test development purposes a number of navigation and test
  18. ;; function construction routines are available as a git submodule
  19. ;; (jump.el)
  20. ;; Install with...
  21. ;; $ git submodule init
  22. ;; $ git submodule update
  23. ;;;; Code:
  24. (let* ((org-test-dir (expand-file-name
  25. (file-name-directory
  26. (or load-file-name buffer-file-name))))
  27. (org-lisp-dir (expand-file-name
  28. (concat org-test-dir "../lisp"))))
  29. (unless (featurep 'org)
  30. (setq load-path (cons org-lisp-dir load-path))
  31. (require 'org)
  32. (require 'org-id)
  33. (org-babel-do-load-languages
  34. 'org-babel-load-languages '((sh . t) (org . t))))
  35. (let* ((load-path (cons
  36. org-test-dir
  37. (cons
  38. (expand-file-name "jump" org-test-dir)
  39. load-path))))
  40. (require 'cl)
  41. (when (= emacs-major-version 22)
  42. (defvar special-mode-map
  43. (let ((map (make-sparse-keymap)))
  44. (suppress-keymap map)
  45. (define-key map "q" 'quit-window)
  46. (define-key map " " 'scroll-up)
  47. (define-key map "\C-?" 'scroll-down)
  48. (define-key map "?" 'describe-mode)
  49. (define-key map "h" 'describe-mode)
  50. (define-key map ">" 'end-of-buffer)
  51. (define-key map "<" 'beginning-of-buffer)
  52. (define-key map "g" 'revert-buffer)
  53. (define-key map "z" 'kill-this-buffer)
  54. map))
  55. (put 'special-mode 'mode-class 'special)
  56. (define-derived-mode special-mode nil "Special"
  57. "Parent major mode from which special major modes should inherit."
  58. (setq buffer-read-only t)))
  59. (require 'ert)
  60. (require 'ert-x)
  61. (when (file-exists-p
  62. (expand-file-name "jump/jump.el" org-test-dir))
  63. (require 'jump)
  64. (require 'which-func))))
  65. (defconst org-test-default-test-file-name "tests.el"
  66. "For each defun a separate file with tests may be defined.
  67. tests.el is the fallback or default if you like.")
  68. (defconst org-test-default-directory-name "testing"
  69. "Basename or the directory where the tests live.
  70. org-test searches this directory up the directory tree.")
  71. (defconst org-test-dir
  72. (expand-file-name (file-name-directory (or load-file-name buffer-file-name))))
  73. (defconst org-base-dir
  74. (expand-file-name ".." org-test-dir))
  75. (defconst org-test-example-dir
  76. (expand-file-name "examples" org-test-dir))
  77. (defconst org-test-file
  78. (expand-file-name "normal.org" org-test-example-dir))
  79. (defconst org-test-no-heading-file
  80. (expand-file-name "no-heading.org" org-test-example-dir))
  81. (defconst org-test-link-in-heading-file
  82. (expand-file-name "link-in-heading.org" org-test-dir))
  83. ;;; Functions for writing tests
  84. (put 'missing-test-dependency
  85. 'error-conditions
  86. '(error missing-test-dependency))
  87. (defun org-test-for-executable (exe)
  88. "Throw an error if EXE is not available.
  89. This can be used at the top of code-block-language specific test
  90. files to avoid loading the file on systems without the
  91. executable."
  92. (unless (reduce
  93. (lambda (acc dir)
  94. (or acc (file-exists-p (expand-file-name exe dir))))
  95. exec-path :initial-value nil)
  96. (signal 'missing-test-dependency (list exe))))
  97. (defun org-test-buffer (&optional file)
  98. "TODO: Setup and return a buffer to work with.
  99. If file is non-nil insert it's contents in there.")
  100. (defun org-test-compare-with-file (&optional file)
  101. "TODO: Compare the contents of the test buffer with FILE.
  102. If file is not given, search for a file named after the test
  103. currently executed.")
  104. (defmacro org-test-at-id (id &rest body)
  105. "Run body after placing the point in the headline identified by ID."
  106. (declare (indent 1))
  107. `(let* ((id-location (org-id-find ,id))
  108. (id-file (car id-location))
  109. (visited-p (get-file-buffer id-file))
  110. to-be-removed)
  111. (save-window-excursion
  112. (save-match-data
  113. (org-id-goto ,id)
  114. (setq to-be-removed (current-buffer))
  115. (condition-case nil
  116. (progn
  117. (org-show-subtree)
  118. (org-show-block-all))
  119. (error nil))
  120. (save-restriction ,@body)))
  121. (unless visited-p
  122. (kill-buffer to-be-removed))))
  123. (defmacro org-test-in-example-file (file &rest body)
  124. "Execute body in the Org-mode example file."
  125. (declare (indent 1))
  126. `(let* ((my-file (or ,file org-test-file))
  127. (visited-p (get-file-buffer my-file))
  128. to-be-removed)
  129. (save-window-excursion
  130. (save-match-data
  131. (find-file my-file)
  132. (unless (eq major-mode 'org-mode)
  133. (org-mode))
  134. (setq to-be-removed (current-buffer))
  135. (goto-char (point-min))
  136. (condition-case nil
  137. (progn
  138. (outline-next-visible-heading 1)
  139. (org-show-subtree)
  140. (org-show-block-all))
  141. (error nil))
  142. (save-restriction ,@body)))
  143. (unless visited-p
  144. (kill-buffer to-be-removed))))
  145. (defmacro org-test-at-marker (file marker &rest body)
  146. "Run body after placing the point at MARKER in FILE.
  147. Note the uuidgen command-line command can be useful for
  148. generating unique markers for insertion as anchors into org
  149. files."
  150. (declare (indent 2))
  151. `(org-test-in-example-file ,file
  152. (goto-char (point-min))
  153. (re-search-forward (regexp-quote ,marker))
  154. ,@body))
  155. (defmacro org-test-with-temp-text (text &rest body)
  156. "Run body in a temporary buffer with Org-mode as the active
  157. mode holding TEXT. If the string \"<point>\" appears in TEXT
  158. then remove it and place the point there before running BODY,
  159. otherwise place the point at the beginning of the inserted text."
  160. (declare (indent 1))
  161. (let ((inside-text (if (stringp text) text (eval text))))
  162. `(with-temp-buffer
  163. (org-mode)
  164. ,(let ((point (string-match (regexp-quote "<point>") inside-text)))
  165. (if point
  166. `(progn (insert `(replace-match "" nil nil inside-text))
  167. (goto-char ,(match-beginning 0)))
  168. `(progn (insert ,inside-text)
  169. (goto-char (point-min)))))
  170. ,@body)))
  171. ;;; Navigation Functions
  172. (when (featurep 'jump)
  173. (defjump org-test-jump
  174. (("lisp/\\1.el" . "testing/lisp/test-\\1.el")
  175. ("lisp/\\1.el" . "testing/lisp/\\1.el/test.*.el")
  176. ("contrib/lisp/\\1.el" . "testing/contrib/lisp/test-\\1.el")
  177. ("contrib/lisp/\\1.el" . "testing/contrib/lisp/\\1.el/test.*.el")
  178. ("testing/lisp/test-\\1.el" . "lisp/\\1.el")
  179. ("testing/lisp/\\1.el" . "lisp/\\1.el/test.*.el")
  180. ("testing/contrib/lisp/test-\\1.el" . "contrib/lisp/\\1.el")
  181. ("testing/contrib/lisp/test-\\1.el" . "contrib/lisp/\\1.el/test.*.el"))
  182. (concat org-base-dir "/")
  183. "Jump between org-mode files and their tests."
  184. (lambda (path)
  185. (let* ((full-path (expand-file-name path org-base-dir))
  186. (file-name (file-name-nondirectory path))
  187. (name (file-name-sans-extension file-name)))
  188. (find-file full-path)
  189. (insert
  190. ";;; " file-name "\n\n"
  191. ";; Copyright (c) " (nth 5 (decode-time (current-time)))
  192. " " user-full-name "\n"
  193. ";; Authors: " user-full-name "\n\n"
  194. ";; Released under the GNU General Public License version 3\n"
  195. ";; see: http://www.gnu.org/licenses/gpl-3.0.html\n\n"
  196. ";;;; Comments:\n\n"
  197. ";; Template test file for Org-mode tests\n\n"
  198. " \n"
  199. ";;; Code:\n"
  200. "(let ((load-path (cons (expand-file-name\n"
  201. " \"..\" (file-name-directory\n"
  202. " (or load-file-name buffer-file-name)))\n"
  203. " load-path)))\n"
  204. " (require 'org-test)\n"
  205. " (require 'org-test-ob-consts))\n\n"
  206. " \n"
  207. ";;; Tests\n"
  208. "(ert-deftest " name "/example-test ()\n"
  209. " \"Just an example to get you started.\"\n"
  210. " (should t)\n"
  211. " (should-not nil)\n"
  212. " (should-error (error \"errr...\")))\n\n\n"
  213. "(provide '" name ")\n\n"
  214. ";;; " file-name " ends here\n") full-path))
  215. (lambda () ((lambda (res) (if (listp res) (car res) res)) (which-function)))))
  216. (define-key emacs-lisp-mode-map "\M-\C-j" 'org-test-jump)
  217. ;;; Miscellaneous helper functions
  218. (defun org-test-strip-text-props (s)
  219. "Return S without any text properties."
  220. (let ((noprop (copy-sequence s)))
  221. (set-text-properties 0 (length noprop) nil noprop)
  222. noprop))
  223. (defun org-test-string-exact-match (regex string &optional start)
  224. "case sensative string-match"
  225. (let ((case-fold-search nil)
  226. (case-replace nil))
  227. (if(and (equal regex "")
  228. (not(equal string "")))
  229. nil
  230. (if (equal 0 (string-match regex string start))
  231. t
  232. nil))))
  233. ;;; Load and Run tests
  234. (defun org-test-load ()
  235. "Load up the org-mode test suite."
  236. (interactive)
  237. (flet ((rld (base)
  238. ;; Recursively load all files, if files throw errors
  239. ;; then silently ignore the error and continue to the
  240. ;; next file. This allows files to error out if
  241. ;; required executables aren't available.
  242. (mapc
  243. (lambda (path)
  244. (if (file-directory-p path)
  245. (rld path)
  246. (condition-case err
  247. (when (string-match "^[A-Za-z].*\\.el$"
  248. (file-name-nondirectory path))
  249. (load-file path))
  250. (missing-test-dependency
  251. (let ((name (intern
  252. (concat "org-missing-dependency/"
  253. (file-name-nondirectory
  254. (file-name-sans-extension path))))))
  255. (eval `(ert-deftest ,name ()
  256. :expected-result :failed (should nil))))))))
  257. (directory-files base 'full
  258. "^\\([^.]\\|\\.\\([^.]\\|\\..\\)\\).*\\.el$"))))
  259. (rld (expand-file-name "lisp" org-test-dir))
  260. (rld (expand-file-name "lisp" (expand-file-name "contrib" org-test-dir)))))
  261. (defun org-test-current-defun ()
  262. "Test the current function."
  263. (interactive)
  264. (ert (which-function)))
  265. (defun org-test-current-file ()
  266. "Run all tests for current file."
  267. (interactive)
  268. (ert (concat "test-"
  269. (file-name-sans-extension
  270. (file-name-nondirectory (buffer-file-name)))
  271. "/")))
  272. (defun org-test-touch-all-examples ()
  273. (dolist (file (directory-files
  274. org-test-example-dir 'full
  275. "^\\([^.]\\|\\.\\([^.]\\|\\..\\)\\).*\\.org$"))
  276. (find-file file)))
  277. (defun org-test-update-id-locations ()
  278. (org-id-update-id-locations
  279. (directory-files
  280. org-test-example-dir 'full
  281. "^\\([^.]\\|\\.\\([^.]\\|\\..\\)\\).*\\.org$")))
  282. (defun org-test-run-batch-tests ()
  283. "Run all defined tests matching \"\\(org\\|ob\\)\".
  284. Load all test files first."
  285. (interactive)
  286. (let ((org-id-track-globally t)
  287. (org-id-locations-file
  288. (convert-standard-filename
  289. (expand-file-name
  290. "testing/.test-org-id-locations"
  291. org-base-dir))))
  292. (org-test-touch-all-examples)
  293. (org-test-update-id-locations)
  294. (org-test-load)
  295. (ert-run-tests-batch-and-exit "\\(org\\|ob\\)")))
  296. (defun org-test-run-all-tests ()
  297. "Run all defined tests matching \"\\(org\\|ob\\)\".
  298. Load all test files first."
  299. (interactive)
  300. (org-test-touch-all-examples)
  301. (org-test-load)
  302. (ert "\\(org\\|ob\\)"))
  303. (provide 'org-test)
  304. ;;; org-test.el ends here