I’m a big fan of rigorous unit testing. I believe most slow integrated tests can be replaced by unit tests using mocks, provided that one writes a good set of both unit and contract tests for all pieces of code. Good static type systems and checkers go a long way in ensuring contracts; unfortunately, we don’t have that luxury in Elixir.
When I found Mox I was hooked; I loved that it requires to make the contracts explicit as behaviours. However I was disappointed to find that even though it requires the user to write typespecs for the functions, those are actually never used unless you use Dialyzer. This kind of information might even be dangerous without automated checks, because when somebody forgets to update it when changing the implementations, it becomes misleading and can do more harm then good.
In Elixir, the typespecs are first class citizens and are available as code during runtime. So why not check them? That’s what I did.
The result is Hammox, an enhanced, backwards compatible version of Mox which will stop you any time you pass or return a value not conforming to typespec. In its short lifetime it already saved me from countless bugs during various refactorings of my company’s sizeable Elixir backend system.
I’m loving it, and hopefully anyone who uses Mox, or any kind of mocked unit tests, will love it too. Can I get some feedback?