{-# LANGUAGE CPP #-}
{-# LANGUAGE ExistentialQuantification #-}
{-# LANGUAGE Rank2Types #-}
module Main where
import Imports
import Utils
import Data.Char (chr)
import Data.Word
import Data.ByteArray (Bytes, ScrubbedBytes, ByteArray)
import qualified Data.ByteArray as B
import qualified Data.ByteArray.Encoding as B
import qualified Data.ByteArray.Parse as Parse
import qualified SipHash
#ifdef WITH_FOUNDATION_SUPPORT
import qualified Foundation as F
#endif
data Backend = BackendByte | BackendScrubbedBytes
deriving (Show,Eq,Bounded,Enum)
allBackends :: [Backend]
allBackends = enumFrom BackendByte
data ArbitraryBS = forall a . ByteArray a => ArbitraryBS a
arbitraryBS :: Int -> Gen ArbitraryBS
arbitraryBS n = do
backend <- elements allBackends
case backend of
BackendByte -> ArbitraryBS `fmap` ((B.pack `fmap` replicateM n arbitrary) :: Gen Bytes)
BackendScrubbedBytes -> ArbitraryBS `fmap` ((B.pack `fmap` replicateM n arbitrary) :: Gen ScrubbedBytes)
arbitraryBSof :: Int -> Int -> Gen ArbitraryBS
arbitraryBSof minBytes maxBytes = choose (minBytes, maxBytes) >>= arbitraryBS
newtype SmallList a = SmallList [a]
deriving (Show,Eq)
instance Arbitrary a => Arbitrary (SmallList a) where
arbitrary = choose (0,8) >>= \n -> SmallList `fmap` replicateM n arbitrary
instance Arbitrary ArbitraryBS where
arbitrary = arbitraryBSof 0 259
newtype Words8 = Words8 { unWords8 :: [Word8] }
deriving (Show,Eq)
instance Arbitrary Words8 where
arbitrary = choose (0, 259) >>= \n -> Words8 <$> replicateM n arbitrary
testGroupBackends :: String -> (forall ba . (Show ba, Eq ba, ByteArray ba) => (ba -> ba) -> [TestTree]) -> TestTree
testGroupBackends x l =
testGroup x
[ testGroup "Bytes" (l withBytesWitness)
, testGroup "ScrubbedBytes" (l withScrubbedBytesWitness)
]
testShowProperty :: Testable a
=> String
-> (forall ba . (Show ba, Eq ba, ByteArray ba) => (ba -> ba) -> ([Word8] -> String) -> a)
-> TestTree
testShowProperty x p =
testGroup x
[ testProperty "Bytes" (p withBytesWitness showLikeString)
, testProperty "ScrubbedBytes" (p withScrubbedBytesWitness showLikeEmptySB)
]
where
showLikeString l = show $ map (chr . fromIntegral) l
showLikeEmptySB _ = show (withScrubbedBytesWitness B.empty)
base64Kats =
[ ("pleasure.", "cGxlYXN1cmUu")
, ("leasure.", "bGVhc3VyZS4=")
, ("easure.", "ZWFzdXJlLg==")
, ("asure.", "YXN1cmUu")
, ("sure.", "c3VyZS4=")
]
base64URLKats =
[ ("pleasure.", "cGxlYXN1cmUu")
, ("leasure.", "bGVhc3VyZS4")
, ("easure.", "ZWFzdXJlLg")
, ("asure.", "YXN1cmUu")
, ("sure.", "c3VyZS4")
, ("\DC4\251\156\ETX\217~", "FPucA9l-") -- From RFC4648
, ("\DC4\251\156\ETX\217\DEL", "FPucA9l_")
, ("", "")
]
base16Kats =
[ ("this is a string", "74686973206973206120737472696e67") ]
base32Kats =
[ ("-pleasure.", "FVYGYZLBON2XEZJO")
, ("pleasure.", "OBWGKYLTOVZGKLQ=")
, ("leasure.", "NRSWC43VOJSS4===")
, ("easure.", "MVQXG5LSMUXA====")
, ("asure.", "MFZXK4TFFY======")
, ("sure.", "ON2XEZJO")
, ("ure.", "OVZGKLQ=")
, ("re.", "OJSS4===")
, ("e.", "MUXA====")
, (".", "FY======")
, ("", "")
]
encodingTests witnessID =
[ testGroup "BASE64"
[ testGroup "encode-KAT" encodeKats64
, testGroup "decode-KAT" decodeKats64
]
, testGroup "BASE64URL"
[ testGroup "encode-KAT" encodeKats64URLUnpadded
, testGroup "decode-KAT" decodeKats64URLUnpadded
]
, testGroup "BASE32"
[ testGroup "encode-KAT" encodeKats32
, testGroup "decode-KAT" decodeKats32
]
, testGroup "BASE16"
[ testGroup "encode-KAT" encodeKats16
, testGroup "decode-KAT" decodeKats16
]
]
where
encodeKats64 = map (toTest B.Base64) $ zip [1..] base64Kats
decodeKats64 = map (toBackTest B.Base64) $ zip [1..] base64Kats
encodeKats32 = map (toTest B.Base32) $ zip [1..] base32Kats
decodeKats32 = map (toBackTest B.Base32) $ zip [1..] base32Kats
encodeKats16 = map (toTest B.Base16) $ zip [1..] base16Kats
decodeKats16 = map (toBackTest B.Base16) $ zip [1..] base16Kats
encodeKats64URLUnpadded = map (toTest B.Base64URLUnpadded) $ zip [1..] base64URLKats
decodeKats64URLUnpadded = map (toBackTest B.Base64URLUnpadded) $ zip [1..] base64URLKats
toTest :: B.Base -> (Int, (String, String)) -> TestTree
toTest base (i, (inp, out)) = testCase (show i) $
let inpbs = witnessID $ B.convertToBase base $ witnessID $ B.pack $ unS inp
outbs = witnessID $ B.pack $ unS out
in outbs @=? inpbs
toBackTest :: B.Base -> (Int, (String, String)) -> TestTree
toBackTest base (i, (inp, out)) = testCase (show i) $
let inpbs = witnessID $ B.pack $ unS inp
outbs = B.convertFromBase base $ witnessID $ B.pack $ unS out
in Right inpbs @=? outbs
parsingTests witnessID =
[ testCase "parse" $
let input = witnessID $ B.pack $ unS "xx abctest"
abc = witnessID $ B.pack $ unS "abc"
est = witnessID $ B.pack $ unS "est"
result = Parse.parse ((,,) <$> Parse.take 2 <*> Parse.byte 0x20 <*> (Parse.bytes abc *> Parse.anyByte)) input
in case result of
Parse.ParseOK remaining (_,_,_) -> est @=? remaining
_ -> assertFailure ""
]
main = defaultMain $ testGroup "memory"
[ localOption (QuickCheckTests 5000) $ testGroupBackends "basic" basicProperties
, testGroupBackends "encoding" encodingTests
, testGroupBackends "parsing" parsingTests
, testGroupBackends "hashing" $ \witnessID ->
[ testGroup "SipHash" $ SipHash.tests witnessID
]
, testShowProperty "showing" $ \witnessID expectedShow (Words8 l) ->
(show . witnessID . B.pack $ l) == expectedShow l
#ifdef WITH_FOUNDATION_SUPPORT
, testFoundationTypes
#endif
]
where
basicProperties witnessID =
[ testProperty "unpack . pack == id" $ \(Words8 l) -> l == (B.unpack . witnessID . B.pack $ l)
, testProperty "self-eq" $ \(Words8 l) -> let b = witnessID . B.pack $ l in b == b
, testProperty "add-empty-eq" $ \(Words8 l) ->
let b = witnessID $ B.pack l
in B.append b B.empty == b
, testProperty "zero" $ \(Positive n) ->
let expected = witnessID $ B.pack $ replicate n 0
in expected == B.zero n
, testProperty "Ord" $ \(Words8 l1) (Words8 l2) ->
compare l1 l2 == compare (witnessID $ B.pack l1) (B.pack l2)
, testProperty "Monoid(mappend)" $ \(Words8 l1) (Words8 l2) ->
mappend l1 l2 == (B.unpack $ mappend (witnessID $ B.pack l1) (B.pack l2))
, testProperty "Monoid(mconcat)" $ \(SmallList l) ->
mconcat (map unWords8 l) == (B.unpack $ mconcat $ map (witnessID . B.pack . unWords8) l)
, testProperty "append (append a b) c == append a (append b c)" $ \(Words8 la) (Words8 lb) (Words8 lc) ->
let a = witnessID $ B.pack la
b = witnessID $ B.pack lb
c = witnessID $ B.pack lc
in B.append (B.append a b) c == B.append a (B.append b c)
, testProperty "concat l" $ \(SmallList l) ->
let chunks = map (witnessID . B.pack . unWords8) l
expected = concatMap unWords8 l
in B.pack expected == witnessID (B.concat chunks)
, testProperty "cons b bs == reverse (snoc (reverse bs) b)" $ \(Words8 l) b ->
let b1 = witnessID (B.pack l)
b2 = witnessID (B.pack (reverse l))
expected = B.pack (reverse (B.unpack (B.snoc b2 b)))
in B.cons b b1 == expected
, testProperty "all == Prelude.all" $ \(Words8 l) b ->
let b1 = witnessID (B.pack l)
p = (/= b)
in B.all p b1 == all p l
, testProperty "any == Prelude.any" $ \(Words8 l) b ->
let b1 = witnessID (B.pack l)
p = (== b)
in B.any p b1 == any p l
, testProperty "singleton b == pack [b]" $ \b ->
witnessID (B.singleton b) == B.pack [b]
, testProperty "span" $ \x (Words8 l) ->
let c = witnessID (B.pack l)
(a, b) = B.span (== x) c
in c == B.append a b
, testProperty "span (const True)" $ \(Words8 l) ->
let a = witnessID (B.pack l)
in B.span (const True) a == (a, B.empty)
, testProperty "span (const False)" $ \(Words8 l) ->
let b = witnessID (B.pack l)
in B.span (const False) b == (B.empty, b)
]
#ifdef WITH_FOUNDATION_SUPPORT
testFoundationTypes = testGroup "Foundation"
[ testCase "allocRet 4 _ :: F.UArray Int8 === 4" $ do
x <- (B.length :: F.UArray F.Int8 -> Int) . snd <$> B.allocRet 4 (const $ return ())
assertEqual "" 4 x
, testCase "allocRet 4 _ :: F.UArray Int16 === 4" $ do
x <- (B.length :: F.UArray F.Int16 -> Int) . snd <$> B.allocRet 4 (const $ return ())
assertEqual "" 4 x
, testCase "allocRet 4 _ :: F.UArray Int32 === 4" $ do
x <- (B.length :: F.UArray F.Int32 -> Int) . snd <$> B.allocRet 4 (const $ return ())
assertEqual "" 4 x
, testCase "allocRet 4 _ :: F.UArray Int64 === 8" $ do
x <- (B.length :: F.UArray F.Int64 -> Int) . snd <$> B.allocRet 4 (const $ return ())
assertEqual "" 8 x
]
#endif