Noam Zilberstein | af5d64a | 2014-07-31 15:44:13 -0700 | [diff] [blame] | 1 | {-# OPTIONS_GHC -fno-warn-orphans #-} |
| 2 | |
| 3 | module Thrift.Arbitraries where |
| 4 | |
| 5 | import Data.Bits() |
| 6 | |
| 7 | import Test.QuickCheck.Arbitrary |
| 8 | |
| 9 | import Control.Applicative ((<$>)) |
| 10 | import Data.Map (Map) |
| 11 | import qualified Data.Map as Map |
| 12 | import qualified Data.Set as Set |
| 13 | import qualified Data.Vector as Vector |
| 14 | import qualified Data.Text.Lazy as Text |
| 15 | import qualified Data.HashSet as HSet |
| 16 | import qualified Data.HashMap.Strict as HMap |
| 17 | import Data.Hashable (Hashable) |
| 18 | |
| 19 | import Data.ByteString.Lazy (ByteString) |
| 20 | import qualified Data.ByteString.Lazy as BS |
| 21 | |
| 22 | -- String has an Arbitrary instance already |
| 23 | -- Bool has an Arbitrary instance already |
| 24 | -- A Thrift 'list' is a Vector. |
| 25 | |
| 26 | instance Arbitrary ByteString where |
| 27 | arbitrary = BS.pack . filter (/= 0) <$> arbitrary |
| 28 | |
| 29 | instance (Ord k, Arbitrary k, Arbitrary v) => Arbitrary (Map k v) where |
| 30 | arbitrary = Map.fromList <$> arbitrary |
| 31 | |
| 32 | instance (Ord k, Arbitrary k) => Arbitrary (Set.Set k) where |
| 33 | arbitrary = Set.fromList <$> arbitrary |
| 34 | |
| 35 | instance (Arbitrary k) => Arbitrary (Vector.Vector k) where |
| 36 | arbitrary = Vector.fromList <$> arbitrary |
| 37 | |
| 38 | instance Arbitrary Text.Text where |
| 39 | arbitrary = Text.pack . filter (/= '\0') <$> arbitrary |
| 40 | |
| 41 | instance (Eq k, Hashable k, Arbitrary k) => Arbitrary (HSet.HashSet k) where |
| 42 | arbitrary = HSet.fromList <$> arbitrary |
| 43 | |
| 44 | instance (Eq k, Hashable k, Arbitrary k, Arbitrary v) => |
| 45 | Arbitrary (HMap.HashMap k v) where |
| 46 | arbitrary = HMap.fromList <$> arbitrary |
| 47 | |
| 48 | {- |
| 49 | To handle Thrift 'enum' we would ideally use something like: |
| 50 | |
| 51 | instance (Enum a, Bounded a) => Arbitrary a |
| 52 | where arbitrary = elements (enumFromTo minBound maxBound) |
| 53 | |
| 54 | Unfortunately this doesn't play nicely with the type system. |
| 55 | Instead we'll generate an arbitrary instance along with the code. |
| 56 | -} |
| 57 | |
| 58 | {- |
| 59 | There might be some way to introspect on the Haskell structure of a |
| 60 | Thrift 'struct' or 'exception' but generating the code directly is simpler. |
| 61 | -} |