{-# OPTIONS_GHC -O #-} {-# LANGUAGE NoMonomorphismRestriction #-} import Prelude as P import QCUtils import Test.Framework (defaultMain, testGroup) import Test.Framework.Providers.QuickCheck2 (testProperty) import Test.QuickCheck import Data.Iteratee hiding (head, break) import qualified Data.Iteratee.Char as IC import qualified Data.Iteratee as Iter import Data.Functor.Identity import Data.Monoid import qualified Data.ListLike as LL import Text.Printf (printf) import System.Environment (getArgs) instance Show (a -> b) where show _ = "<>" -- --------------------------------------------- -- Stream instances type ST = Stream [Int] prop_eq str = str == str where types = str :: ST prop_mempty = mempty == (Chunk [] :: Stream [Int]) prop_mappend str1 str2 | isChunk str1 && isChunk str2 = str1 `mappend` str2 == Chunk (chunkData str1 ++ chunkData str2) prop_mappend str1 str2 = isEOF $ str1 `mappend` str2 where types = (str1 :: ST, str2 :: ST) prop_mappend2 str = str `mappend` mempty == mempty `mappend` str where types = str :: ST isChunk (Chunk _) = True isChunk (EOF _) = False chunkData (Chunk xs) = xs isEOF (EOF _) = True isEOF (Chunk _) = False -- --------------------------------------------- -- Iteratee instances runner0 = runIdentity . Iter.run runner1 = runIdentity . Iter.run . runIdentity prop_iterFmap xs f a = runner1 (enumPure1Chunk xs (fmap f $ return a)) == runner1 (enumPure1Chunk xs (return $ f a)) where types = (xs :: [Int], f :: Int -> Int, a :: Int) prop_iterFmap2 xs f i = runner1 (enumPure1Chunk xs (fmap f i)) == f (runner1 (enumPure1Chunk xs i)) where types = (xs :: [Int], i :: I, f :: [Int] -> [Int]) prop_iterMonad1 xs a f = runner1 (enumPureNChunk xs 1 (return a >>= f)) == runner1 (enumPure1Chunk xs (f a)) where types = (xs :: [Int], a :: Int, f :: Int -> I) prop_iterMonad2 m xs = runner1 (enumPureNChunk xs 1 (m >>= return)) == runner1 (enumPure1Chunk xs m) where types = (xs :: [Int], m :: I) prop_iterMonad3 m f g xs = runner1 (enumPureNChunk xs 1 ((m >>= f) >>= g)) == runner1 (enumPure1Chunk xs (m >>= (\x -> f x >>= g))) where types = (xs :: [Int], m :: I, f :: [Int] -> I, g :: [Int] -> I) -- --------------------------------------------- -- List <-> Stream prop_list xs = runner1 (enumPure1Chunk xs stream2list) == xs where types = xs :: [Int] prop_clist xs n = n > 0 ==> runner1 (enumPureNChunk xs n stream2list) == xs where types = xs :: [Int] prop_break f xs = runner1 (enumPure1Chunk xs (Iter.break f)) == fst (break f xs) where types = xs :: [Int] prop_break2 f xs = runner1 (enumPure1Chunk xs (Iter.break f >> stream2list)) == snd (break f xs) where types = xs :: [Int] prop_head xs = P.length xs > 0 ==> runner1 (enumPure1Chunk xs Iter.head) == head xs where types = xs :: [Int] prop_head2 xs = P.length xs > 0 ==> runner1 (enumPure1Chunk xs (Iter.head >> stream2list)) == tail xs where types = xs :: [Int] prop_heads xs = runner1 (enumPure1Chunk xs $ heads xs) == P.length xs where types = xs :: [Int] prop_heads2 xs = runner1 (enumPure1Chunk xs $ heads [] >>= \c -> stream2list >>= \s -> return (c,s)) == (0, xs) where types = xs :: [Int] prop_peek xs = runner1 (enumPure1Chunk xs peek) == sHead xs where types = xs :: [Int] sHead [] = Nothing sHead (x:_) = Just x prop_peek2 xs = runner1 (enumPure1Chunk xs (peek >> stream2list)) == xs where types = xs :: [Int] prop_skip xs = runner1 (enumPure1Chunk xs (skipToEof >> stream2list)) == [] where types = xs :: [Int] -- --------------------------------------------- -- Simple enumerator tests type I = Iteratee [Int] Identity [Int] prop_enumChunks n xs i = n > 0 ==> runner1 (enumPure1Chunk xs i) == runner1 (enumPureNChunk xs n i) where types = (n :: Int, xs :: [Int], i :: I) prop_app1 xs ys i = runner1 (enumPure1Chunk ys (joinIM $ enumPure1Chunk xs i)) == runner1 (enumPure1Chunk (xs ++ ys) i) where types = (xs :: [Int], ys :: [Int], i :: I) prop_app2 xs ys = runner1 ((enumPure1Chunk xs >>> enumPure1Chunk ys) stream2list) == runner1 (enumPure1Chunk (xs ++ ys) stream2list) where types = (xs :: [Int], ys :: [Int]) prop_app3 xs ys i = runner1 ((enumPure1Chunk xs >>> enumPure1Chunk ys) i) == runner1 (enumPure1Chunk (xs ++ ys) i) where types = (xs :: [Int], ys :: [Int], i :: I) prop_eof xs ys i = runner1 (enumPure1Chunk ys $ runIdentity $ (enumPure1Chunk xs >>> enumEof) i) == runner1 (enumPure1Chunk xs i) where types = (xs :: [Int], ys :: [Int], i :: I) prop_isFinished = runner1 (enumEof (isFinished :: Iteratee [Int] Identity Bool)) == True prop_isFinished2 = runner1 (enumErr (iterStrExc "Error") (isFinished :: Iteratee [Int] Identity Bool)) == True prop_null xs i = runner1 (enumPure1Chunk xs =<< enumPure1Chunk [] i) == runner1 (enumPure1Chunk xs i) where types = (xs :: [Int], i :: I) prop_nullH xs = P.length xs > 0 ==> runner1 (enumPure1Chunk xs =<< enumPure1Chunk [] Iter.head) == runner1 (enumPure1Chunk xs Iter.head) where types = xs :: [Int] -- --------------------------------------------- -- Nested Iteratees -- take, mapStream, convStream, and takeR runner2 = runIdentity . run . runner1 prop_mapStream xs i = runner2 (enumPure1Chunk xs $ mapStream id i) == runner1 (enumPure1Chunk xs i) where types = (i :: I, xs :: [Int]) prop_mapStream2 xs n i = n > 0 ==> runner2 (enumPureNChunk xs n $ mapStream id i) == runner1 (enumPure1Chunk xs i) where types = (i :: I, xs :: [Int]) prop_mapjoin xs i = runIdentity (run (joinI . runIdentity $ enumPure1Chunk xs $ mapStream id i)) == runner1 (enumPure1Chunk xs i) where types = (i :: I, xs :: [Int]) convId :: (LL.ListLike s el, Monad m) => Iteratee s m s convId = liftI (\str -> case str of s@(Chunk xs) | LL.null xs -> convId s@(Chunk xs) -> idone xs (Chunk mempty) s@(EOF e) -> idone mempty (EOF e) ) prop_convId xs = runner1 (enumPure1Chunk xs convId) == xs where types = xs :: [Int] prop_convstream xs i = P.length xs > 0 ==> runner2 (enumPure1Chunk xs $ convStream convId i) == runner1 (enumPure1Chunk xs i) where types = (xs :: [Int], i :: I) prop_convstream2 xs = P.length xs > 0 ==> runner2 (enumPure1Chunk xs $ convStream convId Iter.head) == runner1 (enumPure1Chunk xs Iter.head) where types = xs :: [Int] prop_convstream3 xs = P.length xs > 0 ==> runner2 (enumPure1Chunk xs $ convStream convId stream2list) == runner1 (enumPure1Chunk xs stream2list) where types = xs :: [Int] prop_take xs n = n >= 0 ==> runner2 (enumPure1Chunk xs $ Iter.take n stream2list) == runner1 (enumPure1Chunk (P.take n xs) stream2list) where types = xs :: [Int] prop_take2 xs n = n > 0 ==> runner2 (enumPure1Chunk xs $ Iter.take n peek) == runner1 (enumPure1Chunk (P.take n xs) peek) where types = xs :: [Int] prop_takeUpTo xs n = n >= 0 ==> runner2 (enumPure1Chunk xs $ Iter.take n stream2list) == runner2 (enumPure1Chunk xs $ takeUpTo n stream2list) where types = xs :: [Int] -- --------------------------------------------- -- Data.Iteratee.Char {- -- this isn't true, since lines "\r" returns ["\r"], and IC.line should -- return Right "". Not sure what a real test would be... prop_line xs = P.length xs > 0 ==> fromEither (runner1 (enumPure1Chunk xs $ IC.line)) == head (lines xs) where types = xs :: [Char] fromEither (Left l) = l fromEither (Right l) = l -} -- --------------------------------------------- tests = [ testGroup "Elementary" [ testProperty "list" prop_list ,testProperty "chunkList" prop_clist] ,testGroup "Stream tests" [ testProperty "mempty" prop_mempty ,testProperty "mappend" prop_mappend ,testProperty "mappend associates" prop_mappend2 ,testProperty "eq" prop_eq ] ,testGroup "Simple Iteratees" [ testProperty "break" prop_break ,testProperty "break remaineder" prop_break2 ,testProperty "head" prop_head ,testProperty "head remainder" prop_head2 ,testProperty "heads" prop_heads ,testProperty "null heads" prop_heads2 ,testProperty "peek" prop_peek ,testProperty "peek2" prop_peek2 ,testProperty "skipToEof" prop_skip ,testProperty "iteratee Functor 1" prop_iterFmap ,testProperty "iteratee Functor 2" prop_iterFmap2 ,testProperty "iteratee Monad LI" prop_iterMonad1 ,testProperty "iteratee Monad RI" prop_iterMonad2 ,testProperty "iteratee Monad Assc" prop_iterMonad3 ] ,testGroup "Simple Enumerators/Combinators" [ testProperty "enumPureNChunk" prop_enumChunks ,testProperty "enum append 1" prop_app1 ,testProperty "enum sequencing" prop_app2 ,testProperty "enum sequencing 2" prop_app3 ,testProperty "enumEof" prop_eof ,testProperty "isFinished" prop_isFinished ,testProperty "isFinished error" prop_isFinished2 ,testProperty "null data idempotence" prop_null ,testProperty "null data head idempotence" prop_nullH ] ,testGroup "Nested iteratees" [ testProperty "mapStream identity" prop_mapStream ,testProperty "mapStream identity 2" prop_mapStream2 ,testProperty "mapStream identity joinI" prop_mapjoin ,testProperty "take" prop_take ,testProperty "take (finished iteratee)" prop_take2 ,testProperty "takeUpTo" prop_takeUpTo ,testProperty "convStream EOF" prop_convstream2 ,testProperty "convStream identity" prop_convstream ,testProperty "convStream identity 2" prop_convstream3 ] ,testGroup "Data.Iteratee.Char" [ --testProperty "line" prop_line ] ] ------------------------------------------------------------------------ -- The entry point main = defaultMain tests