View Source Chapter 4: Spies and Fakes
In Chapter 2: Patching and Chapter 3: Mock Values we saw how we can patch functions to return particular mock values.
There are two common cases for patching that have special helpers.
Spies
If a test wishes to assert / refute calls that happen to a module without actually changing the behavior of the module it can simply spy/1
the module. Spies behave identically to the original module but all calls are recorded so assert_called/1
, refute_called/1
, assert_any_called/2
, and refute_any_called/2
work as expected.
defmodule PatchExample do
use ExUnit.Case
use Patch
def example(value) do
String.upcase(value)
end
test "spies can see what calls happen without changing functionality" do
spy(String)
assert "HELLO" == example("hello")
assert_called String.upcase("hello")
end
end
Fakes
Sometimes we want to replace one module with another for testing, for example we might want to replace a module that connects to a real datastore with a fake that stores data in memory while providing the same API.
The fake/2,3
functions can be used to replace one module with another. The replacement module can be completely stand alone or can utilize the functionality of the replaced module, it will be made available through use of the real/1
function.
defmodule HighLatencyDatabase do
@latency System.convert_time_unit(20, :second, :microsecond)
def get(id) do
{elapsed, response} = :timer.tc(fn -> Patch.real(Database).get(id) end)
induce_latency(elapsed)
response
end
defp induce_latency(elapsed) when elapsed < @latency do
time_to_sleep = System.convert_time_unit(@latency - elapsed, :microsecond, :millisecond)
Process.sleep(time_to_sleep)
end
defp induce_latency(_), do: :ok
end
This fake module uses the real module to actually get the record from the database and then makes sure that a minimum amount of latency, in this case 20 seconds, is introduced before returning the result.
To swap out our real Database with our fake HighLatencyDatabase in a test we can now do the following
defmodule PatchExample do
use ExUnit.Case
use Patch
def example(value) do
String.upcase(value)
end
test "API raises TimeoutError when database is experiencing high latency" do
fake(Database, HighLatencyDatabase)
assert_raises TimeoutError, fn ->
API.get(:user, 1)
end
end
end