{-# LANGUAGE TypeFamilies , DeriveGeneric , FlexibleContexts , OverloadedStrings , FlexibleInstances , StandaloneDeriving , UndecidableInstances , MultiParamTypeClasses #-} import Test.Tasty (defaultMain, testGroup, TestTree) import Test.Tasty.HUnit (testCase) import Test.Tasty.QuickCheck (testProperty) import Test.Serialization.Symbiote ( SymbioteT, register, firstPeer, secondPeer, SymbioteOperation (..), Symbiote (..), SimpleSerialization , First, Second, Generating, Operating, Topic, simpleTest, defaultSuccess, defaultFailure, defaultProgress) import Test.Serialization.Symbiote.Cereal () import Test.Serialization.Symbiote.Aeson () import Test.Serialization.Symbiote.Abides import Test.QuickCheck (Arbitrary (..)) import Test.QuickCheck.Gen (elements, oneof, scale, getSize) import Test.QuickCheck.Instances () import Data.Proxy (Proxy (..)) import qualified Data.Aeson as Json import qualified Data.Aeson.Types as Json import qualified Data.Serialize as Cereal import qualified Data.ByteString as BS import qualified Data.ByteString.Lazy as LBS import GHC.Generics (Generic) main :: IO () main = defaultMain tests tests :: TestTree tests = testGroup "All Tests" [ testGroup "Symbiote Sanity Checks" [ simpleTests , bytestringTests , jsonTests ] , testGroup "Local Isomorphisms" [ testGroup "Json" $ let go (n,p) = testProperty n (jsonIso p) in [ testGroup "Abides" [ go ("AbidesSemigroup [Int]", Proxy :: Proxy (AbidesSemigroup [Int])) , go ("AbidesMonoid [Int]", Proxy :: Proxy (AbidesMonoid [Int])) , go ("AbidesEq Int", Proxy :: Proxy (AbidesEq Int)) , go ("AbidesOrd Int", Proxy :: Proxy (AbidesOrd Int)) , go ("AbidesEnum Int", Proxy :: Proxy (AbidesEnum Int)) , go ("AbidesSemiring Int", Proxy :: Proxy (AbidesSemiring Int)) , go ("AbidesRing Int", Proxy :: Proxy (AbidesRing Int)) , go ("AbidesCommutativeRing Int", Proxy :: Proxy (AbidesCommutativeRing Int)) , go ("AbidesDivisionRing Int", Proxy :: Proxy (AbidesDivisionRing Int)) , go ("AbidesEuclideanRing Int", Proxy :: Proxy (AbidesEuclideanRing Int)) , go ("AbidesField Int", Proxy :: Proxy (AbidesField Int)) ] , testGroup "Symbiote" [ go ("Generating Int", Proxy :: Proxy (Generating Int)) , go ("Operating Int", Proxy :: Proxy (Operating Int)) , go ("First Int", Proxy :: Proxy (First Int)) , go ("Second Int", Proxy :: Proxy (Second Int)) , go ("Topic", Proxy :: Proxy Topic) ] ] , testGroup "Cereal" $ let go (n,p) = testProperty n (cerealIso p) in [ testGroup "Abides" [ go ("AbidesSemigroup [Int]", Proxy :: Proxy (AbidesSemigroup [Int])) , go ("AbidesMonoid [Int]", Proxy :: Proxy (AbidesMonoid [Int])) , go ("AbidesEq Int", Proxy :: Proxy (AbidesEq Int)) , go ("AbidesOrd Int", Proxy :: Proxy (AbidesOrd Int)) , go ("AbidesEnum Int", Proxy :: Proxy (AbidesEnum Int)) , go ("AbidesSemiring Int", Proxy :: Proxy (AbidesSemiring Int)) , go ("AbidesRing Int", Proxy :: Proxy (AbidesRing Int)) , go ("AbidesCommutativeRing Int", Proxy :: Proxy (AbidesCommutativeRing Int)) , go ("AbidesDivisionRing Int", Proxy :: Proxy (AbidesDivisionRing Int)) , go ("AbidesEuclideanRing Int", Proxy :: Proxy (AbidesEuclideanRing Int)) , go ("AbidesField Int", Proxy :: Proxy (AbidesField Int)) ] , testGroup "Symbiote" [ go ("Generating Int", Proxy :: Proxy (Generating Int)) , go ("Operating Int", Proxy :: Proxy (Operating Int)) , go ("First Int", Proxy :: Proxy (First Int)) , go ("Second Int", Proxy :: Proxy (Second Int)) , go ("Topic", Proxy :: Proxy Topic) ] ] ] ] where simpleTests :: TestTree simpleTests = testGroup "Simple Tests" [ testCase "Unit over id" (simpleTest unitSuite) , testCase "Int over various" (simpleTest intSuite) , testCase "Double over various" (simpleTest doubleSuite) , testCase "List over various" (simpleTest listSuite) ] where unitSuite :: SymbioteT (SimpleSerialization () ()) IO () unitSuite = register "Unit" 100 (Proxy :: Proxy ()) intSuite :: SymbioteT (SimpleSerialization Int Bool) IO () intSuite = register "Int" 100 (Proxy :: Proxy Int) doubleSuite :: SymbioteT (SimpleSerialization Double Bool) IO () doubleSuite = register "Double" 100 (Proxy :: Proxy Double) listSuite :: SymbioteT (SimpleSerialization [Int] (Either Bool [Int])) IO () listSuite = register "List" 100 (Proxy :: Proxy [Int]) bytestringTests :: TestTree bytestringTests = testGroup "ByteString Tests" [ testCase "Json over id" (simpleTest jsonSuite) , testCase "Int over various" (simpleTest intSuite) , testCase "Double over various" (simpleTest doubleSuite) , testCase "List over various" (simpleTest listSuite) ] where jsonSuite :: SymbioteT LBS.ByteString IO () jsonSuite = register "Json" 100 (Proxy :: Proxy Json.Value) intSuite :: SymbioteT BS.ByteString IO () intSuite = register "Int" 100 (Proxy :: Proxy Int) doubleSuite :: SymbioteT BS.ByteString IO () doubleSuite = register "Double" 100 (Proxy :: Proxy Double) listSuite :: SymbioteT BS.ByteString IO () listSuite = register "List" 100 (Proxy :: Proxy [Int]) jsonTests :: TestTree jsonTests = testGroup "Json Tests" [ testCase "Int over various" (simpleTest intSuite) , testCase "Double over various" (simpleTest doubleSuite) , testCase "List over various" (simpleTest listSuite) ] where intSuite :: SymbioteT Json.Value IO () intSuite = register "Int" 100 (Proxy :: Proxy Int) doubleSuite :: SymbioteT Json.Value IO () doubleSuite = register "Double" 100 (Proxy :: Proxy Double) listSuite :: SymbioteT Json.Value IO () listSuite = register "List" 100 (Proxy :: Proxy [Int]) instance SymbioteOperation () () where data Operation () = UnitId perform UnitId () = () deriving instance Show (Operation ()) deriving instance Generic (Operation ()) instance Arbitrary (Operation ()) where arbitrary = pure UnitId instance SymbioteOperation Int Bool where data Operation Int = IntCommutativeRing (Operation (AbidesCommutativeRing Int)) perform op x = case op of IntCommutativeRing op' -> perform op' (AbidesCommutativeRing x) deriving instance Show (Operation Int) deriving instance Generic (Operation Int) instance Cereal.Serialize (Operation Int) instance Json.ToJSON (Operation Int) instance Json.FromJSON (Operation Int) instance Arbitrary (Operation Int) where arbitrary = IntCommutativeRing <$> arbitrary instance SymbioteOperation Double Bool where data Operation Double = DoubleField (Operation (AbidesField Double)) perform op x = case op of DoubleField op' -> perform op' (AbidesField x) deriving instance Show (Operation Double) deriving instance Generic (Operation Double) instance Cereal.Serialize (Operation Double) instance Json.ToJSON (Operation Double) instance Json.FromJSON (Operation Double) instance Arbitrary (Operation Double) where arbitrary = DoubleField <$> arbitrary instance Eq a => SymbioteOperation [a] (Either Bool [a]) where data Operation [a] = ListMonoid (Operation (AbidesMonoid [a])) | ReverseList | InitList | TailList perform op x = case op of ListMonoid op' -> Left (perform op' (AbidesMonoid x)) ReverseList -> Right (reverse x) InitList -> Right $ if null x then [] else init x TailList -> Right $ if null x then [] else tail x deriving instance Show a => Show (Operation [a]) deriving instance Generic (Operation [a]) instance Cereal.Serialize a => Cereal.Serialize (Operation [a]) instance Json.ToJSON a => Json.ToJSON (Operation [a]) instance Json.FromJSON a => Json.FromJSON (Operation [a]) instance Arbitrary a => Arbitrary (Operation [a]) where arbitrary = oneof [ pure ReverseList , pure InitList , pure TailList , ListMonoid <$> arbitrary ] instance SymbioteOperation Json.Value Json.Value where data Operation Json.Value = JsonId perform _ x = x deriving instance Show (Operation Json.Value) deriving instance Generic (Operation Json.Value) instance Arbitrary (Operation Json.Value) where arbitrary = pure JsonId instance Symbiote Json.Value Json.Value LBS.ByteString where encode = Json.encode decode = Json.decode encodeOut _ = Json.encode decodeOut _ = Json.decode encodeOp _ = "id" decodeOp x | x == "id" = Just JsonId | otherwise = Nothing instance Arbitrary Json.Value where arbitrary = do s <- getSize if s <= 1 then oneof [ pure Json.Null , Json.Bool <$> arbitrary , Json.Number <$> arbitrary ] else oneof [ Json.String <$> scale (`div` 2) arbitrary , Json.Array <$> scale (`div` 2) arbitrary , Json.Object <$> scale (`div` 2) arbitrary ] jsonIso :: Json.ToJSON a => Json.FromJSON a => Eq a => Proxy a -> a -> Bool jsonIso Proxy x = Json.decode (Json.encode x) == Just x cerealIso :: Cereal.Serialize a => Eq a => Proxy a -> a -> Bool cerealIso Proxy x = Cereal.decode (Cereal.encode x) == Right x