Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/ulidtko/cabal-doctest
A Setup.hs shim for running doctests
https://github.com/ulidtko/cabal-doctest
doctest haskell
Last synced: about 19 hours ago
JSON representation
A Setup.hs shim for running doctests
- Host: GitHub
- URL: https://github.com/ulidtko/cabal-doctest
- Owner: ulidtko
- License: bsd-3-clause
- Created: 2017-01-31T07:01:31.000Z (almost 8 years ago)
- Default Branch: master
- Last Pushed: 2024-11-22T15:05:49.000Z (about 1 month ago)
- Last Synced: 2024-12-21T12:10:49.348Z (8 days ago)
- Topics: doctest, haskell
- Language: Haskell
- Homepage: https://hackage.haskell.org/package/cabal-doctest
- Size: 129 KB
- Stars: 31
- Watchers: 5
- Forks: 24
- Open Issues: 11
-
Metadata Files:
- Readme: README.md
- Changelog: changelog.md
- License: LICENSE
Awesome Lists containing this project
README
cabal-doctest
=============[![Hackage](https://img.shields.io/hackage/v/cabal-doctest.svg)](https://hackage.haskell.org/package/cabal-doctest) [![Haskell-CI](https://github.com/ulidtko/cabal-doctest/actions/workflows/haskell-ci.yml/badge.svg?branch=master)](https://github.com/ulidtko/cabal-doctest/actions/workflows/haskell-ci.yml) [![stack test](https://github.com/ulidtko/cabal-doctest/actions/workflows/stack-test.yml/badge.svg)](https://github.com/ulidtko/cabal-doctest/actions/workflows/stack-test.yml)
A `Setup.hs` helper for running [doctests][doctest].
[doctest]: https://github.com/sol/doctest#readme
Why this exists
---------------**Doctesting** is a nifty technique that stimulates 3 good things to happen:
* library documentation gains *runnable code examples* that are also tested;
* library test suite gains *documented usage examples* as "tests for free";
* get both of the above for the price of one.That's what the [doctest][] tool does — not this package! — just for clarity.
Off the shelf, `doctest` doesn't require any package management mumbo-jumbo:
you just run it on a source file with haddocks with doctests.Issues come in when library authors and maintainers wish to integrate doctests
into CI pipelines. When doctests start to require dependencies or non-default
compiler flags: that's when it gets hairy. There, if you want `stack test` and/or
`cabal test` to run doctests too with minimal shenanigans, then read on.Among different available approaches, this package `cabal-doctest` helps with
one, which is known as [custom setup][], `build-type: Custom` more precisely.
You should stick to the default `build-type: Simple`, unless you know what
you're doing.In a nutshell, this custom Setup.hs shim generates a module `Build_doctests`
that allows your doctest driver `test-suite` to look like this:```haskell
module Main whereimport Build_doctests (flags, pkgs, module_sources)
import Test.Doctest (doctest)main :: IO ()
main = doctest (flags ++ pkgs ++ module_sources)
```More detailed examples below.
Regardless of the name, this **also works with Stack**.
For old versions of stack, cabal-install, GHC, see [caveats](#notes) below.
[custom setup]: https://cabal.readthedocs.io/en/stable/cabal-package-description-file.html#pkg-field-build-type
Simple example
--------------[simple example]: https://github.com/ulidtko/cabal-doctest/tree/master/simple-example
[simple-example.cabal]: https://github.com/ulidtko/cabal-doctest/tree/master/simple-example/simple-example.cabalFollow [simple example][] for the common case of a single-library `.cabal` package with doctests.
To recap the example's code:
1. specify `build-type: Custom` in your `.cabal` file;
2. declare dependencies of `Setup.hs`:
```
custom-setup
setup-depends:
base >= 4 && <5,
cabal-doctest >= 1 && <1.1
```See [Notes](#notes) below for a caveat with cabal-install < 2.4.
3. Populate `Setup.hs` like so:
```haskell
module Main whereimport Distribution.Extra.Doctest (defaultMainWithDoctests)
main :: IO ()
main = defaultMainWithDoctests "doctests"
```Assuming your test-suite is called `doctests`, this `Setup` will generate a `Build_doctests`
module during package build. If your test-suite goes by name `foo`,
`defaultMainWithDoctests "foo"` creates a `Build_foo` module.4. Use the generated module in a testsuite, simply like so:
```haskell
module Main whereimport Build_doctests (flags, pkgs, module_sources)
import Data.Foldable (traverse_)
import System.Environment (unsetEnv)
import Test.DocTest (doctest)main :: IO ()
main = do
traverse_ putStrLn args -- optionally print arguments
unsetEnv "GHC_ENVIRONMENT" -- see 'Notes'; you may not need this
doctest args
where
args = flags ++ pkgs ++ module_sources
```Ultimately, `cabal test` or `stack test` should run the doctests of your package.
Example with multiple cabal components
--------------------------------------`cabal-doctest` also supports more exotic use cases where a `.cabal` file
contains more components with doctests than just the main library, including:* doctests in executables,
* doctests in internal libraries (if using `Cabal-2.0` or later).Unlike the simple example shown above, these examples involve _named_
components. You don't need to change the `Setup.hs` script to support
this use case. However, in this scenario `Build_doctests` will generate extra
copies of the `flags`, `pkgs`, and `module_sources` values for each additional
named component.The simplest approach is to use `x-doctest-components` field in `.cabal`:
```
x-doctest-components: lib lib:internal exe:example
```In that case, the test driver is generally:
```haskell
module Main whereimport Build_doctests (Component (..), components)
import Data.Foldable (for_)
import System.Environment (unsetEnv)
import Test.DocTest (doctest)main :: IO ()
main = for_ components $ \(Component name flags pkgs sources) -> do
print name
putStrLn "----------------------------------------"
let args = flags ++ pkgs ++ sources
for_ args putStrLn
unsetEnv "GHC_ENVIRONMENT"
doctest args
```There is also a more explicit approach: if you have an executable named `foo`, then
`Build_doctest` will contain `flags_exe_foo`, `pkgs_exe_foo`, and `module_sources_exe_foo`.
If the name has hyphens in it (e.g., `my-exe`), `cabal-doctest` will convert them to
underscores (e.g., you'd get `flags_my_exe`, `pkgs_my_exe`, `module_sources_my_exe`).
Internal library `bar` values will have a `_lib_bar` suffix.An example testsuite driver for this use case might look like this:
```haskell
module Main whereimport Build_doctests
(flags, pkgs, module_sources,
flags_exe_my_exe, pkgs_exe_my_exe, module_sources_exe_my_exe)
import Data.Foldable (traverse_)
import System.Environment (unsetEnv)
import Test.DocTestmain :: IO ()
main = do
unsetEnv "GHC_ENVRIONMENT"
-- doctests for library
traverse_ putStrLn libArgs
doctest libArgs-- doctests for executable
traverse_ putStrLn exeArgs
doctest exeArgs
where
libArgs = flags ++ pkgs ++ module_sources
exeArgs = flags_exe_my_exe ++ pkgs_exe_my_exe ++ module_sources_exe_my_exe
```See the [multiple-components-example][].
[multiple-components-example]: https://github.com/ulidtko/cabal-doctest/tree/master/multiple-components-example
Additional configuration
------------------------The `cabal-doctest` based `Setup.hs` supports a few extensions fields
in `pkg.cabal` files to customize the `doctest` runner behavior, without
customizing the default `doctest.hs`.```
test-suite doctests:
if impl(ghc >= 8.0)
x-doctest-options: -fdiagnostics-color=never
x-doctest-source-dirs: test
x-doctest-modules: Servant.Utils.LinksSpec
```* `x-doctest-options` Additional arguments passed into `doctest` command.
* `x-doctest-modules` Additional modules to `doctest`. May be useful if you
have doctests in tests or executables (i.e not the default library component).
* `x-doctest-src-dirs` Additional source directories to look for the modules.Notes
-----* If support for cabal-install < 2.4 is required, you'll have to
add `Cabal` to `setup-depends`; see issue [haskell/cabal#4288][].* Some versions of `Cabal` (for instance, 2.0) can choose to build a
package's `doctest` test suite _before_ the library. However, in order for
`cabal-doctest` to work correctly, the library _must_ be built first, as
`doctest` relies on the presence of generated files that are only created
when the library is built. See [#19][].A hacky workaround for this problem is to depend on the library itself in a
`doctests` test suite. See [simple-example.cabal][]
for a demonstration. (This assumes that the test suite has the ability to
read build artifacts from the library, a separate build component. In
practice, this assumption holds, which is why this library works at all.)* `custom-setup` section is supported starting from `cabal-install-1.24`.
For older `cabal-install's` you have to install custom setup dependencies
manually.* `stack` respects `custom-setup` starting from version 1.3.3. Before that
you have to use `explicit-setup-deps` setting in your `stack.yaml`;
[stack#2094][].* With base < 4.7 (GHC < 7.8, pre-2014), `System.Environment.unsetEnv` function
will need to be imported from `base-compat` library. It is already in transitive
dependencies of `doctest`. Simply declare the dependency upon `base-compat`, and
then `import System.Environment.Compat (unsetEnv)` if you need that old GHC.* You can use `x-doctest-options` field in `test-suite doctests` to
pass additional flags to the `doctest`.* For `build-type: Configure` packages, you can use
`defaultMainAutoconfWithDoctests` function to make custom `Setup.hs` script.* If you use the default `.` in `hs-source-dirs`, then running `doctests`
might fail with weird errors (ambiguous module errors). Workaround is
to move sources under `src/` or some non-top-level directory.* The `extensions:` field isn't supported. Upgrade your `.cabal` file to use at least
`cabal-version: >= 1.10` and use `default-extensions` or `other-extensions`.* If you use QuickCheck properties (`prop>`) in your doctests,
the `test-suite doctest` should depend on `QuickCheck` and `template-haskell`.
This is a little HACK: These dependencies aren't needed to build the
`doctests` test-suite executable. However, as we let `Cabal` resolve
dependencies, we can pass the resolved (and installed!) package identifiers to
to the `doctest` command. This way, `QuickCheck` and `template-haskell` are
available to `doctest`, otherwise you'll get errors like:```
Variable not in scope:
mkName
:: [Char]
-> template-haskell-2.11.1.0:Language.Haskell.TH.Syntax.Name
```or
```
Variable not in scope:
polyQuickCheck
:: Language.Haskell.TH.Syntax.Name -> Language.Haskell.TH.Lib.ExpQ
```* From version 2, Stack sets the `GHC_ENVIRONMENT` variable, and GHC
(as invoked by `doctest`) will pick that up. This is undesirable:
`cabal-doctest` passes all the necessary information on the command
line already, and can lead to ambiguous module errors as GHC will
load the environment in addition to what `cabal-doctest` instructs
it to.Hence, `cabal-doctest` tells GHC to ignore package environments
altogether on the command line. However, this is only possible since
GHC 8.2. If you are using `cabal-doctest` with Stack 2 and GHC 8.0
or earlier and seeing ambiguous module errors or other mysterious
failures, try manually unsetting `GHC_ENVIRONMENT` before invoking
`doctest`.* If you are on Nix. `doctest` will not pick up your version of GHC if you
don't point it towards it, and therefore will result in "cannot satisfy -package-id" errors.
You will need to set `NIX_GHC` and `NIX_GHC_LIBDIR` within your environment in order
for doctest to pick up your GHC. Put the following in `shell.nix` and run `nix-shell`.```nix
# shell.nix
{ pkgs ? import {} }:
let
myHaskell = (pkgs.haskellPackages.ghcWithHoogle (p: with p; [
# Put your dependencies here
containers
hslogger
]));
in
pkgs.mkShell {
name = "myPackage";# These environment variables are important. Without these,
# doctest doesn't pick up nix's version of ghc, and will fail
# claiming it can't find your dependencies
shellHook = ''
export NIX_GHC=${myHaskell}/bin/ghc
export NIX_GHC_LIBDIR=${myHaskell}/lib/ghc-8.10.7
'';
buildInputs = with pkgs; [
myHaskell
];
}
```[#19]: https://github.com/ulidtko/cabal-doctest/issues/19
[haskell/cabal#4288]: https://github.com/haskell/cabal/issues/4288
[stack#2094]: https://github.com/commercialhaskell/stack/issues/2094Copyright
---------Copyright 2017 Oleg Grenrus.
With contributions from:
* Ryan Scott
* Andreas Abel
* Max UlidtkoAvailable under the BSD 3-clause license.