blob: 8d1506778964a2400664e4db98053049d7834a8b [file] [log] [blame]
Chih-Hung Hsiehbde955c2019-04-01 20:05:54 -07001%% Unit test helpers
2
3% Write one line message.
4msg(A) :- write(A), nl.
5msg(A,B) :- write(A), msg(B).
6msg(A,B,C) :- write(A), msg(B,C).
7msg(A,B,C,D) :- write(A), msg(B,C,D).
8msg(A,B,C,D,E) :- write(A), msg(B,C,D,E).
9msg(A,B,C,D,E,F) :- write(A), msg(B,C,D,E,F).
10
11% Redefine a caluse.
12redefine(Atom,Arity,Clause) :- abolish(Atom/Arity), assertz(Clause).
13
14% Increment/decrement of pass/fail counters.
15set_counters(N,X,Y) :- redefine(test_count,3,test_count(N,X,Y)).
16get_counters(N,X,Y) :- clause(test_count(N,X,Y), _) -> true ; (X=0, Y=0).
17inc_pass_count :- get_counters(N,P,F), P1 is P + 1, set_counters(N,P1,F).
18inc_fail_count :- get_counters(N,P,F), F1 is F + 1, set_counters(N,P,F1).
19
20% Report pass or fail of G.
21pass_1(G) :- msg('PASS: ', G), inc_pass_count.
22fail_1(G) :- msg('FAIL: ', G), inc_fail_count.
23
24% Report pass or fail of not(G).
25pass_0(G) :- msg('PASS: not(', G, ')'), inc_pass_count.
26fail_0(G) :- msg('FAIL: not(', G, ')'), inc_fail_count.
27
28% Report a test as failed if it passed 2 or more times
29pass_twice(G) :-
30 msg('FAIL: (pass twice): ', G),
31 inc_fail_count.
32pass_many(G) :-
33 G = [A,B|_],
34 length(G, N),
35 msg('FAIL: (pass ', N, ' times): ', [A,B,'...']),
36 inc_fail_count.
37
38% Test if G fails.
39test0(G) :- once(G) -> fail_0(G) ; pass_0(G).
40
41% Test if G passes exactly once.
42test1(G) :-
43 findall(G, G, S), length(S, N),
44 (N == 0
45 -> fail_1(G)
46 ; (N == 1
47 -> pass_1(S)
48 ; (N == 2 -> pass_twice(S) ; pass_many(S))
49 )
50 ).
51
52% Report the begin of test N.
53begin_tests(N) :-
54 nl,
55 msg('BEGIN test ',N),
56 set_counters(N,0,0).
57
58% Repot the end of test N and total pass/fail counts,
59% and check if the numbers are as exected OutP/OutF.
60end_tests(OutP,OutF) :-
61 get_counters(N,P,F),
62 (OutP = P
63 -> msg('Expected #PASS: ', OutP)
64 ; (msg('ERROR: expected #PASS is ',OutP), !, fail)
65 ),
66 (OutF = F
67 -> msg('Expected #FAIL: ', OutF)
68 ; (msg('ERROR: expected #FAIL is ',OutF), !, fail)
69 ),
70 msg('END test ', N),
71 nl.
72
73% Repot the end of test N and total pass/fail counts.
74end_tests(N) :- end_tests(N,_,_).
75
76% Call end_tests/2 and halt if the fail count is unexpected.
77end_tests_or_halt(ExpectedFails) :-
78 end_tests(_,ExpectedFails); (flush_output, halt(1)).