package alcotest

  1. Overview
  2. Docs
Alcotest is a lightweight and colourful test framework

Install

Dune Dependency

Authors

Maintainers

Sources

alcotest-mirage-1.4.0.tbz
sha256=b1aaccfb2d651c902592c04953e2619169c91f797cf4f04a7dda2cab09b93ec1
sha512=8a13d5d4c8c77f115903e6b8e58160c6e6ec27870440bd38a674e9406f57f1eff299e65f006fd77728015d1a8f0ae30a714fe47e035824950a71ebfdff2cf3c9

Description

Alcotest exposes simple interface to perform unit tests. It exposes a simple TESTABLE module type, a check function to assert test predicates and a run function to perform a list of unit -> unit test callbacks.

Alcotest provides a quiet and colorful output where only faulty runs are fully displayed at the end of the run (with the full logs ready to inspect), with a simple (yet expressive) query language to select the tests to run.

Published: 16 Apr 2021

README

Alcotest is a lightweight and colourful test framework.

Alcotest exposes simple interface to perform unit tests. It exposes a simple TESTABLE module type, a check function to assert test predicates and a run function to perform a list of unit -> unit test callbacks.

Alcotest provides a quiet and colorful output where only faulty runs are fully displayed at the end of the run (with the full logs ready to inspect), with a simple (yet expressive) query language to select the tests to run. See the manpage for details.

For information on contributing to Alcotest, see CONTRIBUTING.md.

Examples

A simple example (taken from examples/simple.ml):

Generated by the following test suite specification:

(* Build with `ocamlbuild -pkg alcotest simple.byte` *)

(* A module with functions to test *)
module To_test = struct
  let lowercase = String.lowercase_ascii
  let capitalize = String.capitalize_ascii
  let str_concat = String.concat ""
  let list_concat = List.append
end

(* The tests *)
let test_lowercase () =
  Alcotest.(check string) "same string" "hello!" (To_test.lowercase "hELLO!")

let test_capitalize () =
  Alcotest.(check string) "same string" "World." (To_test.capitalize "world.")

let test_str_concat () =
  Alcotest.(check string) "same string" "foobar" (To_test.str_concat ["foo"; "bar"])

let test_list_concat () =
  Alcotest.(check (list int)) "same lists" [1; 2; 3] (To_test.list_concat [1] [2; 3])

(* Run it *)
let () =
  let open Alcotest in
  run "Utils" [
      "string-case", [
          test_case "Lower case"     `Quick test_lowercase;
          test_case "Capitalization" `Quick test_capitalize;
        ];
      "string-concat", [ test_case "String mashing" `Quick test_str_concat  ];
      "list-concat",   [ test_case "List mashing"   `Slow  test_list_concat ];
    ]

The result is a self-contained binary which displays the test results. Use dune exec examples/simple.exe -- --help to see the runtime options.

Here's an example of a of failing test suite:

By default, only the first failing test log is printed to the console (and all test logs are captured on disk). Pass --show-errors to print all error messages.

Selecting tests to execute

You can filter which tests to run by supplying a regular expression matching the names of the tests to execute, or by passing a regular expression and a comma-separated list of test numbers (or ranges of test numbers, e.g. 2,4..9):

$ ./simple.native test '.*concat*'
Testing Utils.
[SKIP]     string-case            0   Lower case.
[SKIP]     string-case            1   Capitalization.
[OK]       string-concat          0   String mashing.
[OK]       list-concat            0   List mashing.
The full test results are available in `_build/_tests`.
Test Successful in 0.000s. 2 tests run.

$ ./simple.native test 'string-case' '1..3'
Testing Utils.
[SKIP]     string-case            0   Lower case.
[OK]       string-case            1   Capitalization.
[SKIP]     string-concat          0   String mashing.
[SKIP]     list-concat            0   List mashing.
The full test results are available in `_build/_tests`.
Test Successful in 0.000s. 1 test run.

Note that you cannot filter by test case name (i.e. Lower case or Capitalization), you must filter by test name & number instead.

See the examples folder for more examples.

Quick and Slow tests

In general you should use `Quick tests: tests that are ran on any invocations of the test suite. You should only use `Slow tests for stress tests that are ran only on occasion (typically before a release or after a major change). These slow tests can be suppressed by passing the -q flag on the command line, e.g.:

