package ppx_inline_test
Install
Dune Dependency
Authors
Maintainers
Sources
sha256=b71e4f01ab8aed418a3358688241a94b6d16d723deec7caaf5e4e917c2a76d2c
Description
Part of the Jane Street's PPX rewriters collection.
Published: 23 May 2024
README
ppx_inline_test
Syntax extension for writing in-line tests in ocaml code.
New syntactic constructs
The following constructs are now valid structure items:
let%test "name" = <boolean expr> (* true means ok, false or exn means broken *)
let%test_unit "name" = <unit expr> (* () means ok, exn means broken *)
let%test_module "name" = (module <module-expr>) (* to group tests (to share
some setup for instance) *)
We may write _
instead of "name"
for anonymous tests. It is also possible to use [%name <string expr>]
for a dynamically computed name.
When running tests, they will be executed when the control flow reaches the structure item (i.e. at toplevel for a toplevel test; when the functor is applied for a test defined in the body of a functor, etc.).
Tags
One can tag tests with the following construct:
let%test "name" [@tags "no-js"] = <expr>
let%test "name" [@tags "no-js", "other-tag"] = <expr>
let%test _ [@tags "no-js"] = <expr>
let%test _ [@tags "js-only"] = <expr>
Available tags are:
no-js
for tests that should not run when compiling OCaml to Javascriptjs-only
for tests that should only run in Javascript32-bits-only
for tests that should only run in 32 bits architectures64-bits-only
for tests that should only run in 64 bits architecturesfast-flambda
for tests that might only pass when compiling with flambda or flambda2, -O3, and cross library inliningfast-flambda2
for tests that might only pass when compiling with flambda2, -O3, and cross library inliningx-library-inlining-sensitive
for tests that might only pass when compiling with cross library inlining switched ondisabled
for tests that should not run (unless requested with -require-tag)
One can also tag entire test modules similarly:
let%test_module "name" [@tags "no-js"] = (module struct end)
The flags -drop-tag
and -require-tag
can be passed to the test runner to restrict which tests are run. We say the tags of a test are the union of the tags applied directly to that test using [@tags ...]
and the tags of all enclosing modules. It is to this union that the predicates -drop-tag
and -require-tag
are applied.
If it is clear, from a test-module's tags, that none of the tests within will possibly match the tag predicates imposed by the command line flags, then additionally the top-level of that module will not be run.
Examples
prime.ml
let is_prime = <magic>
let%test _ = is_prime 5
let%test _ = is_prime 7
let%test _ = not (is_prime 1)
let%test _ = not (is_prime 8)
Tests in a functor.
module Make(C : S) = struct
<magic>
let%test _ = <some expression>
end
module M = Make(Int)
Grouping test and side-effecting initialisation.
Since the module passed as an argument to let%test_module
is only initialised when we run the tests, it is ok to perform side-effects in the module-expression argument.
let%test_module _ = (module struct
module UID = Unique_id.Int(struct end)
let%test _ = UID.create() <> UID.create()
end)
Building and running the tests with Dune
Inline tests can only be used in libraries, not executables.
To use this with dune, see dune's documentation. At the time of writing of the current document, the short version is:
define a library this way:
(library
(name foo)
(inline_tests)
(preprocess (pps ppx_inline_test)))
add tests to it
call
dune runtest
Building and running the tests without Dune
Code using this extension must be compiled and linked using the ppx_inline_test.runtime-lib
library. The ppx_inline_test
syntax extension will reject any test if it wasn't passed a -inline-test-lib libname
flag.
Execution
Tests are only executed when both these conditions are met:
the executable containing the tests is linked with
ppx_inline_test.runner.lib
the executable containing the tests is called with command line arguments:
your.exe inline-test-runner libname [options]
This libname
is a way of restricting the tests run by the executable. The dependencies of your library (or executable) could also use ppx_inline_test
, but you don't necessarily want to run their tests too. For instance, core
is built by giving -inline-test-lib core
and core_extended
is built by giving -inline-test-lib core_extended
. And now when an executable linked with both core
and core_extended
is run with a libname
of core_extended
, only the tests of core_extended
are run.
Finally, after running tests, Ppx_inline_test_lib.exit ()
should be called (to exit with an error and a summary of the number of failed tests if there were errors or exit normally otherwise).
One can construct a dual-use binary that only runs the tests when prompted to (through the command line), by sticking the following piece of code in it, after the tests have run but before the binary starts doing non-test side effects. However be aware that Base.am_testing
will be true
even when not running tests, which may be undesirable.
match Ppx_inline_test_lib.testing with
| `Testing `Am_test_runner ->
print_endline "Exiting test suite";
Ppx_inline_test_lib.exit ()
| `Testing _ -> exit 0
| `Not_testing -> ()
Command line arguments
The executable that runs tests can take additional command line arguments. The most useful of these are:
-stop-on-error
Stop running tests after the first error.
-verbose
to see the tests as they run
-only-test location
where location is either a filename
-only-test main.ml
, a filename with a line number-only-test main.ml:32
, or with the syntax that the compiler uses:File "main.ml"
, orFile "main.ml", line 32
orFile "main.ml", line 32, characters 2-6
(characters are ignored). The position that matters is the position of thelet%test
orlet%test_unit
.The positions shown by
-verbose
are valid inputs for-only-test
.If no
-only-test
flag is given, all the tests are run. Otherwise all the tests matching any of the locations are run.-drop-tag tag
drop all the tests tagged with
tag
.
These can be specified to jenga like this:
(library
(...
(inline_tests ((flags (-stop-on-error))))
...
))
and to dune like this:
(library
...
(inline_tests (flags (-stop-on-error)))
...)
Parallelizing tests
If you pass arguments of the form -inline-test-lib lib:partition
to ppx_inline_test
, then you will be able to run tests from a given source file in parallel with tests from other source files. All the tests inside the same source file are still run sequentially.
You should pick different partition
names for the different files in your library (the name of the .ml files for instance).
ppx_inline_test_lib
currently requires some external system like a build system to run it multiple times in parallel, although we may make it possible to run the inline tests in parallel directly in the future.
If you do that, you can now use two new flags of the executable containing the tests:
-list-partitions
lists all the partitions that contain at least one test, one per line.
-partition P
only run the tests of the library that are encountered at toplevel of the source file that was preprocessed with the given partition
P
(the tests need not be syntactically in the file, they could be the result of applying a functor)
A build system can combine these two commands by first listing partitions, and then running one command for each partition.
Dependencies (5)
Dev Dependencies
None
-
alba
>= "0.4.1"
- autofonce
- autofonce_config
- autofonce_core
- autofonce_lib
- autofonce_m4
- autofonce_misc
- autofonce_patch
- autofonce_share
-
aws-s3
>= "4.0.0" & != "4.5.0"
-
bio_io
>= "0.2.1"
- bitpack_serializer
-
bitwuzla
>= "1.0.0"
- bitwuzla-c
- bitwuzla-cxx
-
caisar
>= "0.2.1"
- caisar-ir
- colibri2
-
coq-lsp
>= "0.1.9+8.17"
-
core
>= "v0.17.0"
- drom
- drom_lib
- drom_toml
-
dune
>= "3.17.0"
-
electrod
>= "0.1.6" & != "0.2.1" & < "0.5"
-
embedded_ocaml_templates
>= "0.6"
-
encoding
< "0.0.4"
- extism
- extism-manifest
-
ez_cmdliner
>= "0.2.0"
-
ez_config
>= "0.2.0"
-
ez_file
>= "0.2.0"
-
ez_hash
< "0.5.3"
- ez_opam_file
- ez_search
- ez_subst
- fiber-lwt
-
fmlib
< "0.5.0"
- fmlib_browser
-
fmlib_js
< "0.5.0"
- fmlib_parse
- fmlib_pretty
- fmlib_std
-
GT
>= "0.5.0"
-
guardian
< "0.1.0"
-
hdf5
>= "0.1.5"
- header-check
- hexstring
- idds
- knights_tour
-
lablqml
>= "0.7"
-
learn-ocaml
>= "0.16.0"
-
learn-ocaml-client
>= "0.16.0"
- little_logger
- module-graph
- mula
- mysql8
-
nuscr
< "2.0.0"
-
OCanren
>= "0.3.0~alpha1"
-
OCanren-ppx
>= "0.3.0~alpha1"
-
ocaml-protoc-plugin
>= "1.0.0"
- ocp-search
-
ocplib_stuff
>= "0.3.0"
-
opam-bin
>= "0.9.5"
- opam-check-npm-deps
-
opam_bin_lib
>= "0.9.5"
- patricia-tree
- pp-binary-ints
-
ppx_bench
>= "v0.17.0"
- ppx_deriving_cad
- ppx_deriving_decoders
- ppx_deriving_scad
-
ppx_expect
>= "v0.17.0"
-
ppx_jane
>= "v0.17.0"
- ppx_partial
-
ppx_protocol_conv_json
>= "5.0.0"
- ppx_ts
-
psmt2-frontend
>= "0.3.0"
- pyml_bindgen
- randoml
- res_tailwindcss
- sel
-
serde
< "0.0.2"
- serde_json
- serde_sexpr
-
sexp_decode
>= "0.5"
- simple63
- solidity-alcotest
- solidity-common
- solidity-parser
- solidity-test
- solidity-typechecker
-
splittable_random
>= "v0.17.0"
-
sqlite3
>= "5.0.1"
-
tezos-micheline
< "16.0"
-
tezos-stdlib
< "16.0"
-
toplevel_expect_test
>= "v0.17.0"
-
torch
>= "v0.17.0"
- vscoq-language-server
- yosqlite
- zanuda
Conflicts
None