FlowAssertions.Define.Tabular (Flow Assertions v0.7.1) View Source
Generate "runners" used in tabular tests of assertions and assertion-like functions.
A typical example:
a = assertion_runners_for(&assert_equal/2)
["value", "value"] |> a.pass.()
["value", " "] |> a.fail.("Assertion with === failed")
|> a.plus.(left: "value", right: " ")
assert_equal
takes two arguments, and those are passed to the runner
functions (pass
and fail
) in a list.
If there's only a single argument, the enclosing list can be omitted:
a = assertion_runners_for(&assert_ok/1)
:ok |> a.pass.()
:error |> a.fail.(Messages.not_ok)
The functions created are, unless noted in the individual descriptions below, these:
pass: A runner that applies a function to arguments and checks for success. What, precisely, counts as success depends on the function under test. For a flow assertion, success means returning its first argument.
In the examples above,
pass
took only one argument (the value on the left side of the|>
). That's because the expected output ofassert_ok
is trivial. In other cases, you must give the expected output topass
. That looks like this:test "ok_content" do a = content_runners_for(&ok_content/1) {:ok, "content"} |> a.pass.("content") ...
fail: takes an argument that is matched against an
AssertionError
.- A string, which must match the error message exactly.
- A regular expression, like
~r/different lengths/
, which can match part of the message. - A keyword list, which is checked with
FlowAssertions.MapA.assert_fields/2
. A typical use might be:
Note that the left and right values are the true values from[ {1, 2}, 3 ] |> a.fail.(left: {1, 2}, right: 3)
ExUnit.AssertionError
, not string versions as frominspect
. If the error message is also to be tested, it should be given in keyword form:
Warning: field names likea.fail.(left: "a", right: "bb", message: ~r/different lengths/)
:left
,:right
, and:message
are not part of the public API ofExUnit.AssertionError
, so those names could change in the future (though I doubt it). See theAssertionError
source for some other fields.
plus: Provides an arguably nicer way to check both the
:message
and otherAssertionError
fields. It's appended tofail
like this:[datetime] |> a.fail.(Messages.wrong_struct_name(NaiveDateTime, Date)) |> a.plus.(left: datetime)
inspect_. This is a function to help with test-writing workflow. It mainly exists because it's hard for me to get the display of error information right without looking at it. That is, a check like this:
[ [1, 2, 3], [7, 1, 3] ] |> a.fail.(~r/different elements/) |> a.plus.(...)
is likely to come from looking a message like the one at the end of this bullet list, then tweaking the message and maybe the
code:
,left:
, orright:
output. I only "solidify" the final design in a test after everything looks right. Normally, producing such error output would mean writing throwaway code, butinspect_
avoids that. Once I pick test inputs (the values before the|>
), I just write this:[ [1, 2, 3], [7, 1, 3] ] |> a.inspect_.(~r/a first message version/) ^^^^^^^^
The trailing
_
indicates thatinspect_
is replacing a function that takes one argument. Useinspect
to replace a function with zero arguments, likepass
.
Beware the typo
The most common mistake I make with this library is this kind of typo:
["", "a" ] |> a.fail(~s/Checker `has_slice("a")` failed/)
There should be a period after fail
. The result (as of Elixir 1.11) is
** (ArgumentError) you attempted to apply a function on %{arity: 1,
fail: #Function<5.65840318/2 in FlowAssertions.Define.Tabular.make_
assertion_fail/1>, inspect: #Function<0.65840318/1 in FlowAssertion
...
Assertions.Define.Tabular.start/2>}. Modules (the first argument of
apply) must always be an atom
... which is perhaps not as clear as it should be. But now you're forewarned.
Link to this section Summary
Functions
Create runners for flow-style assertions.
Create runners for functions like those in FlowAssertions.Checkers
.
Create runners for functions that either return part of a compound
value or raise an AssertionError
.
Adjust the results of a fail
function to assert a value for left:
.
Create runners for regular Elixir assertions.
Link to this section Functions
Create runners for flow-style assertions.
a = assertion_runners_for(&assert_empty/1)
asserter
should be a function that either raises an AssertionError
or returns its first argument.
pass
and fail
are as described above.
Create runners for functions like those in FlowAssertions.Checkers
.
a = checker_runners_for(&in_any_order/1)]
# actual checked against
[ [1, 2, 3], [1, 2, 3] ] |> a.pass.()
[ [1, 2, 3], [7, 1, 3] ] |> a.fail.(~r/different elements/)
checker
should be a function that can return a
FlowAssertions.Define.Defchecker.Failure
value. As for what that
means... Well, as of late 2020, checker creation is not documented.
Create runners for functions that either return part of a compound
value or raise an AssertionError
.
Consider FlowAssertions.MiscA.ok_content
:
a = content_runners_for(&ok_content/1)
{:ok, "content"} |> a.pass.("content")
:ok |> a.fail.(Messages.not_ok_tuple)
{:error, "content"} |> a.fail.(Messages.not_ok_tuple)
Note that pass
takes a single value.
Adjust the results of a fail
function to assert a value for left:
.
Suppose you create tabular functions for FlowAssertions.MiscA.ok_content
like this:
a = content_runners_for(&ok_content/1) |> left_is_actual
^^^^^^^^^^^^^^^^^
Then a table entry like this:
:ok |> a.fail.(Messages.not_ok_tuple)
... will check that the :left
value of the AssertionError
is :ok
. (That is,
it is the argument given to ok_content
). There is no need to add a
|> plus.(left: :ok)
.
If the function under test takes more than one argument, the left:
value must
be the first element of the list on the left-hand side of the |>
. That is,
in the following, the left:
value is checked to be [1]
.
a = assertion_runners_for(&assert_equal/2) |> left_is_actual
[[1], 1] |> x.fail.("Assertion with === failed")
Create runners for regular Elixir assertions.
asserter
should be a function that raises an AssertionError
or
returns an unspecified value. pass
, then, always succeeds when there's no
error.