srfi-64.upstream.scm 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981
  1. ;; Copyright (c) 2005, 2006 Per Bothner
  2. ;;
  3. ;; Permission is hereby granted, free of charge, to any person
  4. ;; obtaining a copy of this software and associated documentation
  5. ;; files (the "Software"), to deal in the Software without
  6. ;; restriction, including without limitation the rights to use, copy,
  7. ;; modify, merge, publish, distribute, sublicense, and/or sell copies
  8. ;; of the Software, and to permit persons to whom the Software is
  9. ;; furnished to do so, subject to the following conditions:
  10. ;;
  11. ;; The above copyright notice and this permission notice shall be
  12. ;; included in all copies or substantial portions of the Software.
  13. ;;
  14. ;; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  15. ;; EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  16. ;; MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  17. ;; NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
  18. ;; BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
  19. ;; ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
  20. ;; CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  21. ;; SOFTWARE.
  22. (cond-expand
  23. (chicken
  24. (require-extension syntax-case))
  25. (guile
  26. (use-modules (srfi srfi-9)
  27. ;;(srfi srfi-34) (srfi srfi-35) - not in Guile 1.6.7
  28. (srfi srfi-39)))
  29. (sisc
  30. (require-extension (srfi 9 34 35 39)))
  31. (kawa
  32. (module-compile-options warn-undefined-variable: #t
  33. warn-invoke-unknown-method: #t)
  34. (provide 'srfi-64)
  35. (provide 'testing)
  36. (require 'srfi-34)
  37. (require 'srfi-35))
  38. (else ()
  39. ))
  40. (cond-expand
  41. (kawa
  42. (define-syntax %test-export
  43. (syntax-rules ()
  44. ((%test-export test-begin . other-names)
  45. (module-export %test-begin . other-names)))))
  46. (else
  47. (define-syntax %test-export
  48. (syntax-rules ()
  49. ((%test-export . names) (if #f #f))))))
  50. ;; List of exported names
  51. (%test-export
  52. test-begin ;; must be listed first, since in Kawa (at least) it is "magic".
  53. test-end test-assert test-eqv test-eq test-equal
  54. test-approximate test-assert test-error test-apply test-with-runner
  55. test-match-nth test-match-all test-match-any test-match-name
  56. test-skip test-expect-fail test-read-eval-string
  57. test-runner-group-path test-group-with-cleanup
  58. test-result-ref test-result-set! test-result-clear test-result-remove
  59. test-result-kind test-passed?
  60. test-log-to-file
  61. ; Misc test-runner functions
  62. test-runner? test-runner-reset test-runner-null
  63. test-runner-simple test-runner-current test-runner-factory test-runner-get
  64. test-runner-create test-runner-test-name
  65. ;; test-runner field setter and getter functions - see %test-record-define:
  66. test-runner-pass-count test-runner-pass-count!
  67. test-runner-fail-count test-runner-fail-count!
  68. test-runner-xpass-count test-runner-xpass-count!
  69. test-runner-xfail-count test-runner-xfail-count!
  70. test-runner-skip-count test-runner-skip-count!
  71. test-runner-group-stack test-runner-group-stack!
  72. test-runner-on-test-begin test-runner-on-test-begin!
  73. test-runner-on-test-end test-runner-on-test-end!
  74. test-runner-on-group-begin test-runner-on-group-begin!
  75. test-runner-on-group-end test-runner-on-group-end!
  76. test-runner-on-final test-runner-on-final!
  77. test-runner-on-bad-count test-runner-on-bad-count!
  78. test-runner-on-bad-end-name test-runner-on-bad-end-name!
  79. test-result-alist test-result-alist!
  80. test-runner-aux-value test-runner-aux-value!
  81. ;; default/simple call-back functions, used in default test-runner,
  82. ;; but can be called to construct more complex ones.
  83. test-on-group-begin-simple test-on-group-end-simple
  84. test-on-bad-count-simple test-on-bad-end-name-simple
  85. test-on-final-simple test-on-test-end-simple
  86. test-on-final-simple)
  87. (cond-expand
  88. (srfi-9
  89. (define-syntax %test-record-define
  90. (syntax-rules ()
  91. ((%test-record-define alloc runner? (name index setter getter) ...)
  92. (define-record-type test-runner
  93. (alloc)
  94. runner?
  95. (name setter getter) ...)))))
  96. (else
  97. (define %test-runner-cookie (list "test-runner"))
  98. (define-syntax %test-record-define
  99. (syntax-rules ()
  100. ((%test-record-define alloc runner? (name index getter setter) ...)
  101. (begin
  102. (define (runner? obj)
  103. (and (vector? obj)
  104. (> (vector-length obj) 1)
  105. (eq (vector-ref obj 0) %test-runner-cookie)))
  106. (define (alloc)
  107. (let ((runner (make-vector 22)))
  108. (vector-set! runner 0 %test-runner-cookie)
  109. runner))
  110. (begin
  111. (define (getter runner)
  112. (vector-ref runner index)) ...)
  113. (begin
  114. (define (setter runner value)
  115. (vector-set! runner index value)) ...)))))))
  116. (%test-record-define
  117. %test-runner-alloc test-runner?
  118. ;; Cumulate count of all tests that have passed and were expected to.
  119. (pass-count 1 test-runner-pass-count test-runner-pass-count!)
  120. (fail-count 2 test-runner-fail-count test-runner-fail-count!)
  121. (xpass-count 3 test-runner-xpass-count test-runner-xpass-count!)
  122. (xfail-count 4 test-runner-xfail-count test-runner-xfail-count!)
  123. (skip-count 5 test-runner-skip-count test-runner-skip-count!)
  124. (skip-list 6 %test-runner-skip-list %test-runner-skip-list!)
  125. (fail-list 7 %test-runner-fail-list %test-runner-fail-list!)
  126. ;; Normally #t, except when in a test-apply.
  127. (run-list 8 %test-runner-run-list %test-runner-run-list!)
  128. (skip-save 9 %test-runner-skip-save %test-runner-skip-save!)
  129. (fail-save 10 %test-runner-fail-save %test-runner-fail-save!)
  130. (group-stack 11 test-runner-group-stack test-runner-group-stack!)
  131. (on-test-begin 12 test-runner-on-test-begin test-runner-on-test-begin!)
  132. (on-test-end 13 test-runner-on-test-end test-runner-on-test-end!)
  133. ;; Call-back when entering a group. Takes (runner suite-name count).
  134. (on-group-begin 14 test-runner-on-group-begin test-runner-on-group-begin!)
  135. ;; Call-back when leaving a group.
  136. (on-group-end 15 test-runner-on-group-end test-runner-on-group-end!)
  137. ;; Call-back when leaving the outermost group.
  138. (on-final 16 test-runner-on-final test-runner-on-final!)
  139. ;; Call-back when expected number of tests was wrong.
  140. (on-bad-count 17 test-runner-on-bad-count test-runner-on-bad-count!)
  141. ;; Call-back when name in test=end doesn't match test-begin.
  142. (on-bad-end-name 18 test-runner-on-bad-end-name test-runner-on-bad-end-name!)
  143. ;; Cumulate count of all tests that have been done.
  144. (total-count 19 %test-runner-total-count %test-runner-total-count!)
  145. ;; Stack (list) of (count-at-start . expected-count):
  146. (count-list 20 %test-runner-count-list %test-runner-count-list!)
  147. (result-alist 21 test-result-alist test-result-alist!)
  148. ;; Field can be used by test-runner for any purpose.
  149. ;; test-runner-simple uses it for a log file.
  150. (aux-value 22 test-runner-aux-value test-runner-aux-value!)
  151. )
  152. (define (test-runner-reset runner)
  153. (test-runner-pass-count! runner 0)
  154. (test-runner-fail-count! runner 0)
  155. (test-runner-xpass-count! runner 0)
  156. (test-runner-xfail-count! runner 0)
  157. (test-runner-skip-count! runner 0)
  158. (%test-runner-total-count! runner 0)
  159. (%test-runner-count-list! runner '())
  160. (%test-runner-run-list! runner #t)
  161. (%test-runner-skip-list! runner '())
  162. (%test-runner-fail-list! runner '())
  163. (%test-runner-skip-save! runner '())
  164. (%test-runner-fail-save! runner '())
  165. (test-runner-group-stack! runner '()))
  166. (define (test-runner-group-path runner)
  167. (reverse (test-runner-group-stack runner)))
  168. (define (%test-null-callback runner) #f)
  169. (define (test-runner-null)
  170. (let ((runner (%test-runner-alloc)))
  171. (test-runner-reset runner)
  172. (test-runner-on-group-begin! runner (lambda (runner name count) #f))
  173. (test-runner-on-group-end! runner %test-null-callback)
  174. (test-runner-on-final! runner %test-null-callback)
  175. (test-runner-on-test-begin! runner %test-null-callback)
  176. (test-runner-on-test-end! runner %test-null-callback)
  177. (test-runner-on-bad-count! runner (lambda (runner count expected) #f))
  178. (test-runner-on-bad-end-name! runner (lambda (runner begin end) #f))
  179. runner))
  180. ;; Not part of the specification. FIXME
  181. ;; Controls whether a log file is generated.
  182. (define test-log-to-file #t)
  183. (define (test-runner-simple)
  184. (let ((runner (%test-runner-alloc)))
  185. (test-runner-reset runner)
  186. (test-runner-on-group-begin! runner test-on-group-begin-simple)
  187. (test-runner-on-group-end! runner test-on-group-end-simple)
  188. (test-runner-on-final! runner test-on-final-simple)
  189. (test-runner-on-test-begin! runner test-on-test-begin-simple)
  190. (test-runner-on-test-end! runner test-on-test-end-simple)
  191. (test-runner-on-bad-count! runner test-on-bad-count-simple)
  192. (test-runner-on-bad-end-name! runner test-on-bad-end-name-simple)
  193. runner))
  194. (cond-expand
  195. (srfi-39
  196. (define test-runner-current (make-parameter #f))
  197. (define test-runner-factory (make-parameter test-runner-simple)))
  198. (else
  199. (define %test-runner-current #f)
  200. (define-syntax test-runner-current
  201. (syntax-rules ()
  202. ((test-runner-current)
  203. %test-runner-current)
  204. ((test-runner-current runner)
  205. (set! %test-runner-current runner))))
  206. (define %test-runner-factory test-runner-simple)
  207. (define-syntax test-runner-factory
  208. (syntax-rules ()
  209. ((test-runner-factory)
  210. %test-runner-factory)
  211. ((test-runner-factory runner)
  212. (set! %test-runner-factory runner))))))
  213. ;; A safer wrapper to test-runner-current.
  214. (define (test-runner-get)
  215. (let ((r (test-runner-current)))
  216. (if (not r)
  217. (cond-expand
  218. (srfi-23 (error "test-runner not initialized - test-begin missing?"))
  219. (else #t)))
  220. r))
  221. (define (%test-specificier-matches spec runner)
  222. (spec runner))
  223. (define (test-runner-create)
  224. ((test-runner-factory)))
  225. (define (%test-any-specifier-matches list runner)
  226. (let ((result #f))
  227. (let loop ((l list))
  228. (cond ((null? l) result)
  229. (else
  230. (if (%test-specificier-matches (car l) runner)
  231. (set! result #t))
  232. (loop (cdr l)))))))
  233. ;; Returns #f, #t, or 'xfail.
  234. (define (%test-should-execute runner)
  235. (let ((run (%test-runner-run-list runner)))
  236. (cond ((or
  237. (not (or (eqv? run #t)
  238. (%test-any-specifier-matches run runner)))
  239. (%test-any-specifier-matches
  240. (%test-runner-skip-list runner)
  241. runner))
  242. (test-result-set! runner 'result-kind 'skip)
  243. #f)
  244. ((%test-any-specifier-matches
  245. (%test-runner-fail-list runner)
  246. runner)
  247. (test-result-set! runner 'result-kind 'xfail)
  248. 'xfail)
  249. (else #t))))
  250. (define (%test-begin suite-name count)
  251. (if (not (test-runner-current))
  252. (test-runner-current (test-runner-create)))
  253. (let ((runner (test-runner-current)))
  254. ((test-runner-on-group-begin runner) runner suite-name count)
  255. (%test-runner-skip-save! runner
  256. (cons (%test-runner-skip-list runner)
  257. (%test-runner-skip-save runner)))
  258. (%test-runner-fail-save! runner
  259. (cons (%test-runner-fail-list runner)
  260. (%test-runner-fail-save runner)))
  261. (%test-runner-count-list! runner
  262. (cons (cons (%test-runner-total-count runner)
  263. count)
  264. (%test-runner-count-list runner)))
  265. (test-runner-group-stack! runner (cons suite-name
  266. (test-runner-group-stack runner)))))
  267. (cond-expand
  268. (kawa
  269. ;; Kawa has test-begin built in, implemented as:
  270. ;; (begin
  271. ;; (cond-expand (srfi-64 #!void) (else (require 'srfi-64)))
  272. ;; (%test-begin suite-name [count]))
  273. ;; This puts test-begin but only test-begin in the default environment.,
  274. ;; which makes normal test suites loadable without non-portable commands.
  275. )
  276. (else
  277. (define-syntax test-begin
  278. (syntax-rules ()
  279. ((test-begin suite-name)
  280. (%test-begin suite-name #f))
  281. ((test-begin suite-name count)
  282. (%test-begin suite-name count))))))
  283. (define (test-on-group-begin-simple runner suite-name count)
  284. (if (null? (test-runner-group-stack runner))
  285. (begin
  286. (display "%%%% Starting test ")
  287. (display suite-name)
  288. (if test-log-to-file
  289. (let* ((log-file-name
  290. (if (string? test-log-to-file) test-log-to-file
  291. (string-append suite-name ".log")))
  292. (log-file
  293. (cond-expand (mzscheme
  294. (open-output-file log-file-name 'truncate/replace))
  295. (guile-2
  296. (with-fluids ((%default-port-encoding
  297. "UTF-8"))
  298. (let ((p (open-output-file log-file-name)))
  299. (setvbuf p _IOLBF)
  300. p)))
  301. (else (open-output-file log-file-name)))))
  302. (display "%%%% Starting test " log-file)
  303. (display suite-name log-file)
  304. (newline log-file)
  305. (test-runner-aux-value! runner log-file)
  306. (display " (Writing full log to \"")
  307. (display log-file-name)
  308. (display "\")")))
  309. (newline)))
  310. (let ((log (test-runner-aux-value runner)))
  311. (if (output-port? log)
  312. (begin
  313. (display "Group begin: " log)
  314. (display suite-name log)
  315. (newline log))))
  316. #f)
  317. (define (test-on-group-end-simple runner)
  318. (let ((log (test-runner-aux-value runner)))
  319. (if (output-port? log)
  320. (begin
  321. (display "Group end: " log)
  322. (display (car (test-runner-group-stack runner)) log)
  323. (newline log))))
  324. #f)
  325. (define (%test-on-bad-count-write runner count expected-count port)
  326. (display "*** Total number of tests was " port)
  327. (display count port)
  328. (display " but should be " port)
  329. (display expected-count port)
  330. (display ". ***" port)
  331. (newline port)
  332. (display "*** Discrepancy indicates testsuite error or exceptions. ***" port)
  333. (newline port))
  334. (define (test-on-bad-count-simple runner count expected-count)
  335. (%test-on-bad-count-write runner count expected-count (current-output-port))
  336. (let ((log (test-runner-aux-value runner)))
  337. (if (output-port? log)
  338. (%test-on-bad-count-write runner count expected-count log))))
  339. (define (test-on-bad-end-name-simple runner begin-name end-name)
  340. (let ((msg (string-append (%test-format-line runner) "test-end " begin-name
  341. " does not match test-begin " end-name)))
  342. (cond-expand
  343. (srfi-23 (error msg))
  344. (else (display msg) (newline)))))
  345. (define (%test-final-report1 value label port)
  346. (if (> value 0)
  347. (begin
  348. (display label port)
  349. (display value port)
  350. (newline port))))
  351. (define (%test-final-report-simple runner port)
  352. (%test-final-report1 (test-runner-pass-count runner)
  353. "# of expected passes " port)
  354. (%test-final-report1 (test-runner-xfail-count runner)
  355. "# of expected failures " port)
  356. (%test-final-report1 (test-runner-xpass-count runner)
  357. "# of unexpected successes " port)
  358. (%test-final-report1 (test-runner-fail-count runner)
  359. "# of unexpected failures " port)
  360. (%test-final-report1 (test-runner-skip-count runner)
  361. "# of skipped tests " port))
  362. (define (test-on-final-simple runner)
  363. (%test-final-report-simple runner (current-output-port))
  364. (let ((log (test-runner-aux-value runner)))
  365. (if (output-port? log)
  366. (%test-final-report-simple runner log))))
  367. (define (%test-format-line runner)
  368. (let* ((line-info (test-result-alist runner))
  369. (source-file (assq 'source-file line-info))
  370. (source-line (assq 'source-line line-info))
  371. (file (if source-file (cdr source-file) "")))
  372. (if source-line
  373. (string-append file ":"
  374. (number->string (cdr source-line)) ": ")
  375. "")))
  376. (define (%test-end suite-name line-info)
  377. (let* ((r (test-runner-get))
  378. (groups (test-runner-group-stack r))
  379. (line (%test-format-line r)))
  380. (test-result-alist! r line-info)
  381. (if (null? groups)
  382. (let ((msg (string-append line "test-end not in a group")))
  383. (cond-expand
  384. (srfi-23 (error msg))
  385. (else (display msg) (newline)))))
  386. (if (and suite-name (not (equal? suite-name (car groups))))
  387. ((test-runner-on-bad-end-name r) r suite-name (car groups)))
  388. (let* ((count-list (%test-runner-count-list r))
  389. (expected-count (cdar count-list))
  390. (saved-count (caar count-list))
  391. (group-count (- (%test-runner-total-count r) saved-count)))
  392. (if (and expected-count
  393. (not (= expected-count group-count)))
  394. ((test-runner-on-bad-count r) r group-count expected-count))
  395. ((test-runner-on-group-end r) r)
  396. (test-runner-group-stack! r (cdr (test-runner-group-stack r)))
  397. (%test-runner-skip-list! r (car (%test-runner-skip-save r)))
  398. (%test-runner-skip-save! r (cdr (%test-runner-skip-save r)))
  399. (%test-runner-fail-list! r (car (%test-runner-fail-save r)))
  400. (%test-runner-fail-save! r (cdr (%test-runner-fail-save r)))
  401. (%test-runner-count-list! r (cdr count-list))
  402. (if (null? (test-runner-group-stack r))
  403. ((test-runner-on-final r) r)))))
  404. (define-syntax test-group
  405. (syntax-rules ()
  406. ((test-group suite-name . body)
  407. (let ((r (test-runner-current)))
  408. ;; Ideally should also set line-number, if available.
  409. (test-result-alist! r (list (cons 'test-name suite-name)))
  410. (if (%test-should-execute r)
  411. (dynamic-wind
  412. (lambda () (test-begin suite-name))
  413. (lambda () . body)
  414. (lambda () (test-end suite-name))))))))
  415. (define-syntax test-group-with-cleanup
  416. (syntax-rules ()
  417. ((test-group-with-cleanup suite-name form cleanup-form)
  418. (test-group suite-name
  419. (dynamic-wind
  420. (lambda () #f)
  421. (lambda () form)
  422. (lambda () cleanup-form))))
  423. ((test-group-with-cleanup suite-name cleanup-form)
  424. (test-group-with-cleanup suite-name #f cleanup-form))
  425. ((test-group-with-cleanup suite-name form1 form2 form3 . rest)
  426. (test-group-with-cleanup suite-name (begin form1 form2) form3 . rest))))
  427. (define (test-on-test-begin-simple runner)
  428. (let ((log (test-runner-aux-value runner)))
  429. (if (output-port? log)
  430. (let* ((results (test-result-alist runner))
  431. (source-file (assq 'source-file results))
  432. (source-line (assq 'source-line results))
  433. (source-form (assq 'source-form results))
  434. (test-name (assq 'test-name results)))
  435. (display "Test begin:" log)
  436. (newline log)
  437. (if test-name (%test-write-result1 test-name log))
  438. (if source-file (%test-write-result1 source-file log))
  439. (if source-line (%test-write-result1 source-line log))
  440. (if source-file (%test-write-result1 source-form log))))))
  441. (define-syntax test-result-ref
  442. (syntax-rules ()
  443. ((test-result-ref runner pname)
  444. (test-result-ref runner pname #f))
  445. ((test-result-ref runner pname default)
  446. (let ((p (assq pname (test-result-alist runner))))
  447. (if p (cdr p) default)))))
  448. (define (test-on-test-end-simple runner)
  449. (let ((log (test-runner-aux-value runner))
  450. (kind (test-result-ref runner 'result-kind)))
  451. (if (memq kind '(fail xpass))
  452. (let* ((results (test-result-alist runner))
  453. (source-file (assq 'source-file results))
  454. (source-line (assq 'source-line results))
  455. (test-name (assq 'test-name results)))
  456. (if (or source-file source-line)
  457. (begin
  458. (if source-file (display (cdr source-file)))
  459. (display ":")
  460. (if source-line (display (cdr source-line)))
  461. (display ": ")))
  462. (display (if (eq? kind 'xpass) "XPASS" "FAIL"))
  463. (if test-name
  464. (begin
  465. (display " ")
  466. (display (cdr test-name))))
  467. (newline)))
  468. (if (output-port? log)
  469. (begin
  470. (display "Test end:" log)
  471. (newline log)
  472. (let loop ((list (test-result-alist runner)))
  473. (if (pair? list)
  474. (let ((pair (car list)))
  475. ;; Write out properties not written out by on-test-begin.
  476. (if (not (memq (car pair)
  477. '(test-name source-file source-line source-form)))
  478. (%test-write-result1 pair log))
  479. (loop (cdr list)))))))))
  480. (define (%test-write-result1 pair port)
  481. (display " " port)
  482. (display (car pair) port)
  483. (display ": " port)
  484. (write (cdr pair) port)
  485. (newline port))
  486. (define (test-result-set! runner pname value)
  487. (let* ((alist (test-result-alist runner))
  488. (p (assq pname alist)))
  489. (if p
  490. (set-cdr! p value)
  491. (test-result-alist! runner (cons (cons pname value) alist)))))
  492. (define (test-result-clear runner)
  493. (test-result-alist! runner '()))
  494. (define (test-result-remove runner pname)
  495. (let* ((alist (test-result-alist runner))
  496. (p (assq pname alist)))
  497. (if p
  498. (test-result-alist! runner
  499. (let loop ((r alist))
  500. (if (eq? r p) (cdr r)
  501. (cons (car r) (loop (cdr r)))))))))
  502. (define (test-result-kind . rest)
  503. (let ((runner (if (pair? rest) (car rest) (test-runner-current))))
  504. (test-result-ref runner 'result-kind)))
  505. (define (test-passed? . rest)
  506. (let ((runner (if (pair? rest) (car rest) (test-runner-get))))
  507. (memq (test-result-ref runner 'result-kind) '(pass xpass))))
  508. (define (%test-report-result)
  509. (let* ((r (test-runner-get))
  510. (result-kind (test-result-kind r)))
  511. (case result-kind
  512. ((pass)
  513. (test-runner-pass-count! r (+ 1 (test-runner-pass-count r))))
  514. ((fail)
  515. (test-runner-fail-count! r (+ 1 (test-runner-fail-count r))))
  516. ((xpass)
  517. (test-runner-xpass-count! r (+ 1 (test-runner-xpass-count r))))
  518. ((xfail)
  519. (test-runner-xfail-count! r (+ 1 (test-runner-xfail-count r))))
  520. (else
  521. (test-runner-skip-count! r (+ 1 (test-runner-skip-count r)))))
  522. (%test-runner-total-count! r (+ 1 (%test-runner-total-count r)))
  523. ((test-runner-on-test-end r) r)))
  524. (cond-expand
  525. (guile
  526. (define-syntax %test-evaluate-with-catch
  527. (syntax-rules ()
  528. ((%test-evaluate-with-catch test-expression)
  529. (catch #t
  530. (lambda () test-expression)
  531. (lambda (key . args) #f)
  532. (lambda (key . args)
  533. (display-backtrace (make-stack #t) (current-error-port))))))))
  534. (kawa
  535. (define-syntax %test-evaluate-with-catch
  536. (syntax-rules ()
  537. ((%test-evaluate-with-catch test-expression)
  538. (try-catch test-expression
  539. (ex <java.lang.Throwable>
  540. (test-result-set! (test-runner-current) 'actual-error ex)
  541. #f))))))
  542. (srfi-34
  543. (define-syntax %test-evaluate-with-catch
  544. (syntax-rules ()
  545. ((%test-evaluate-with-catch test-expression)
  546. (guard (err (else #f)) test-expression)))))
  547. (chicken
  548. (define-syntax %test-evaluate-with-catch
  549. (syntax-rules ()
  550. ((%test-evaluate-with-catch test-expression)
  551. (condition-case test-expression (ex () #f))))))
  552. (else
  553. (define-syntax %test-evaluate-with-catch
  554. (syntax-rules ()
  555. ((%test-evaluate-with-catch test-expression)
  556. test-expression)))))
  557. (cond-expand
  558. ((or kawa mzscheme)
  559. (cond-expand
  560. (mzscheme
  561. (define-for-syntax (%test-syntax-file form)
  562. (let ((source (syntax-source form)))
  563. (cond ((string? source) file)
  564. ((path? source) (path->string source))
  565. (else #f)))))
  566. (kawa
  567. (define (%test-syntax-file form)
  568. (syntax-source form))))
  569. (define-for-syntax (%test-source-line2 form)
  570. (let* ((line (syntax-line form))
  571. (file (%test-syntax-file form))
  572. (line-pair (if line (list (cons 'source-line line)) '())))
  573. (cons (cons 'source-form (syntax-object->datum form))
  574. (if file (cons (cons 'source-file file) line-pair) line-pair)))))
  575. (else
  576. (define (%test-source-line2 form)
  577. '())))
  578. (define (%test-on-test-begin r)
  579. (%test-should-execute r)
  580. ((test-runner-on-test-begin r) r)
  581. (not (eq? 'skip (test-result-ref r 'result-kind))))
  582. (define (%test-on-test-end r result)
  583. (test-result-set! r 'result-kind
  584. (if (eq? (test-result-ref r 'result-kind) 'xfail)
  585. (if result 'xpass 'xfail)
  586. (if result 'pass 'fail))))
  587. (define (test-runner-test-name runner)
  588. (test-result-ref runner 'test-name ""))
  589. (define-syntax %test-comp2body
  590. (syntax-rules ()
  591. ((%test-comp2body r comp expected expr)
  592. (let ()
  593. (if (%test-on-test-begin r)
  594. (let ((exp expected))
  595. (test-result-set! r 'expected-value exp)
  596. (let ((res (%test-evaluate-with-catch expr)))
  597. (test-result-set! r 'actual-value res)
  598. (%test-on-test-end r (comp exp res)))))
  599. (%test-report-result)))))
  600. (define (%test-approximimate= error)
  601. (lambda (value expected)
  602. (and (>= value (- expected error))
  603. (<= value (+ expected error)))))
  604. (define-syntax %test-comp1body
  605. (syntax-rules ()
  606. ((%test-comp1body r expr)
  607. (let ()
  608. (if (%test-on-test-begin r)
  609. (let ()
  610. (let ((res (%test-evaluate-with-catch expr)))
  611. (test-result-set! r 'actual-value res)
  612. (%test-on-test-end r res))))
  613. (%test-report-result)))))
  614. (cond-expand
  615. ((or kawa mzscheme)
  616. ;; Should be made to work for any Scheme with syntax-case
  617. ;; However, I haven't gotten the quoting working. FIXME.
  618. (define-syntax test-end
  619. (lambda (x)
  620. (syntax-case (list x (list 'quote (%test-source-line2 x))) ()
  621. (((mac suite-name) line)
  622. (syntax
  623. (%test-end suite-name line)))
  624. (((mac) line)
  625. (syntax
  626. (%test-end #f line))))))
  627. (define-syntax test-assert
  628. (lambda (x)
  629. (syntax-case (list x (list 'quote (%test-source-line2 x))) ()
  630. (((mac tname expr) line)
  631. (syntax
  632. (let* ((r (test-runner-get))
  633. (name tname))
  634. (test-result-alist! r (cons (cons 'test-name tname) line))
  635. (%test-comp1body r expr))))
  636. (((mac expr) line)
  637. (syntax
  638. (let* ((r (test-runner-get)))
  639. (test-result-alist! r line)
  640. (%test-comp1body r expr)))))))
  641. (define-for-syntax (%test-comp2 comp x)
  642. (syntax-case (list x (list 'quote (%test-source-line2 x)) comp) ()
  643. (((mac tname expected expr) line comp)
  644. (syntax
  645. (let* ((r (test-runner-get))
  646. (name tname))
  647. (test-result-alist! r (cons (cons 'test-name tname) line))
  648. (%test-comp2body r comp expected expr))))
  649. (((mac expected expr) line comp)
  650. (syntax
  651. (let* ((r (test-runner-get)))
  652. (test-result-alist! r line)
  653. (%test-comp2body r comp expected expr))))))
  654. (define-syntax test-eqv
  655. (lambda (x) (%test-comp2 (syntax eqv?) x)))
  656. (define-syntax test-eq
  657. (lambda (x) (%test-comp2 (syntax eq?) x)))
  658. (define-syntax test-equal
  659. (lambda (x) (%test-comp2 (syntax equal?) x)))
  660. (define-syntax test-approximate ;; FIXME - needed for non-Kawa
  661. (lambda (x)
  662. (syntax-case (list x (list 'quote (%test-source-line2 x))) ()
  663. (((mac tname expected expr error) line)
  664. (syntax
  665. (let* ((r (test-runner-get))
  666. (name tname))
  667. (test-result-alist! r (cons (cons 'test-name tname) line))
  668. (%test-comp2body r (%test-approximimate= error) expected expr))))
  669. (((mac expected expr error) line)
  670. (syntax
  671. (let* ((r (test-runner-get)))
  672. (test-result-alist! r line)
  673. (%test-comp2body r (%test-approximimate= error) expected expr))))))))
  674. (else
  675. (define-syntax test-end
  676. (syntax-rules ()
  677. ((test-end)
  678. (%test-end #f '()))
  679. ((test-end suite-name)
  680. (%test-end suite-name '()))))
  681. (define-syntax test-assert
  682. (syntax-rules ()
  683. ((test-assert tname test-expression)
  684. (let* ((r (test-runner-get))
  685. (name tname))
  686. (test-result-alist! r '((test-name . tname)))
  687. (%test-comp1body r test-expression)))
  688. ((test-assert test-expression)
  689. (let* ((r (test-runner-get)))
  690. (test-result-alist! r '())
  691. (%test-comp1body r test-expression)))))
  692. (define-syntax %test-comp2
  693. (syntax-rules ()
  694. ((%test-comp2 comp tname expected expr)
  695. (let* ((r (test-runner-get))
  696. (name tname))
  697. (test-result-alist! r (list (cons 'test-name tname)))
  698. (%test-comp2body r comp expected expr)))
  699. ((%test-comp2 comp expected expr)
  700. (let* ((r (test-runner-get)))
  701. (test-result-alist! r '())
  702. (%test-comp2body r comp expected expr)))))
  703. (define-syntax test-equal
  704. (syntax-rules ()
  705. ((test-equal . rest)
  706. (%test-comp2 equal? . rest))))
  707. (define-syntax test-eqv
  708. (syntax-rules ()
  709. ((test-eqv . rest)
  710. (%test-comp2 eqv? . rest))))
  711. (define-syntax test-eq
  712. (syntax-rules ()
  713. ((test-eq . rest)
  714. (%test-comp2 eq? . rest))))
  715. (define-syntax test-approximate
  716. (syntax-rules ()
  717. ((test-approximate tname expected expr error)
  718. (%test-comp2 (%test-approximimate= error) tname expected expr))
  719. ((test-approximate expected expr error)
  720. (%test-comp2 (%test-approximimate= error) expected expr))))))
  721. (cond-expand
  722. (guile
  723. (define-syntax %test-error
  724. (syntax-rules ()
  725. ((%test-error r etype expr)
  726. (%test-comp1body r (catch #t (lambda () expr) (lambda (key . args) #t)))))))
  727. (mzscheme
  728. (define-syntax %test-error
  729. (syntax-rules ()
  730. ((%test-error r etype expr)
  731. (%test-comp1body r (with-handlers (((lambda (h) #t) (lambda (h) #t)))
  732. (let ()
  733. (test-result-set! r 'actual-value expr)
  734. #f)))))))
  735. (chicken
  736. (define-syntax %test-error
  737. (syntax-rules ()
  738. ((%test-error r etype expr)
  739. (%test-comp1body r (condition-case expr (ex () #t)))))))
  740. (kawa
  741. (define-syntax %test-error
  742. (syntax-rules ()
  743. ((%test-error r etype expr)
  744. (let ()
  745. (if (%test-on-test-begin r)
  746. (let ((et etype))
  747. (test-result-set! r 'expected-error et)
  748. (%test-on-test-end r
  749. (try-catch
  750. (let ()
  751. (test-result-set! r 'actual-value expr)
  752. #f)
  753. (ex <java.lang.Throwable>
  754. (test-result-set! r 'actual-error ex)
  755. (cond ((and (instance? et <gnu.bytecode.ClassType>)
  756. (gnu.bytecode.ClassType:isSubclass et <java.lang.Throwable>))
  757. (instance? ex et))
  758. (else #t)))))
  759. (%test-report-result))))))))
  760. ((and srfi-34 srfi-35)
  761. (define-syntax %test-error
  762. (syntax-rules ()
  763. ((%test-error r etype expr)
  764. (%test-comp1body r (guard (ex ((condition-type? etype)
  765. (and (condition? ex) (condition-has-type? ex etype)))
  766. ((procedure? etype)
  767. (etype ex))
  768. ((equal? type #t)
  769. #t)
  770. (else #t))
  771. expr))))))
  772. (srfi-34
  773. (define-syntax %test-error
  774. (syntax-rules ()
  775. ((%test-error r etype expr)
  776. (%test-comp1body r (guard (ex (else #t)) expr))))))
  777. (else
  778. (define-syntax %test-error
  779. (syntax-rules ()
  780. ((%test-error r etype expr)
  781. (begin
  782. ((test-runner-on-test-begin r) r)
  783. (test-result-set! r 'result-kind 'skip)
  784. (%test-report-result)))))))
  785. (cond-expand
  786. ((or kawa mzscheme)
  787. (define-syntax test-error
  788. (lambda (x)
  789. (syntax-case (list x (list 'quote (%test-source-line2 x))) ()
  790. (((mac tname etype expr) line)
  791. (syntax
  792. (let* ((r (test-runner-get))
  793. (name tname))
  794. (test-result-alist! r (cons (cons 'test-name tname) line))
  795. (%test-error r etype expr))))
  796. (((mac etype expr) line)
  797. (syntax
  798. (let* ((r (test-runner-get)))
  799. (test-result-alist! r line)
  800. (%test-error r etype expr))))
  801. (((mac expr) line)
  802. (syntax
  803. (let* ((r (test-runner-get)))
  804. (test-result-alist! r line)
  805. (%test-error r #t expr))))))))
  806. (else
  807. (define-syntax test-error
  808. (syntax-rules ()
  809. ((test-error name etype expr)
  810. (test-assert name (%test-error etype expr)))
  811. ((test-error etype expr)
  812. (test-assert (%test-error etype expr)))
  813. ((test-error expr)
  814. (test-assert (%test-error #t expr)))))))
  815. (define (test-apply first . rest)
  816. (if (test-runner? first)
  817. (test-with-runner first (apply test-apply rest))
  818. (let ((r (test-runner-current)))
  819. (if r
  820. (let ((run-list (%test-runner-run-list r)))
  821. (cond ((null? rest)
  822. (%test-runner-run-list! r (reverse! run-list))
  823. (first)) ;; actually apply procedure thunk
  824. (else
  825. (%test-runner-run-list!
  826. r
  827. (if (eq? run-list #t) (list first) (cons first run-list)))
  828. (apply test-apply rest)
  829. (%test-runner-run-list! r run-list))))
  830. (let ((r (test-runner-create)))
  831. (test-with-runner r (apply test-apply first rest))
  832. ((test-runner-on-final r) r))))))
  833. (define-syntax test-with-runner
  834. (syntax-rules ()
  835. ((test-with-runner runner form ...)
  836. (let ((saved-runner (test-runner-current)))
  837. (dynamic-wind
  838. (lambda () (test-runner-current runner))
  839. (lambda () form ...)
  840. (lambda () (test-runner-current saved-runner)))))))
  841. ;;; Predicates
  842. (define (%test-match-nth n count)
  843. (let ((i 0))
  844. (lambda (runner)
  845. (set! i (+ i 1))
  846. (and (>= i n) (< i (+ n count))))))
  847. (define-syntax test-match-nth
  848. (syntax-rules ()
  849. ((test-match-nth n)
  850. (test-match-nth n 1))
  851. ((test-match-nth n count)
  852. (%test-match-nth n count))))
  853. (define (%test-match-all . pred-list)
  854. (lambda (runner)
  855. (let ((result #t))
  856. (let loop ((l pred-list))
  857. (if (null? l)
  858. result
  859. (begin
  860. (if (not ((car l) runner))
  861. (set! result #f))
  862. (loop (cdr l))))))))
  863. (define-syntax test-match-all
  864. (syntax-rules ()
  865. ((test-match-all pred ...)
  866. (%test-match-all (%test-as-specifier pred) ...))))
  867. (define (%test-match-any . pred-list)
  868. (lambda (runner)
  869. (let ((result #f))
  870. (let loop ((l pred-list))
  871. (if (null? l)
  872. result
  873. (begin
  874. (if ((car l) runner)
  875. (set! result #t))
  876. (loop (cdr l))))))))
  877. (define-syntax test-match-any
  878. (syntax-rules ()
  879. ((test-match-any pred ...)
  880. (%test-match-any (%test-as-specifier pred) ...))))
  881. ;; Coerce to a predicate function:
  882. (define (%test-as-specifier specifier)
  883. (cond ((procedure? specifier) specifier)
  884. ((integer? specifier) (test-match-nth 1 specifier))
  885. ((string? specifier) (test-match-name specifier))
  886. (else
  887. (error "not a valid test specifier"))))
  888. (define-syntax test-skip
  889. (syntax-rules ()
  890. ((test-skip pred ...)
  891. (let ((runner (test-runner-get)))
  892. (%test-runner-skip-list! runner
  893. (cons (test-match-all (%test-as-specifier pred) ...)
  894. (%test-runner-skip-list runner)))))))
  895. (define-syntax test-expect-fail
  896. (syntax-rules ()
  897. ((test-expect-fail pred ...)
  898. (let ((runner (test-runner-get)))
  899. (%test-runner-fail-list! runner
  900. (cons (test-match-all (%test-as-specifier pred) ...)
  901. (%test-runner-fail-list runner)))))))
  902. (define (test-match-name name)
  903. (lambda (runner)
  904. (equal? name (test-runner-test-name runner))))
  905. (define (test-read-eval-string string)
  906. (let* ((port (open-input-string string))
  907. (form (read port)))
  908. (if (eof-object? (read-char port))
  909. (eval form)
  910. (cond-expand
  911. (srfi-23 (error "(not at eof)"))
  912. (else "error")))))