How to set tests as pending in your ExUnit suite
Table of Contents
Elixir’s built in testing library is called ExUnit. It’s a proper testing
framework, which, although simple, gives the developers a lot of power and
flexibility. If you come from Ruby land, I am sure you’ve been in a position
where you want to set a certain test to be skipped. For example, RSpec in Ruby
does it with the pending method. Let’s see how we can customize our test
suite so ExUnit can skip over tests in our test suite.
Tags #
ExUnit comes with this neat feature called tags. Tags are basically module variables in the test module. In the context of the test, you can think of them as annotations. So, by tagging a test, the value of the tag can be used in the test itself, by passing the context.
defmodule FileTest do
# Changing directory cannot be async
use ExUnit.Case, async: false
setup context do
# Read the :cd tag value
if cd = context[:cd] do
prev_cd = File.cwd!
File.cd!(cd)
on_exit fn -> File.cd!(prev_cd) end
end
:ok
end
@tag cd: "fixtures"
test "reads utf-8 fixtures" do
File.read("hello")
end
endNow, how can we utilize tags to get RSpec’s pending funcionality in ExUnit?
Running tests by tag #
Running tests by tag is really simple in ExUnit. Basically, ExUnit has the ability to skip/include any test by it’s tag.
For example, let’s say you have tagged couple of tests that are brittle:
defmodule SomeTest do
use ExUnit.Case, async: false
@tag :brittle
test "some brittle code" do
# test here...
end
endNow, you can run only the tests with the brittle tag, by executing:
mix test test/some_test.exs --include brittleHaving this in mind, let’s introduce a custom tag of ours so we can make our
tests pending.
Custom tags #
Although the word “custom” used in programming often means something complicated done from scracth by the programmer, in this case it’s very simple.
Let’s take any test and tag it with :pending.
defmodule SomeTest do
use ExUnit.Case, async: false
@tag :pending
test "always true" do
assert 1 == 1
end
endWe can exclude this “pending” test by executing our test suite with:
mix test --exclude pendingThis will skip any tests that have the :pending tag. Simple as that, we have
RSpec’s pending functionality just by excluding a tag. If you are wondering
if we can make the exclusion of the tag automatic - yes we can. In your
test_helper.exs add this line, before the ExUnit.start line:
ExUnit.configure(exclude: [pending: true])This configuration will tell ExUnit that it should always skip tests that are
tagged as :pending. Now, every time you run your tests, the tests tagged with
:pending will be excluded.
Do you have any other interesting configuration trick in your ExUnit suite? Feel free to share them in the comments, I would love to learn more about ExUnit!
Thanks for reading!