$ ./test.exe -q # run only the quick tests
$ ./test.exe    # run quick and slow tests

Passing custom options to the tests

In most cases, the base tests are unit -> unit functions. However, it is also possible to pass an extra option to all the test functions by using 'a -> unit, where 'a is the type of the extra parameter.

In order to do this, you need to specify how this extra parameter is read on the command-line, by providing a Cmdliner term for command-line arguments which explains how to parse and serialize values of type 'a (note: do not use positional arguments, only optional arguments are supported).

For instance:

let test_nice i = Alcotest.(check int) "Is it a nice integer?" i 42

let int =
  let doc = "What is your prefered number?" in
  Cmdliner.Arg.(required & opt (some int) None & info ["n"] ~doc ~docv:"NUM")

let () =
  Alcotest.run_with_args "foo" int [
    "all", ["nice", `Quick, test_nice]
  ]

Will generate test.exe such that:

$ test.exe test
test.exe: required option -n is missing

$ test.exe test -n 42
Testing foo.
[OK]                all          0   int.

Lwt

Alcotest provides an Alcotest_lwt module that you could use to wrap Lwt test cases. The basic idea is that instead of providing a test function in the form unit -> unit, you provide one with the type unit -> unit Lwt.t and alcotest-lwt calls Lwt_main.run for you.

However, there are a couple of extra features:

  • If an async exception occurs, it will cancel your test case for you and fail it (rather than exiting the process).

  • You get given a switch, which will be turned off when the test case finishes (or fails). You can use that to free up any resources.

For instance:

let free () = print_endline "freeing all resources"; Lwt.return ()

let test_lwt switch () =
  Lwt_switch.add_hook (Some switch) free;
  Lwt.async (fun () -> failwith "All is broken");
  Lwt_unix.sleep 10.

let () =
  Lwt_main.run @@ Alcotest_lwt.run "foo" [
    "all", [
      Alcotest_lwt.test_case "one" `Quick test_lwt
    ]
  ]

Will generate:

$ test.exe
Testing foo.
[ERROR]             all          0   one.
-- all.000 [one.] Failed --
in _build/_tests/all.000.output:
freeing all resources
[failure] All is broken

Comparison with other testing frameworks

The README is pretty clear about that:

Alcotest is the only testing framework using colors!

More seriously, Alcotest is similar to ounit but it fixes a few of the problems found in that library:

  • Alcotest has a nicer output, it is easier to see what failed and what succeeded and to read the log outputs of the failed tests;

  • Alcotest uses combinators to define pretty-printers and comparators between the things to test.

Other nice tools doing different kind of testing also exist:

  • qcheck qcheck does random generation and property testing (e.g. Quick Check)

  • crowbar and bun are similar to qcheck, but use compiler-directed randomness, e.g. it takes advantage of the AFL support the OCaml compiler.

  • ppx_inline_tests allows to write tests in the same file as your source-code; they will be run only in a special mode of compilation.

Dependencies (9)

  1. uutf >= "1.0.0"
  2. stdlib-shims
  3. re >= "1.7.2"
  4. uuidm
  5. cmdliner >= "1.0.3"
  6. astring
  7. fmt >= "0.8.7"
  8. ocaml >= "4.03.0"
  9. dune >= "2.2"

