-
Notifications
You must be signed in to change notification settings - Fork 344
/
haskell-test-utils.el
306 lines (259 loc) · 11.5 KB
/
haskell-test-utils.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
;;; haskell-test-utils.el --- Utilities for Haskell Mode tests. -*- lexical-binding: t -*-
;; Copyright © 2016 Arthur Fayzrakhmanov. All rights reserved.
;; This file is part of haskell-mode package.
;; You can contact with authors using GitHub issue tracker:
;; https://github.com/haskell/haskell-mode/issues
;; This file 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, or (at your option)
;; any later version.
;; This file is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;; You should have received a copy of the GNU General Public License
;; along with GNU Emacs; see the file COPYING. If not, write to
;; the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
;; Boston, MA 02110-1301, USA.
;;; Commentary:
;; This package provides utilities for `haskell-mode' tests.
;;; Code:
(require 'cl-lib)
(defun insert-lines (&rest lines)
"Insert all LINES in current buffer."
(let ((ls lines)
(line ""))
(while ls
(setq line (car ls))
(setq ls (cdr ls))
(when ls
(insert line)
(insert "\n")))
(insert line)))
(defmacro with-temp-switch-to-buffer (&rest body)
"Create a temporary buffer and evalute BODY there.
Uses `switch-to-buffer' and evaluates BODY in temp buffer like `progn'.
Seems that `execute-kbd-macro' is not able to correctly execute
keybindings without this."
(declare (indent 0) (debug t))
(let ((temp-buffer (make-symbol "temp-buffer")))
`(let ((,temp-buffer (generate-new-buffer " *temp*")))
;; FIXME: kill-buffer can change current-buffer in some odd cases.
(unwind-protect
(progn
(switch-to-buffer ,temp-buffer)
,@body)
(and (buffer-name ,temp-buffer)
(kill-buffer ,temp-buffer))))))
(defun check-syntax-and-face-match-range (beg end syntax face)
"Check if all charaters between positions BEG and END have
syntax set to SYNTAX and face set to FACE.
If SYNTAX or FACE are set to t then any syntex respective face is
not checked."
(let (all-syntaxes
all-faces
(syntax-classes "-.w_()'\"$\\/<>@!|")
(text (buffer-substring-no-properties beg end)))
(while (< beg end)
(cl-pushnew (char-to-string (aref syntax-classes (syntax-class (syntax-after beg)))) all-syntaxes :test #'equal)
(cl-pushnew (get-text-property beg 'face) all-faces :test #'equal)
(setq beg (1+ beg)))
(unless (eq syntax t)
(should (equal (list text (mapconcat #'identity (sort (mapcar (lambda (syn) (char-to-string syn)) syntax) #'string<) ""))
(list text (mapconcat #'identity (sort all-syntaxes #'string<) "")))))
(unless (eq face t)
(should (equal (list text (list face))
(list text all-faces))))))
(defun check-face-match-range (face n)
(let ((beg (match-beginning n))
(end (match-end n)))
(while (< beg end)
(should (eq face (get-text-property beg 'face)))
(setq beg (1+ beg)))))
(defmacro with-haskell-test-buffer (mode &rest body)
"Run BODY in the context of a new buffer set to `haskell-mode'.
Buffer is named *haskell-mode-buffer*. It is not deleted
after a test as this aids interactive debugging."
(declare (indent 1) (debug t))
`(progn
;; we want to create buffer from scratch so that there are no
;; leftover state from the previous test
(when (get-buffer "*haskell-test-buffer*")
(kill-buffer "*haskell-test-buffer*"))
(save-current-buffer
(set-buffer (get-buffer-create "*haskell-test-buffer*"))
(funcall ,mode)
,@body)))
(defun check-properties (lines-or-contents props &optional mode)
"Check if syntax properties and font-lock properties as set properly.
LINES is a list of strings that will be inserted to a new
buffer. Then PROPS is a list of tripples of (string syntax
face). String is searched for in the buffer and then is checked
if all of its characters have syntax and face. See
`check-syntax-and-face-match-range`."
(with-haskell-test-buffer (or mode #'haskell-mode)
(if (consp lines-or-contents)
(dolist (line lines-or-contents)
(let ((pos (point)))
(insert line "\n")
(save-excursion
;; For some reason font-lock-fontify-region moves the
;; point. I do not think it is guaranteed it should not,
;; but then it might be our fault. Investigate later.
(font-lock-fontify-region pos (point)))))
(insert lines-or-contents)
(font-lock-fontify-buffer))
(goto-char (point-min))
(dolist (prop props)
(cl-destructuring-bind (string syntax face) prop
(let ((case-fold-search nil))
(search-forward string))
(check-syntax-and-face-match-range (match-beginning 0) (match-end 0) syntax face)))))
(defun message-stderr (&rest args)
"Output a message to stderr in batch mode.
ARGS are formatted according to `format'. A newline is automatically appended."
(apply #'message args))
(defun message-stdout (&rest args)
"Output a message to stdout in batch mode.
ARGS are formatted according to `format'. A newline is automatically appended."
(princ (apply #'format args))
(terpri))
(defun read-stdin ()
"Read a line from stdin in batch mode.
A line is read and returned. End of input is signalled by
nil. Newlines are stripped. Last line is returned even if there
is no final newline."
(condition-case nil
(read-from-minibuffer "")
(error nil)))
(defmacro with-script-path-unix (cmdvar func &rest body)
"Temporarily substitute a command line executable.
Creates a temporary executable script and sets CMDVAR to point to
the script. When the script is run it spawns another Emacs
instance and executes function FUNC. Substitution is in effect
throughout BODY.
In FUNC variable `argv' is a list of all arguments that the
script received when invoked. If the FUNC returns a number then
it will be used as exit code for `kill-emacs' function, otherwise
0 will be used."
(declare (indent 2) (debug t))
`(let ((,cmdvar (make-temp-file "haskell-mode-tests-script")))
(with-current-buffer (find-file-noselect ,cmdvar)
(insert "#!/bin/sh\n")
(insert "\":\"; exec \"" invocation-directory invocation-name "\" -Q --batch -l \"$0\" -- \"$@\"\n")
(insert "(setq debug-on-error t)\n")
(insert "(pop argv)\n")
(insert "(setq load-path '" (format "%S" load-path) ")\n")
(insert "(load \"" (symbol-file ',func) "\" nil t)\n")
(insert "(let ((return-value (" (symbol-name ',func) ")))\n")
(insert " (if (numberp return-value)\n")
(insert " (kill-emacs return-value)\n")
(insert " (kill-emacs 0)))\n")
(basic-save-buffer)
(kill-buffer))
(set-file-modes ,cmdvar (string-to-number "700" 8))
(unwind-protect
(progn ,@body)
(delete-file ,cmdvar))))
(defmacro with-script-path-windows (cmdvar func &rest body)
"Temporarily substitute a command line executable.
Creates a temporary executable script and sets CMDVAR to point to
the script. When the script is run it spawns another Emacs
instance and executes function FUNC. Substitution is in effect
throughout BODY.
In FUNC variable `argv' is a list of all arguments that the
script received when invoked. If the FUNC returns a number then
it will be used as exit code for `kill-emacs' function, otherwise
0 will be used."
(declare (indent 2) (debug t))
`(let* ((,cmdvar (make-temp-file "haskell-mode-tests-script" nil ".bat"))
(el (concat (file-name-sans-extension ,cmdvar) ".el")))
(with-current-buffer (find-file-noselect ,cmdvar)
(insert "@\"" invocation-directory invocation-name "\" -Q --batch -l \"%~dpn0.el\" -- %*\n")
(basic-save-buffer)
(kill-buffer))
(with-current-buffer (find-file-noselect el)
(insert "(setq debug-on-error t)\n")
(insert "(pop argv)\n")
(insert "(setq load-path '" (format "%S" load-path) ")\n")
(insert "(load \"" (symbol-file ',func) "\" nil t)\n")
(insert "(let ((return-value (" (symbol-name ',func) ")))\n")
(insert " (if (numberp return-value)\n")
(insert " (kill-emacs return-value)\n")
(insert " (kill-emacs 0)))\n")
(basic-save-buffer)
(kill-buffer))
(unwind-protect
(progn ,@body)
(delete-file ,cmdvar)
(delete-file el))))
(if (equal system-type 'windows-nt)
(defalias 'with-script-path 'with-script-path-windows)
(defalias 'with-script-path 'with-script-path-unix))
(defun create-directory-structure (entries)
(dolist (entry entries)
(cond
((stringp (cdr entry))
(with-current-buffer (find-file-noselect (car entry))
(insert (cdr entry))
(basic-save-buffer)
(kill-buffer)))
((bufferp (cdr entry))
(with-current-buffer (find-file-noselect (car entry))
(insert (with-current-buffer (cdr entry)
(buffer-substring-no-properties (point-min) (point-max))))
(basic-save-buffer)
(kill-buffer)))
(t
(make-directory (car entry))
(let ((default-directory (file-name-as-directory (concat default-directory (car entry)))))
(create-directory-structure (cdr entry)))))))
(defmacro with-temp-dir-structure (entries &rest body)
"Create a temporary directory structure.
ENTRIES is an alist with file or directory names as keys. If
associated value is a string or buffer then a file is created, if
value is an association list then a directory is created
recursively.
Throughout BODY `default-directory' is set to the root of the
hierarchy created.
Whole hierarchy is removed after BODY finishes and value of
`default-directory' is restored."
(declare (indent 2) (debug t))
`(let ((tmpdir (make-temp-name "haskell-mode-test-dir")))
(make-directory tmpdir)
(unwind-protect
(let ((default-directory (file-name-as-directory (concat default-directory tmpdir))))
(create-directory-structure ',entries)
,@body)
(delete-directory tmpdir t))))
(defun haskell-bypass-confirmation (function &rest args)
"Call FUNCTION with ARGS, bypassing all prompts.
This includes both `y-or-n-p' and `yes-or-no-p'.
from `https://emacs.stackexchange.com/questions/19077/how-to-programmatically-answer-yes-to-those-commands-that-prompt-for-a-decisio'"
(haskell-with-advice
((#'y-or-n-p :override (lambda (prompt) t))
(#'yes-or-no-p :override (lambda (prompt) t)))
(apply function args)))
(defmacro haskell-with-advice (adlist &rest body)
"Execute BODY with temporary advice in ADLIST.
Each element of ADLIST should be a list of the form
(SYMBOL WHERE FUNCTION [PROPS])
suitable for passing to `advice-add'. The BODY is wrapped in an
`unwind-protect' form, so the advice will be removed even in the
event of an error or nonlocal exit."
(declare (debug ((&rest (&rest form)) body))
(indent 1))
`(progn
,@(mapcar (lambda (adform)
(cons 'advice-add adform))
adlist)
(unwind-protect (progn ,@body)
,@(mapcar (lambda (adform)
`(advice-remove ,(car adform) ,(nth 2 adform)))
adlist))))
(defun haskell-unconditional-kill-buffer (buffer)
"Buffer names are passed"
(when (buffer-live-p (get-buffer buffer))
(haskell-bypass-confirmation #'kill-buffer buffer)))
(provide 'haskell-test-utils)
;;; haskell-test-utils.el ends here