{-# OPTIONS_GHC -fno-warn-orphans #-} module Thrift.Arbitraries where import Data.Bits() import Test.QuickCheck.Arbitrary import Control.Applicative ((<$>)) import Data.Map (Map) import qualified Data.Map as Map import qualified Data.Set as Set import qualified Data.Vector as Vector import qualified Data.Text.Lazy as Text import qualified Data.HashSet as HSet import qualified Data.HashMap.Strict as HMap import Data.Hashable (Hashable) import Data.ByteString.Lazy (ByteString) import qualified Data.ByteString.Lazy as BS -- String has an Arbitrary instance already -- Bool has an Arbitrary instance already -- A Thrift 'list' is a Vector. instance Arbitrary ByteString where arbitrary = BS.pack . filter (/= 0) <$> arbitrary instance (Arbitrary k) => Arbitrary (Vector.Vector k) where arbitrary = Vector.fromList <$> arbitrary instance Arbitrary Text.Text where arbitrary = Text.pack . filter (/= '\0') <$> arbitrary instance (Eq k, Hashable k, Arbitrary k) => Arbitrary (HSet.HashSet k) where arbitrary = HSet.fromList <$> arbitrary instance (Eq k, Hashable k, Arbitrary k, Arbitrary v) => Arbitrary (HMap.HashMap k v) where arbitrary = HMap.fromList <$> arbitrary {- To handle Thrift 'enum' we would ideally use something like: instance (Enum a, Bounded a) => Arbitrary a where arbitrary = elements (enumFromTo minBound maxBound) Unfortunately this doesn't play nicely with the type system. Instead we'll generate an arbitrary instance along with the code. -} {- There might be some way to introspect on the Haskell structure of a Thrift 'struct' or 'exception' but generating the code directly is simpler. -}