Dev Dependencies (1)

  1. cmdliner with-test & < "1.1.0"

  1. ahrocksdb
  2. albatross >= "1.5.0"
  3. alcotest-async < "1.0.0" | = "1.4.0"
  4. alcotest-lwt < "1.0.0" | = "1.4.0"
  5. alcotest-mirage = "1.4.0"
  6. alg_structs_qcheck
  7. ambient-context
  8. ambient-context-eio
  9. ambient-context-lwt
  10. angstrom >= "0.7.0"
  11. ansi >= "0.6.0"
  12. anycache >= "0.7.4"
  13. anycache-async
  14. anycache-lwt
  15. archetype >= "1.4.2"
  16. archi
  17. arp != "2.3.1"
  18. arp-mirage < "2.0.0"
  19. arrakis
  20. art
  21. asak >= "0.2"
  22. asli >= "0.2.0"
  23. asn1-combinators >= "0.2.2"
  24. atd >= "2.3.3"
  25. atdgen >= "2.10.0"
  26. atdpy
  27. atdts
  28. base32
  29. base64 >= "2.1.2" & < "3.2.0" | >= "3.4.0"
  30. bastet
  31. bastet_async
  32. bastet_lwt
  33. bech32
  34. bechamel >= "0.5.0"
  35. bigarray-overlap
  36. bigstringaf
  37. bitlib
  38. blake2
  39. bloomf
  40. bls12-381 < "0.4.1" | >= "3.0.0" & < "18.0"
  41. bls12-381-hash
  42. bls12-381-js >= "0.4.2"
  43. bls12-381-js-gen >= "0.4.2"
  44. bls12-381-legacy
  45. bls12-381-signature
  46. bls12-381-unix
  47. blurhash
  48. builder-web
  49. bulletml
  50. bytebuffer
  51. ca-certs
  52. ca-certs-nss
  53. cactus
  54. caldav
  55. calendar >= "3.0.0"
  56. callipyge
  57. camlix
  58. camlkit
  59. camlkit-base
  60. capnp-rpc < "1.2.3"
  61. capnp-rpc-lwt < "0.3"
  62. capnp-rpc-mirage >= "0.9.0"
  63. capnp-rpc-unix >= "0.9.0" & < "1.2.3"
  64. carray
  65. carton
  66. carton-git
  67. carton-lwt >= "0.4.1"
  68. cborl
  69. ccss >= "1.6"
  70. cf-lwt
  71. chacha
  72. channel
  73. charrua-client
  74. charrua-client-lwt
  75. charrua-client-mirage < "0.11.0"
  76. charrua-server >= "1.4.1"
  77. checked_oint < "0.1.1"
  78. checkseum >= "0.0.3"
  79. cid
  80. clarity-lang
  81. class_group_vdf
  82. cohttp >= "0.17.0"
  83. cohttp-curl-async
  84. cohttp-curl-lwt
  85. cohttp-eio >= "6.0.0~beta2"
  86. colombe >= "0.2.0"
  87. color
  88. conan
  89. conan-cli
  90. conan-database
  91. conan-lwt
  92. conan-unix
  93. conduit = "3.0.0"
  94. conex < "0.10.0"
  95. conex-mirage-crypto
  96. conex-nocrypto
  97. conformist
  98. cookie
  99. cow >= "2.2.0"
  100. css
  101. css-parser
  102. cstruct >= "3.3.0"
  103. cstruct-sexp
  104. ctypes-zarith
  105. cuid
  106. curly
  107. current >= "0.4"
  108. current-albatross-deployer
  109. current_git >= "0.6.4"
  110. current_incr
  111. cwe_checker
  112. data-encoding
  113. datakit >= "0.12.0"
  114. datakit-bridge-github >= "0.12.0"
  115. datakit-ci
  116. datakit-client-git >= "0.12.0"
  117. decompress >= "0.8" & < "1.5.3"
  118. depyt
  119. digestif >= "0.8.1"
  120. dirsp-exchange-kbb2017
  121. dirsp-proscript-mirage
  122. dirsp-ps2ocaml
  123. dispatch >= "0.4.1"
  124. dkim
  125. dkim-bin
  126. dkim-mirage
  127. dkml-install
  128. dkml-install-installer
  129. dkml-install-runner
  130. dkml-package-console
  131. dns >= "4.0.0"
  132. dns-cli
  133. dns-client >= "4.6.0"
  134. dns-forward < "0.9.0"
  135. dns-forward-lwt-unix
  136. dns-resolver
  137. dns-server
  138. dns-tsig
  139. dnssd
  140. dnssec
  141. docfd >= "2.2.0"
  142. dog < "0.2.1"
  143. domain-name
  144. dream
  145. dream-pure
  146. duff
  147. dune-release >= "1.0.0"
  148. duration >= "0.1.1"
  149. eio < "0.12"
  150. eio_linux < "0.12"
  151. eio_windows < "0.12"
  152. emile
  153. encore
  154. eqaf >= "0.5"
  155. equinoxe
  156. equinoxe-cohttp
  157. equinoxe-hlc
  158. eris
  159. eris-lwt
  160. ezgzip
  161. ezjsonm >= "0.4.2" & < "1.3.0"
  162. ezjsonm-lwt
  163. FPauth
  164. FPauth-core
  165. FPauth-responses
  166. FPauth-strategies
  167. faraday != "0.2.0"
  168. farfadet
  169. fat-filesystem >= "0.12.0"
  170. ff
  171. ff-pbt
  172. flex-array
  173. fsevents-lwt
  174. functoria >= "2.2.0"
  175. functoria-runtime >= "2.2.0" & < "3.0.1" | = "3.1.2"
  176. geojson
  177. geoml >= "0.1.1"
  178. git = "1.4.10" | = "1.5.0" | >= "1.5.2" & != "1.10.0"
  179. git-cohttp
  180. git-cohttp-mirage
  181. git-cohttp-unix
  182. git-mirage
  183. git-unix >= "1.10.0" & != "2.1.0"
  184. git_split
  185. gitlab-unix
  186. glicko2
  187. gmap >= "0.3.0"
  188. gobba
  189. gpt
  190. graphql
  191. graphql-async
  192. graphql-cohttp >= "0.13.0"
  193. graphql-lwt
  194. graphql_parser != "0.11.0"
  195. graphql_ppx >= "0.7.1"
  196. h1_parser
  197. h2
  198. hacl
  199. hacl-star >= "0.6.0"
  200. hacl_func
  201. hacl_x25519 >= "0.2.0"
  202. highlexer
  203. hkdf
  204. hockmd
  205. html_of_jsx
  206. http
  207. http-multipart-formdata < "2.0.0"
  208. httpaf >= "0.2.0"
  209. httpun
  210. httpun-ws
  211. hvsock
  212. icalendar >= "0.1.4"
  213. imagelib >= "20200929"
  214. index
  215. inferno >= "20220603"
  216. influxdb-async
  217. influxdb-lwt
  218. inquire < "0.2.0"
  219. interval-map
  220. iomux
  221. irmin < "0.8.0" | >= "0.9.6" & != "0.11.1" & < "1.0.0" | >= "2.0.0" & != "2.3.0"
  222. irmin-bench >= "2.7.0"
  223. irmin-chunk < "1.3.0" | >= "2.3.0"
  224. irmin-cli
  225. irmin-containers
  226. irmin-fs < "1.3.0" | >= "2.3.0"
  227. irmin-git < "2.0.0" | >= "2.3.0"
  228. irmin-graphql >= "2.3.0"
  229. irmin-http < "2.0.0"
  230. irmin-mem < "1.3.0" | >= "2.3.0"
  231. irmin-pack >= "2.4.0" & != "2.6.1"
  232. irmin-pack-tools
  233. irmin-test >= "2.2.0" & < "3.0.0"
  234. irmin-tezos
  235. irmin-tezos-utils
  236. irmin-unix >= "1.0.0" & < "1.3.3" | >= "2.4.0" & != "2.6.1"
  237. irmin-watcher != "0.3.0"
  238. jekyll-format
  239. jerboa
  240. jitsu
  241. jose
  242. json-data-encoding >= "0.9"
  243. json_decoder
  244. jsonxt
  245. junit_alcotest
  246. jwto
  247. ke >= "0.2"
  248. kkmarkdown
  249. lambda-runtime
  250. lambda_streams
  251. lambda_streams_async
  252. lambdapi >= "2.0.0"
  253. lambdoc >= "1.0-beta4"
  254. ledgerwallet-tezos >= "0.2.1" & < "0.4.0"
  255. letters
  256. lmdb >= "1.0"
  257. logical
  258. logtk >= "1.6"
  259. lp
  260. lp-glpk
  261. lp-glpk-js
  262. lp-gurobi
  263. lru
  264. lt-code
  265. luv
  266. mbr-format >= "1.0.0"
  267. mdx >= "1.6.0"
  268. mec
  269. mechaml = "1.0.0" | >= "1.2.1"
  270. merge-queues >= "0.2.0"
  271. merge-ropes >= "0.2.0"
  272. metrics
  273. middleware
  274. mimic
  275. minicaml = "0.3.1" | >= "0.4"
  276. mirage >= "4.0.0~beta1"
  277. mirage-block-partition
  278. mirage-block-ramdisk = "0.3"
  279. mirage-channel >= "4.0.0"
  280. mirage-channel-lwt < "3.1.0"
  281. mirage-crypto-ec != "0.9.2"
  282. mirage-flow >= "1.0.2" & < "1.2.0"
  283. mirage-flow-unix != "1.3.0" & < "1.5.0" | = "2.0.0" | >= "3.0.0"
  284. mirage-fs-mem
  285. mirage-fs-unix >= "1.2.0" & < "1.4.1"
  286. mirage-kv >= "2.0.0"
  287. mirage-kv-mem
  288. mirage-kv-unix >= "3.0.0"
  289. mirage-logs >= "0.3.0"
  290. mirage-nat
  291. mirage-net-unix >= "2.3.0"
  292. mirage-runtime >= "4.0.0~beta1" & < "4.5.0"
  293. mirage-tc
  294. mjson
  295. mmdb < "0.3.0"
  296. mnd
  297. monocypher
  298. mrmime >= "0.2.0"
  299. mrt-format
  300. msgpck >= "1.6"
  301. mssql >= "2.0.3"
  302. multibase
  303. multihash
  304. multihash-digestif
  305. multipart-form-data
  306. multipart_form
  307. multipart_form-eio
  308. multipart_form-lwt
  309. named-pipe
  310. nanoid
  311. nbd >= "4.0.3"
  312. nbd-tool
  313. nloge
  314. nocoiner
  315. non_empty_list
  316. OCADml >= "0.6.0"
  317. ocaml-r >= "0.4.0"
  318. ocaml-version >= "3.1.0"
  319. ocamlformat >= "0.13.0" & != "0.19.0~4.13preview" & < "0.25.1"
  320. ocamlformat-lib
  321. ocamlformat-rpc < "removed"
  322. ocamline
  323. ocluster < "0.3.0"
  324. odoc >= "1.4.0" & < "2.1.0"
  325. ohex
  326. oidc
  327. opam-0install
  328. opam-compiler
  329. opam-file-format >= "2.1.1"
  330. opentelemetry >= "0.6"
  331. opentelemetry-client-cohttp-lwt >= "0.6"
  332. opentelemetry-client-ocurl >= "0.6"
  333. opentelemetry-cohttp-lwt >= "0.6"
  334. opentelemetry-lwt >= "0.6"
  335. opium >= "0.15.0"
  336. opium-graphql
  337. opium-testing
  338. opium_kernel
  339. orewa
  340. orgeat
  341. ortac-core
  342. osnap < "0.3.0"
  343. osx-acl
  344. osx-attr
  345. osx-cf
  346. osx-fsevents
  347. osx-membership
  348. osx-mount
  349. osx-xattr
  350. otoggl
  351. owl >= "0.6.0" & != "0.9.0" & != "1.0.0"
  352. owl-base < "0.5.0"
  353. owl-ode >= "0.1.0" & != "0.2.0"
  354. owl-symbolic
  355. passmaker
  356. patch
  357. pbkdf
  358. pecu >= "0.2"
  359. pf-qubes
  360. pg_query >= "0.9.6"
  361. pgx >= "1.0"
  362. pgx_unix >= "1.0"
  363. pgx_value_core
  364. pgx_value_ptime
  365. phylogenetics
  366. piaf
  367. polyglot
  368. polynomial
  369. ppx_blob >= "0.3.0"
  370. ppx_deriving_cmdliner
  371. ppx_deriving_qcheck
  372. ppx_deriving_rpc
  373. ppx_deriving_yaml
  374. ppx_graphql >= "0.2.0"
  375. ppx_inline_alcotest
  376. ppx_parser
  377. ppx_protocol_conv >= "5.0.0"
  378. ppx_protocol_conv_json >= "5.0.0"
  379. ppx_protocol_conv_jsonm >= "5.0.0"
  380. ppx_protocol_conv_msgpack >= "5.0.0"
  381. ppx_protocol_conv_xml_light >= "5.0.0"
  382. ppx_protocol_conv_xmlm
  383. ppx_protocol_conv_yaml >= "5.0.0"
  384. ppx_repr
  385. ppx_subliner
  386. ppx_units
  387. ppx_yojson >= "1.1.0"
  388. pratter
  389. prbnmcn-ucb1 >= "0.0.2"
  390. prc
  391. preface
  392. pretty_expressive
  393. prettym
  394. proc-smaps
  395. producer < "0.2.0"
  396. progress
  397. prom
  398. prometheus < "1.2"
  399. prometheus-app
  400. protocell
  401. protocol-9p >= "0.3" & < "0.11.0" | >= "0.11.2"
  402. protocol-9p-unix
  403. psq
  404. pyast
  405. qcheck >= "0.18"
  406. qcheck-alcotest
  407. qcheck-core >= "0.18"
  408. quickjs
  409. radis
  410. randii
  411. reason-standard
  412. reparse >= "2.0.0" & < "3.0.0"
  413. reparse-unix < "2.1.0"
  414. resp
  415. resp-unix >= "0.10.0"
  416. rfc1951 < "1.0.0"
  417. routes < "2.0.0"
  418. rpc >= "7.1.0"
  419. rpclib >= "7.1.0"
  420. rpclib-async
  421. rpclib-lwt >= "7.1.0"
  422. rpmfile
  423. rubytt
  424. SZXX >= "4.0.0"
  425. salsa20
  426. salsa20-core
  427. sanddb >= "0.2"
  428. scaml >= "1.5.0"
  429. scrypt-kdf
  430. secp256k1 >= "0.4.1"
  431. secp256k1-internal
  432. semver >= "0.2.1"
  433. sendmail
  434. sendmail-lwt
  435. sendmsg
  436. server-reason-react
  437. session-cookie
  438. session-cookie-async
  439. session-cookie-lwt
  440. sherlodoc
  441. sihl < "0.2.0"
  442. sihl-type
  443. slug
  444. smol
  445. smol-helpers
  446. sodium-fmt
  447. solidity-alcotest
  448. spdx_licenses
  449. spectrum
  450. spin >= "0.7.0"
  451. squirrel
  452. ssh-agent
  453. ssl >= "0.6.0"
  454. stramon-lib
  455. styled-ppx
  456. syslog-rfc5424
  457. tcpip >= "2.4.2" & < "3.4.2" | >= "6.2.0" & < "7.0.0"
  458. tdigest < "2.1.0"
  459. term-indexing
  460. terminal
  461. terminal_size >= "0.1.1"
  462. terminus
  463. terminus-cohttp
  464. terminus-hlc
  465. terml
  466. textmate-language >= "0.3.0"
  467. textrazor
  468. tezos-base-test-helpers < "13.0"
  469. tezos-bls12-381-polynomial
  470. tezos-client-base < "12.0"
  471. tezos-crypto >= "8.0" & < "9.0"
  472. tezos-lmdb
  473. tezos-plompiler = "0.1.3"
  474. tezos-plonk = "0.1.3"
  475. tezos-signer-backends >= "8.0" & < "13.0"
  476. tezos-stdlib >= "8.0" & < "12.0"
  477. tezos-test-helpers < "12.0"
  478. tftp
  479. timedesc
  480. timere
  481. timmy
  482. timmy-jsoo
  483. timmy-unix
  484. tls >= "0.12.0"
  485. toc
  486. topojson
  487. topojsone
  488. transept
  489. twostep
  490. type_eq
  491. type_id
  492. typebeat
  493. typeid >= "1.0.1"
  494. tyre >= "0.4"
  495. tyxml >= "4.0.0"
  496. tyxml-jsx
  497. tyxml-ppx >= "4.3.0"
  498. tyxml-syntax
  499. uecc
  500. ulid
  501. universal-portal
  502. unix-dirent
  503. unix-errno >= "0.3.0"
  504. unix-fcntl >= "0.3.0"
  505. unix-sys-resource
  506. unix-sys-stat
  507. unix-time
  508. unstrctrd
  509. uring < "0.4"
  510. user-agent-parser
  511. uspf
  512. uspf-lwt
  513. uspf-unix
  514. utop >= "2.13.0"
  515. validate
  516. validator
  517. vercel
  518. vpnkit
  519. wayland >= "2.0"
  520. wcwidth
  521. websocketaf
  522. x509 >= "0.7.0"
  523. xapi-rrd >= "1.8.2"
  524. xapi-stdext-date
  525. xapi-stdext-encodings
  526. xapi-stdext-std >= "4.16.0"
  527. yaml < "3.2.0"
  528. yaml-sexp
  529. yocaml
  530. yocaml_yaml
  531. yojson >= "1.6.0"
  532. yojson-five
  533. yuscii >= "0.3.0"
  534. yuujinchou = "1.0.0"
  535. zar
  536. zed >= "3.2.2"
  537. zlist < "0.4.0"

Conflicts

None

OCaml

Innovation. Community. Security.