Working through Real World Haskell right now. Here’s a solution to a very early exercise in the book:
-- | 4) Counts the number of characters in a file
numCharactersInFile :: FilePath -> IO Int
numCharactersInFile fileName = do
contents <- readFile fileName
return (length contents)
My question is: How would you test this function? Is there a way to make a “mock” input instead of actually needing to interact with the file system to test it out? Haskell places such an emphasis on pure functions that I have to imagine that this is easy to do.
As Alexander Poluektov already pointed out, the code you are trying to test can easily be separated into a pure and an impure part.
Nevertheless I think it is good to know how to test such impure functions in haskell.
The usual approach to testing in haskell is to use quickcheck and that’s what I also tend to use for impure code.
Here is an example of how you might achieve what you are trying to do which gives you kind of a mock behavior * :
Now provide an alternative function (Testing against a model):
Provide an Arbitrary instance for the test environment:
Property testing against the model (using quickcheck for monadic code):
And a little helper function:
This will let quickCheck create some random files for you and test your implementation against a model function.
Of course you could also test some other properties depending on your usecase.
* Note about the my usage of the term mock behavior:
The term mock in the object oriented sense is perhaps not the best here. But what is the intention behind a mock?
It let’s you test code that needs access to a resource that usually is
By shifting the responsibility of providing such a resource to quickcheck, it suddenly becomes feasible to provide an environment for the code under test that can be verified after a test run.
Martin Fowler describes this nicely in an article about mocks :
“Mocks are … objects pre-programmed with expectations which form a specification of the calls they are expected to receive.”
For the quickcheck setup I’d say that files generated as input are “pre-programmed” such that we know about their size (== expectation). And then they are verified against our specification (== property).