-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathtest.clj
157 lines (142 loc) · 5.66 KB
/
test.clj
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
;; Authors: Sung Pae <[email protected]>
(ns vim-clojure-static.test
(:require [clojure.edn :as edn]
[clojure.java.io :as io]
[clojure.java.shell :as shell]
[clojure.string :as string]
[clojure.test :as test])
(:import (java.io File)
(java.util List)))
(defmacro with-tempfile
{:requires [File]}
[[tmp-sym] & body]
`(let [~tmp-sym (File/createTempFile "vim-clojure-static" ".tmp")]
(try
~@body
(finally
(.delete ~tmp-sym)))))
(defn vim-exec
"Spit buf into file, then execute vim-expr after Vim loads the file. The
value of vim-expr is evaluated as EDN and returned."
[file buf vim-expr & opts]
(let [{:keys [pre]} (apply hash-map opts)]
(with-tempfile [tmp]
(io/make-parents file)
(spit file buf)
(spit tmp (str "let @x = " vim-expr))
(let [{:keys [exit err]}
(shell/sh "vim" "-N" "-u" "vim/test-runtime.vim"
"-c" (or pre "execute")
"-c" (str "source " tmp)
"-c" (str "call writefile([@x], " (pr-str (str tmp)) ")")
"-c" "quitall!"
file)]
(when-not (zero? exit)
(throw (RuntimeException. ^String err))))
(edn/read-string (slurp tmp)))))
(defn syn-id-names
"Map lines of clojure text to vim synID names at each column as keywords:
(syn-id-names \"foo\" …) -> {\"foo\" [:clojureString :clojureString :clojureString] …}
First parameter is the file that is used to communicate with Vim. The file
is not deleted to allow manual inspection."
[file & lines]
(into {} (map (fn [l ids] [l (mapv keyword ids)])
lines
(vim-exec file (string/join \newline lines) "ClojureSynIDNames()"))))
(defn subfmt
"Extract a subsequence of seq s corresponding to the character positions of
%s in format spec fmt"
[fmt s]
(let [f (seq (format fmt \o001))
i (.indexOf ^List f \o001)]
(->> s
(drop i)
(drop-last (- (count f) i 1)))))
(defmacro defsyntaxtest
"Create a new testing var with tests in the format:
(defsyntaxtest example
[format
[test-string test-predicate
…]]
[\"#\\\"%s\\\"\"
[\"123\" #(every? (partial = :clojureRegexp) %)
…]]
[…])
At runtime the syn-id-names of the strings (which are placed in the format
spec) are passed to their associated predicates. The format spec should
contain a single `%s`."
{:requires [#'test/deftest syn-id-names subfmt]}
[name & body]
(assert (every? (fn [[fmt tests]] (and (string? fmt)
(coll? tests)
(even? (count tests))))
body))
(let [[strings contexts] (reduce (fn [[strings contexts] [fmt tests]]
(let [[ss λs] (apply map list (partition 2 tests))
ss (map #(format fmt %) ss)]
[(concat strings ss)
(conj contexts {:fmt fmt :ss ss :λs λs})]))
[[] []] body)
test-file (str "tmp/" name ".clj")
syntable (gensym "syntable")]
`(test/deftest ~name
(~io/make-parents ~test-file)
(spit ~test-file "")
(let [~syntable (syn-id-names ~test-file ~@strings)]
~@(map (fn [{:keys [fmt ss λs]}]
`(test/testing ~fmt
~@(map (fn [s λ] `(test/is (~λ (subfmt ~fmt (get ~syntable ~s)))))
ss λs)))
contexts)))))
(defmacro defpredicates
"Create two complementary predicate vars, `sym` and `!sym`, which test if
all members of a passed collection are equal to `kw`"
[sym kw]
`(do
(defn ~sym
~(str "Returns true if all elements of coll equal " kw)
{:arglists '~'[coll]}
[coll#]
(every? (partial = ~kw) coll#))
(defn ~(symbol (str \! sym))
~(str "Returns true if any elements of coll do not equal " kw)
{:arglists '~'[coll]}
[coll#]
(boolean (some (partial not= ~kw) coll#)))))
(defmacro with-transform-test
"Copy contents of `in` to a tempfile, execute body with tempfile bound to
tmp-sym, then finally compare the transformed contents of the tempfile with
the contents of `out`.
`in` and `out` are urls that will be passed to clojure.java.io/resource."
{:requires [#'test/testing #'with-tempfile]}
[string {:keys [in out]} [tmp-sym :as tmp-binding] & body]
`(test/testing ~string
(with-tempfile ~tmp-binding
(try
(spit ~tmp-sym (slurp (~io/resource ~in)))
~@body
(catch Throwable e#
(spit ~tmp-sym e#))
(finally
(test/is (= (slurp ~tmp-sym)
(slurp (~io/resource ~out)))))))))
(defmacro test-indent
{:requires [#'with-transform-test]}
[string & opts]
(let [{:keys [in out pre keys]} (apply hash-map opts)
test-file (str "tmp/test-indent-" (string/replace string #"[^\w-]" "-") ".clj")
vim-expr (if keys
(format "TypeKeys(%s)" (string/replace (pr-str keys) "\\\\" "\\"))
"IndentFile()")]
`(with-transform-test ~string
{:in ~in :out ~out}
[tmp#]
;; FIXME: Too much file IO
(~io/make-parents ~test-file)
(spit ~test-file "")
(~vim-exec ~test-file (slurp tmp#) ~vim-expr ~@(when pre [:pre pre]))
(spit tmp# (slurp ~test-file)))))
(defn benchmark [n file buf & exprs]
(vim-exec file buf (format "Benchmark(%d, %s)"
n
(string/join \, (map pr-str exprs)))))