Unit and Fuzz testing support with Console/Html/String outputs.


Keywords
fuzz-testing, testing
License
BSD-3-Clause
Install
elm-package install elm-community/elm-test 4.2.0

Contributors

Richard Feldman Max Goldstein Alex Neslusan Filip Haglund Max S. New Jasper Woudenberg Noah Robert Looby Martin Janiczek Greg Thornton Bob Whitelock Rudolf Adamkovič Aaron VonderHaar Jaap Broekhuizen Fred Yankowski Zachary Kessin Ray Toal Arnaud Bailly Louis Pilfold Sebastian Graf Janis Voigtländer Pete Graeme Cross Kieran Trezona-le Comte


See all contributors


Documentation

elm-test Travis build Status

Write unit and fuzz tests for your Elm code, in Elm.

Quick Start

Here are three example tests:

suite : Test
suite =
    describe "The String module"
        [ describe "String.reverse" -- Nest as many descriptions as you like.
            [ test "has no effect on a palindrome" <|
                \_ ->
                    let
                        palindrome =
                            "hannah"
                    in
                        Expect.equal palindrome (String.reverse palindrome)

            -- Expect.equal is designed to be used in pipeline style, like this.
            , test "reverses a known string" <|
                \_ ->
                    "ABCDEFG"
                        |> String.reverse
                        |> Expect.equal "GFEDCBA"

            -- fuzz runs the test 100 times with randomly-generated inputs!
            , fuzz string "restores the original string if you run it again" <|
                \randomlyGeneratedString ->
                    randomlyGeneratedString
                        |> String.reverse
                        |> String.reverse
                        |> Expect.equal randomlyGeneratedString
            ]
        ]

This code uses a few common functions:

  • describe to add a description string to a list of tests
  • test to write a unit test
  • Expect to determine if a test should pass or fail
  • fuzz to run a function that produces a test several times with randomly-generated inputs

Check out a large real-world test suite for more.

Running tests locally

There are several ways you can run tests locally:

Here's how to set up and run your tests using the CLI test runner:

  1. Run npm install -g elm-test if you haven't already.
  2. cd into the project's root directory that has your elm-package.json.
  3. Run elm-test init. It will create a tests directory inside this one, with some files in it.
  4. Copy all the dependencies from elm-package.json into tests/elm-package.json. These dependencies need to stay in sync, so make sure whenever you change your dependencies in your current elm-package.json, you make the same change to tests/elm-package.json.
  5. Run elm-test.
  6. Edit tests/Example.elm to introduce new tests.

Running tests on CI

Here are some examples of running tests on CI servers:

Not running tests

During development, you'll often want to focus on specific tests, silence failing tests, or jot down many ideas for tests that you can't implement all at once. We've got you covered with skip, only, and todo:

wipSuite : Test
wipSuite =
    describe "skip, only, and todo"
        [ only <| describe "Marking this test as `only` means no other tests will be run!"
            [ test "This test will be run" <|
                  \_ -> 1 + 1 |> Expect.equal 2
            , skip <| test "This test will be skipped, even though it's in an only!" <|
                  \_ -> 2 + 3 |> Expect.equal 4
            ]
        , test "This test will be skipped because it has no only" <|
            \_ -> "left" |> Expect.equal "right"
        , todo "Make sure all splines are reticulated"
        ]

If you run this example, or any suite that uses one of these three functions, it will result in an incomplete test run. No tests failed, but you also didn't run your entire suite, so we can't call it a success either. Incomplete test runs are reported to CI systems as indistinguishable from failed test runs, to safeguard against accidentally committing a gutted test suite!

Strategies for effective testing

  1. Make impossible states unrepresentable so that you don't have to test that they can't occur.
  2. When doing TDD, treat compiler errors as a red test. So feel free to write the test you wish you had even if it means calling functions that don't exist yet!
  3. If your API is difficult for you to test, it will be difficult for someone else to use. You are your API's first client.
  4. Prefer fuzz tests to unit tests, where possible. If you have a union type with a small number of values, list them all and map over the list with a unit test for each. Unit tests are also great for when you know the edge cases, and for regression tests.
  5. If you're writing a library that wraps an existing standard or protocol, use examples from the specification or docs as unit tests.
  6. For simple functions, it's okay to copy the implementation to the test; this is a useful regression check. But if the implementation isn't obviously right, try to write tests that don't duplicate the suspect logic. The great thing about fuzz tests is that you don't have to arrive at the exact same value as the code under test, just state something that will be true of that value.
  7. Tests should be small and focused: call the code under test and set an expectation about the result. Setup code should be moved into reusable functions, or custom fuzzers. For example, a test to remove an element from a data structure should be given a nonempty data structure; it should not have to create one itself.
  8. If you find yourself inspecting the fuzzed input and making different expectations based on it, split each code path into its own test with a fuzzer that makes only the right kind of values.
  9. Consider using elm-verify-examples to extract examples in your docs into unit tests.
  10. Not even your test modules can import unexposed functions, so test them only as the exposed interface uses them. Don't expose a function just to test it. Every exposed function should have tests. (If you practice TDD, this happens automatically!)
  11. How do you know when to stop testing? This is an engineering tradeoff without a perfect answer. If you don't feel confident in the correctness of your code, write more tests. If you feel you are wasting time testing better spent writing your application, stop writing tests for now.

Application-specific techniques

There are a few extra ideas that apply to testing webapps and reusable view packages:

  1. Avoid importing your Main module. Most of your code belongs in other modules, so import those instead.
  2. Test your views using elm-html-test.
  3. To test effects, consider using elm-testable.
  4. There is currently no Elm solution for integration or end-to-end testing. Use your favorite PhantomJS or Selenium webdriver, such as Capybara.

Upgrading

From 3.1.0

Make sure you grab the latest versions of the test runner that you are using:

  • npm update -g elm-test
  • elm package install rtfeldman/html-test-runner

Fuzz.frequency now fails the test if the frequency is invalid, rather than return a Result. If you are using this function, you can remove your Err handling code. More likely you are using Fuzz.frequencyOrCrash, which you can replace with Fuzz.frequency.

Instead of using Test.filter to avoid running tests, use skip and only (see above for documentation).

We now forbid tests and suites to have descriptions that are blank, or that are identical across siblings or parents and children. If you get failures from this, rename your tests to be clearer about what they're testing.

From 0.17

You will need to delete elm-stuff and tests/elm-stuff.

If you are using the Node runner, you will need to install the latest version (npm update -g elm-test) and pull down the new Main.elm: curl -o tests/Main.elm https://raw.githubusercontent.com/rtfeldman/node-test-runner/3.0.1/templates/Main.elm

From 1.x and elm-check

legacy-elm-test provides a drop-in replacement for the ElmTest 1.0 API, except implemented in terms of the current elm-test. It also includes support for elm-check tests.

This lets you use the latest test runners right now, and upgrade incrementally.

Releases

Version Notes
4.0.0 Add only, skip, todo; change Fuzz.frequency to fail rather than crash on bad input, disallow tests with blank or duplicate descriptions.
3.1.0 Add Expect.all
3.0.0 Update for Elm 0.18; switch the argument order of Fuzz.andMap.
2.1.0 Switch to rose trees for Fuzz.andThen, other API additions.
2.0.0 Scratch-rewrite to project-fuzzball
1.0.0 ElmTest initial release