Artbitrary
While setting up a test suite for CBaaS, I needed to produce random images. An Arbitrary
instance for JuicyPixel’s DynamicImage
type is the way to do this in Haskell. While producing a list of random Pixel
s at different colors would have been enough to make the tests work, the output wouldn’t be very interesting to look at - it would just be colorful white noise.
What if we use QuickCheck’s ability to produce arbitrary functions, and pass an arbitrary (Int, Int) -> Pixel
function into generateImage
? Instead of fully random pixels, we would get random functions that determine a Pixel’s color from its location (it may seem like the difference between random Pixels and random functions producing Pixels are one and the same - but the random pixels case is actually an extremely specific function).
How does QuickCheck produce random functions? What would I expect the output to look like? I have no idea, although I’d to look into it. In the meantime, here are some examples.
data:image/s3,"s3://crabby-images/efe68/efe689e20251f081920904e388b8ece91b458abe" alt=""
An example image created with QuickCheck’s Arbitrary function generator
Another generated image seems to have horizontal bands in one color channel. What sort of (Int, Int) -> Pixel
would produce it?
data:image/s3,"s3://crabby-images/bebb8/bebb833c8d886db222a4726948b91469c47edf78" alt="Different spatial frequencies in different color channels"
Different spatial frequencies in different color channels
data:image/s3,"s3://crabby-images/7b3be/7b3be1f637b59a1594a67768c6a010945c7fac52" alt=""
data:image/s3,"s3://crabby-images/6b1c2/6b1c26877d3466837f071c7cf348bb5f3b0843ca" alt=""
data:image/s3,"s3://crabby-images/46d3a/46d3ab00516cfc9c9bb1805d622e7d434b5e569c" alt=""
An interesting common pattern: horizontal or vertical striations
data:image/s3,"s3://crabby-images/118ec/118ec41204740a2650c1cb3756a85d70f3569eb8" alt="A picture with a kind of even-odd structure in the rows"
A picture with a kind of even-odd structure in the rows
data:image/s3,"s3://crabby-images/d2fa7/d2fa7c8e7fd86e6ed7da4c11d1452afd1c19600f" alt=""
data:image/s3,"s3://crabby-images/8c067/8c067dbcc0fb261c5f951943e8d8f16969f63f4e" alt=""
data:image/s3,"s3://crabby-images/68797/68797c83c409466d61002ffc6167aa9787e3b4da" alt=""
Not all of the pictures generated were this interesting. The majority looked like confetti-colored white noise, as you would expect from a function that throws away the input coordinates and returns a randomly colored pixel. Actually the last picture above is neat: half way through it changes spatial frequency.
Some things decidedly did not show up in the output. I found no examples where the x
and y
axes were independent (no straight horizontal bars, straight vertical bars, or checkerboards). No simple dependencies showed up either (e.g. if x == y then Blue else Green
). Many were not encodable in PNG and so aren’t shown here. Many are solid black (QuickCheck focuses its testing effort on corner cases - I imagine \(x,y) -> Black
is a value that gets picked often while other parameters (width
, height
) are varied. Below are all the pictures that came out in this test run, to give a sense of the distribution.
This is the code used to Gen
erate an Image a
for one of JuicyPixel
s Pixes
types. It comes from on CBaaS. There’s nothing to it - all the magic is in QuickCheck’s Function
module. Proxy
isn’t important to this post - it’s used in my test suite to pick a particular Pixel
type.
genImg :: (Pixel a, Arbitrary a) => Proxy a -> Gen (Image a)
genImg _ = do
wid <- choose (100, 300)
hgt <- choose (100, 300)
f :: Fun (Int,Int) a <- arbitrary
let img = generateImage (curry (apply f)) wid hgt
return img