Page Menu
Home
HEPForge
Search
Configure Global Search
Log In
Files
F8723940
OUnit.ml
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Flag For Later
Size
20 KB
Subscribers
None
OUnit.ml
View Options
(* oUnit.ml -- *)
(***********************************************************************)
(* The OUnit library *)
(* *)
(* Copyright (C) 2002-2008 Maas-Maarten Zeeman. *)
(* Copyright (C) 2010 OCamlCore SARL *)
(* *)
(***********************************************************************)
(* Version 1.1.2, with minor modifications by Thorsten Ohl *)
(************************************************************************
The package OUnit is copyright by Maas-Maarten Zeeman and OCamlCore SARL.
Permission is hereby granted, free of charge, to any person obtaining
a copy of this document and the OUnit software ("the Software"), to
deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute,
sublicense, and/or sell copies of the Software, and to permit persons
to whom the Software is furnished to do so, subject to the following
conditions:
The above copyright notice and this permission notice shall be
included in all copies or substantial portions of the Software.
The Software is provided ``as is'', without warranty of any kind,
express or implied, including but not limited to the warranties of
merchantability, fitness for a particular purpose and noninfringement.
In no event shall Maas-Maarten Zeeman be liable for any claim, damages
or other liability, whether in an action of contract, tort or
otherwise, arising from, out of or in connection with the Software or
the use or other dealings in the software.
************************************************************************)
open
Format
(* TODO: really use Format in printf call. Most of the time, not
* cuts/spaces/boxes are used
*)
let
global_verbose
=
ref
false
let
buff_printf
f
=
let
buff
=
Buffer
.
create
13
in
let
fmt
=
formatter_of_buffer
buff
in
f
fmt
;
pp_print_flush
fmt
()
;
Buffer
.
contents
buff
let
bracket
set_up
f
tear_down
()
=
let
fixture
=
set_up
()
in
let
()
=
try
let
()
=
f
fixture
in
tear_down
fixture
with
e
->
let
()
=
tear_down
fixture
in
raise
e
in
()
let
bracket_tmpfile
?(
prefix
=
"ounit-"
)
?(
suffix
=
".txt"
)
?
mode
f
=
bracket
(
fun
()
->
Filename
.
open_temp_file
?
mode
prefix
suffix
)
f
(
fun
(
fn
,
chn
)
->
begin
try
close_out
chn
with
_
->
()
end
;
begin
try
Sys
.
remove
fn
with
_
->
()
end
)
exception
Skip
of
string
let
skip_if
b
msg
=
if
b
then
raise
(
Skip
msg
)
exception
Todo
of
string
let
todo
msg
=
raise
(
Todo
msg
)
let
assert_failure
msg
=
failwith
(
"OUnit: "
^
msg
)
let
assert_bool
msg
b
=
if
not
b
then
assert_failure
msg
let
assert_string
str
=
if
not
(
str
=
""
)
then
assert_failure
str
let
assert_equal
?(
cmp
=
(
=
))
?
printer
?
pp_diff
?
msg
expected
actual
=
let
get_error_string
()
=
(* let max_len = pp_get_margin fmt () in *)
(* let ellipsis_text = "[...]" in *)
let
print_ellipsis
p
fmt
s
=
(* TODO: find a way to do this
let res = p s in
let len = String.length res in
if diff <> None && len > max_len then
begin
let len_with_ellipsis =
(max_len - (String.length ellipsis_text)) / 2
in
(* TODO: we should use %a here to print values *)
fprintf fmt
"@[%s[...]%s@]"
(String.sub res
0
len_with_ellipsis)
(String.sub res
(len - len_with_ellipsis)
len_with_ellipsis)
end
else
begin
(* TODO: we should use %a here to print values *)
fprintf fmt "@[%s@]" res
end
*)
pp_print_string
fmt
(
p
s
)
in
let
res
=
buff_printf
(
fun
fmt
->
pp_open_vbox
fmt
0
;
begin
match
msg
with
|
Some
s
->
pp_open_box
fmt
0
;
pp_print_string
fmt
s
;
pp_close_box
fmt
()
;
pp_print_cut
fmt
()
|
None
->
()
end
;
begin
match
printer
with
|
Some
p
->
let
p_ellipsis
=
print_ellipsis
p
in
fprintf
fmt
"@[expected: @[%a@]@ but got: @[%a@]@]@,"
p_ellipsis
expected
p_ellipsis
actual
|
None
->
fprintf
fmt
"@[not equal@]@,"
end
;
begin
match
pp_diff
with
|
Some
d
->
fprintf
fmt
"@[differences: %a@]@,"
d
(
expected
,
actual
)
|
None
->
()
end
;
pp_close_box
fmt
()
)
in
let
len
=
String
.
length
res
in
if
len
>
0
&&
res
.[
len
-
1
]
=
'\n'
then
String
.
sub
res
0
(
len
-
1
)
else
res
in
if
not
(
cmp
expected
actual
)
then
assert_failure
(
get_error_string
()
)
let
assert_command
?(
exit_code
=
Unix
.
WEXITED
0
)
?(
use_stderr
=
true
)
?
env
?
verbose
prg
args
=
let
verbose
=
match
verbose
with
|
Some
v
->
v
|
None
->
!
global_verbose
in
bracket_tmpfile
(
fun
(
fn_out
,
chn_out
)
->
let
cmd_print
fmt
=
let
()
=
match
env
with
|
Some
e
->
begin
pp_print_string
fmt
"env"
;
Array
.
iter
(
fprintf
fmt
"@ %s"
)
e
;
pp_print_space
fmt
()
end
|
None
->
()
in
pp_print_string
fmt
prg
;
List
.
iter
(
fprintf
fmt
"@ %s"
)
args
in
(* Start the process *)
let
in_write
=
Unix
.
dup
(
Unix
.
descr_of_out_channel
chn_out
)
in
let
(
out_read
,
out_write
)
=
Unix
.
pipe
()
in
let
err
=
if
use_stderr
then
in_write
else
Unix
.
stderr
in
let
args
=
Array
.
of_list
(
prg
::
args
)
in
let
pid
=
Unix
.
set_close_on_exec
out_write
;
if
verbose
then
printf
"@[Starting command '%t'@]
\n
"
cmd_print
;
match
env
with
|
Some
e
->
Unix
.
create_process_env
prg
args
e
out_read
in_write
err
|
None
->
Unix
.
create_process
prg
args
out_read
in_write
err
in
let
()
=
Unix
.
close
out_read
;
Unix
.
close
in_write
in
let
_,
real_exit_code
=
let
rec
wait_intr
()
=
try
Unix
.
waitpid
[]
pid
with
Unix
.
Unix_error
(
Unix
.
EINTR
,
_,
_)
->
wait_intr
()
in
wait_intr
()
in
let
exit_code_printer
=
function
|
Unix
.
WEXITED
n
->
Printf
.
sprintf
"exit code %d"
n
|
Unix
.
WSTOPPED
n
->
Printf
.
sprintf
"stopped by signal %d"
n
|
Unix
.
WSIGNALED
n
->
Printf
.
sprintf
"killed by signal %d"
n
in
(* Dump process output to stderr *)
if
verbose
then
begin
let
chn
=
open_in
fn_out
in
let
buff
=
Bytes
.
make
4096
'X'
in
let
len
=
ref
(-
1
)
in
while
!
len
<>
0
do
len
:=
input
chn
buff
0
(
Bytes
.
length
buff
);
printf
"%s"
(
Bytes
.
sub_string
buff
0
!
len
);
done
;
printf
"@?"
;
close_in
chn
end
;
(* Check process status *)
assert_equal
~
msg
:(
buff_printf
(
fun
fmt
->
fprintf
fmt
"@[Exit status of command '%t'@]"
cmd_print
))
~
printer
:
exit_code_printer
exit_code
real_exit_code
)
()
let
raises
f
=
try
f
()
;
None
with
e
->
Some
e
let
assert_raises
?
msg
exn
(
f
:
unit
->
'
a
)
=
let
pexn
=
Printexc
.
to_string
in
let
get_error_string
()
=
let
str
=
Format
.
sprintf
"expected exception %s, but no exception was raised."
(
pexn
exn
)
in
match
msg
with
|
None
->
assert_failure
str
|
Some
s
->
assert_failure
(
Format
.
sprintf
"%s
\n
%s"
s
str
)
in
match
raises
f
with
|
None
->
assert_failure
(
get_error_string
()
)
|
Some
e
->
assert_equal
?
msg
~
printer
:
pexn
exn
e
(* Compare floats up to a given relative error *)
let
cmp_float
?(
epsilon
=
0
.
00001
)
a
b
=
abs_float
(
a
-.
b
)
<=
epsilon
*.
(
abs_float
a
)
||
abs_float
(
a
-.
b
)
<=
epsilon
*.
(
abs_float
b
)
(* Now some handy shorthands *)
let
(@?)
=
assert_bool
(* The type of test function *)
type
test_fun
=
unit
->
unit
(* The type of tests *)
type
test
=
|
TestCase
of
test_fun
|
TestList
of
test
list
|
TestLabel
of
string
*
test
(* Some shorthands which allows easy test construction *)
let
(>:)
s
t
=
TestLabel
(
s
,
t
)
(* infix *)
let
(>::)
s
f
=
TestLabel
(
s
,
TestCase
(
f
))
(* infix *)
let
(>:::)
s
l
=
TestLabel
(
s
,
TestList
(
l
))
(* infix *)
(* Utility function to manipulate test *)
let
rec
test_decorate
g
=
function
|
TestCase
f
->
TestCase
(
g
f
)
|
TestList
tst_lst
->
TestList
(
List
.
map
(
test_decorate
g
)
tst_lst
)
|
TestLabel
(
str
,
tst
)
->
TestLabel
(
str
,
test_decorate
g
tst
)
(* Return the number of available tests *)
let
rec
test_case_count
=
function
|
TestCase
_
->
1
|
TestLabel
(_,
t
)
->
test_case_count
t
|
TestList
l
->
List
.
fold_left
(
fun
c
t
->
c
+
test_case_count
t
)
0
l
type
node
=
|
ListItem
of
int
|
Label
of
string
type
path
=
node
list
let
string_of_node
=
function
|
ListItem
n
->
string_of_int
n
|
Label
s
->
s
let
string_of_path
path
=
String
.
concat
":"
(
List
.
rev_map
string_of_node
path
)
(* Some helper function, they are generally applicable *)
(* Applies function f in turn to each element in list. Function f takes
one element, and integer indicating its location in the list *)
let
mapi
f
l
=
let
rec
rmapi
cnt
l
=
match
l
with
|
[]
->
[]
|
h
::
t
->
(
f
h
cnt
)
::
(
rmapi
(
cnt
+
1
)
t
)
in
rmapi
0
l
let
fold_lefti
f
accu
l
=
let
rec
rfold_lefti
cnt
accup
l
=
match
l
with
|
[]
->
accup
|
h
::
t
->
rfold_lefti
(
cnt
+
1
)
(
f
accup
h
cnt
)
t
in
rfold_lefti
0
accu
l
(* Returns all possible paths in the test. The order is from test case
to root
*)
let
test_case_paths
test
=
let
rec
tcps
path
test
=
match
test
with
|
TestCase
_
->
[
path
]
|
TestList
tests
->
List
.
concat
(
mapi
(
fun
t
i
->
tcps
((
ListItem
i
)::
path
)
t
)
tests
)
|
TestLabel
(
l
,
t
)
->
tcps
((
Label
l
)::
path
)
t
in
tcps
[]
test
(* Test filtering with their path *)
module
SetTestPath
=
Set
.
Make
(
String
)
let
test_filter
?(
skip
=
false
)
only
test
=
let
set_test
=
List
.
fold_left
(
fun
st
str
->
SetTestPath
.
add
str
st
)
SetTestPath
.
empty
only
in
let
rec
filter_test
path
tst
=
if
SetTestPath
.
mem
(
string_of_path
path
)
set_test
then
begin
Some
tst
end
else
begin
match
tst
with
|
TestCase
f
->
begin
if
skip
then
Some
(
TestCase
(
fun
()
->
skip_if
true
"Test disabled"
;
f
()
))
else
None
end
|
TestList
tst_lst
->
begin
let
ntst_lst
=
fold_lefti
(
fun
ntst_lst
tst
i
->
let
nntst_lst
=
match
filter_test
((
ListItem
i
)
::
path
)
tst
with
|
Some
tst
->
tst
::
ntst_lst
|
None
->
ntst_lst
in
nntst_lst
)
[]
tst_lst
in
if
not
skip
&&
ntst_lst
=
[]
then
None
else
Some
(
TestList
(
List
.
rev
ntst_lst
))
end
|
TestLabel
(
lbl
,
tst
)
->
begin
let
ntst_opt
=
filter_test
((
Label
lbl
)
::
path
)
tst
in
match
ntst_opt
with
|
Some
ntst
->
Some
(
TestLabel
(
lbl
,
ntst
))
|
None
->
if
skip
then
Some
(
TestLabel
(
lbl
,
tst
))
else
None
end
end
in
filter_test
[]
test
(* The possible test results *)
type
test_result
=
|
RSuccess
of
path
|
RFailure
of
path
*
string
|
RError
of
path
*
string
|
RSkip
of
path
*
string
|
RTodo
of
path
*
string
let
is_success
=
function
|
RSuccess
_
->
true
|
RFailure
_
|
RError
_
|
RSkip
_
|
RTodo
_
->
false
let
is_failure
=
function
|
RFailure
_
->
true
|
RSuccess
_
|
RError
_
|
RSkip
_
|
RTodo
_
->
false
let
is_error
=
function
|
RError
_
->
true
|
RSuccess
_
|
RFailure
_
|
RSkip
_
|
RTodo
_
->
false
let
is_skip
=
function
|
RSkip
_
->
true
|
RSuccess
_
|
RFailure
_
|
RError
_
|
RTodo
_
->
false
let
is_todo
=
function
|
RTodo
_
->
true
|
RSuccess
_
|
RFailure
_
|
RError
_
|
RSkip
_
->
false
let
result_flavour
=
function
|
RError
_
->
"Error"
|
RFailure
_
->
"Failure"
|
RSuccess
_
->
"Success"
|
RSkip
_
->
"Skip"
|
RTodo
_
->
"Todo"
let
result_path
=
function
|
RSuccess
path
|
RError
(
path
,
_)
|
RFailure
(
path
,
_)
|
RSkip
(
path
,
_)
|
RTodo
(
path
,
_)
->
path
let
result_msg
=
function
|
RSuccess
_
->
"Success"
|
RError
(_,
msg
)
|
RFailure
(_,
msg
)
|
RSkip
(_,
msg
)
|
RTodo
(_,
msg
)
->
msg
(* Returns true if the result list contains successes only *)
let
rec
was_successful
=
function
|
[]
->
true
|
RSuccess
_::
t
|
RSkip
_::
t
->
was_successful
t
|
RFailure
_::_
|
RError
_::_
|
RTodo
_::_
->
false
(* Events which can happen during testing *)
type
test_event
=
|
EStart
of
path
|
EEnd
of
path
|
EResult
of
test_result
let
maybe_backtrace
()
=
if
Printexc
.
backtrace_status
()
then
"
\n
"
^
Printexc
.
get_backtrace
()
else
""
(* Run all tests, report starts, errors, failures, and return the results *)
let
perform_test
report
test
=
let
run_test_case
f
path
=
try
f
()
;
RSuccess
path
with
|
Failure
s
->
RFailure
(
path
,
s
^
maybe_backtrace
()
)
|
Skip
s
->
RSkip
(
path
,
s
)
|
Todo
s
->
RTodo
(
path
,
s
)
|
s
->
RError
(
path
,
Printexc
.
to_string
s
^
maybe_backtrace
()
)
in
let
rec
run_test
path
results
=
function
|
TestCase
(
f
)
->
begin
let
result
=
report
(
EStart
path
);
run_test_case
f
path
in
report
(
EResult
result
);
report
(
EEnd
path
);
result
::
results
end
|
TestList
(
tests
)
->
begin
fold_lefti
(
fun
results
t
cnt
->
run_test
((
ListItem
cnt
)::
path
)
results
t
)
results
tests
end
|
TestLabel
(
label
,
t
)
->
begin
run_test
((
Label
label
)::
path
)
results
t
end
in
run_test
[]
[]
test
(* Function which runs the given function and returns the running time
of the function, and the original result in a tuple *)
let
time_fun
f
x
y
=
let
begin_time
=
Unix
.
gettimeofday
()
in
(
Unix
.
gettimeofday
()
-.
begin_time
,
f
x
y
)
(* A simple (currently too simple) text based test runner *)
let
run_test_tt
?
verbose
test
=
let
verbose
=
match
verbose
with
|
Some
v
->
v
|
None
->
!
global_verbose
in
let
printf
=
Format
.
printf
in
let
separator1
=
String
.
make
(
get_margin
()
)
'='
in
let
separator2
=
String
.
make
(
get_margin
()
)
'-'
in
let
string_of_result
=
function
|
RSuccess
_
->
if
verbose
then
"ok
\n
"
else
"."
|
RFailure
(_,
_)
->
if
verbose
then
"FAIL
\n
"
else
"F"
|
RError
(_,
_)
->
if
verbose
then
"ERROR
\n
"
else
"E"
|
RSkip
(_,
_)
->
if
verbose
then
"SKIP
\n
"
else
"S"
|
RTodo
(_,
_)
->
if
verbose
then
"TODO
\n
"
else
"T"
in
let
report_event
=
function
|
EStart
p
->
if
verbose
then
printf
"%s ...
\n
"
(
string_of_path
p
)
|
EEnd
_
->
()
|
EResult
result
->
printf
"%s@?"
(
string_of_result
result
)
in
let
print_result_list
results
=
List
.
iter
(
fun
result
->
printf
"%s
\n
%s: %s
\n\n
%s
\n
%s
\n
"
separator1
(
result_flavour
result
)
(
string_of_path
(
result_path
result
))
(
result_msg
result
)
separator2
)
results
in
(* Now start the test *)
let
running_time
,
results
=
time_fun
perform_test
report_event
test
in
let
errors
=
List
.
filter
is_error
results
in
let
failures
=
List
.
filter
is_failure
results
in
let
skips
=
List
.
filter
is_skip
results
in
let
todos
=
List
.
filter
is_todo
results
in
if
not
verbose
then
printf
"
\n
"
;
(* Print test report *)
print_result_list
errors
;
print_result_list
failures
;
printf
"Ran: %d tests in: %.2f seconds.
\n
"
(
List
.
length
results
)
running_time
;
(* Print final verdict *)
if
was_successful
results
then
(
if
skips
=
[]
then
printf
"OK"
else
printf
"OK: Cases: %d Skip: %d
\n
"
(
test_case_count
test
)
(
List
.
length
skips
)
)
else
printf
"FAILED: Cases: %d Tried: %d Errors: %d \
Failures: %d Skip:%d Todo:%d
\n
"
(
test_case_count
test
)
(
List
.
length
results
)
(
List
.
length
errors
)
(
List
.
length
failures
)
(
List
.
length
skips
)
(
List
.
length
todos
);
(* Return the results possibly for further processing *)
results
(* Call this one from you test suites *)
let
run_test_tt_main
?(
arg_specs
=
[]
)
?(
set_verbose
=
ignore
)
suite
=
let
only_test
=
ref
[]
in
let
()
=
Arg
.
parse
(
Arg
.
align
[
"-verbose"
,
Arg
.
Set
global_verbose
,
" Run the test in verbose mode."
;
"-only-test"
,
Arg
.
String
(
fun
str
->
only_test
:=
str
::
!
only_test
),
"path Run only the selected test"
;
"-list-test"
,
Arg
.
Unit
(
fun
()
->
List
.
iter
(
fun
pth
->
print_endline
(
string_of_path
pth
))
(
test_case_paths
suite
);
exit
0
),
" List tests"
;
]
@
arg_specs
)
(
fun
x
->
raise
(
Arg
.
Bad
(
"Bad argument : "
^
x
)))
(
"usage: "
^
Sys
.
argv
.(
0
)
^
" [-verbose] [-only-test path]*"
)
in
let
nsuite
=
if
!
only_test
=
[]
then
suite
else
begin
match
test_filter
~
skip
:
true
!
only_test
suite
with
|
Some
test
->
test
|
None
->
failwith
(
"Filtering test "
^
(
String
.
concat
", "
!
only_test
)^
" lead to no test"
)
end
in
let
result
=
set_verbose
!
global_verbose
;
run_test_tt
~
verbose
:!
global_verbose
nsuite
in
if
not
(
was_successful
result
)
then
exit
1
else
result
File Metadata
Details
Attached
Mime Type
text/x-ruby
Expires
Mon, Jan 20, 9:57 PM (1 d, 4 h)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
4242574
Default Alt Text
OUnit.ml (20 KB)
Attached To
rWHIZARDSVN whizardsvn
Event Timeline
Log In to Comment