1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166
|
{-# LANGUAGE TypeApplications #-}
{-|
Module : HsLua.ClassesTests
Copyright : © 2007–2012 Gracjan Polak;
© 2012–2016 Ömer Sinan Ağacan;
© 2017-2024 Albert Krewinkel
License : MIT
Maintainer : Albert Krewinkel <tarleb@hslua.org>
Stability : beta
Portability : non-portable (depends on GHC)
Test that conversions from and to the Lua stack are isomorphisms.
-}
module HsLua.ClassesTests (tests) where
import Control.Monad (forM, forM_)
import Data.ByteString (ByteString)
import Data.Map (Map)
import Data.Set (Set)
import HsLua.Class.Peekable
import HsLua.Class.Pushable
import HsLua.Core as Lua
import Lua.Arbitrary ()
import Test.QuickCheck
import Test.QuickCheck.Instances ()
import Test.QuickCheck.Monadic as QCMonadic
import Test.Tasty (TestTree, testGroup)
import Test.Tasty.QuickCheck (testProperty)
import qualified Data.Text as T
-- | Specifications for Attributes parsing functions.
tests :: TestTree
tests = testGroup "peek and push are well behaved"
[ testGroup "Peek can act as left inverse of push"
[ testProperty "round-tripping unit"
(prop_roundtripEqual :: () -> Property)
, testProperty "booleans remain equal under push/peek"
(prop_roundtripEqual :: Bool -> Property)
, testProperty "lua numbers (i.e., doubles) remain equal under push/peek"
(prop_roundtripEqual :: Lua.Number -> Property)
, testProperty "Lua integers remain equal under push/peek"
(prop_roundtripEqual :: Lua.Integer -> Property)
, testProperty "bytestring remain equal under push/peek"
(prop_roundtripEqual :: ByteString -> Property)
, testProperty "Prelude.Integer"
(prop_roundtripEqual :: Prelude.Integer -> Property)
, testProperty "Float"
(prop_roundtripEqual :: Float -> Property)
, testProperty "Double"
(prop_roundtripEqual :: Double -> Property)
, testProperty "round-tripping strings"
(prop_roundtripEqual :: String -> Property)
, testProperty "lists of boolean remain equal under push/peeks"
(prop_roundtripEqual :: [Bool] -> Property)
, testProperty "lists of lua integers remain equal under push/peek"
(prop_roundtripEqual :: [Lua.Integer] -> Property)
, testProperty "lists of bytestrings remain equal under push/peek"
(prop_roundtripEqual :: [ByteString] -> Property)
, testProperty "text"
(prop_roundtripEqual :: T.Text -> Property)
, testProperty "map of strings to Lua.Number"
(prop_roundtripEqual :: Map String Lua.Number -> Property)
, testProperty "set of strings"
(prop_roundtripEqual :: Set Lua.Number -> Property)
, testGroup "tuples"
[ testProperty "pair of Lua.Numbers"
(prop_roundtripEqual :: (Lua.Number, Lua.Number) -> Property)
, testProperty "triple of Lua.Numbers"
(prop_roundtripEqual :: (Lua.Number, Lua.Number, Lua.Number) -> Property)
, testProperty "quadruple of Lua.Numbers"
(prop_roundtripEqual
:: (Lua.Number, Lua.Number, Lua.Number, Lua.Number) -> Property)
, testProperty "quintuple of Lua.Numbers"
(prop_roundtripEqual
:: (Lua.Number, Lua.Number, Lua.Number, Lua.Number, Lua.Number) -> Property)
, testProperty "hextuple of Text, Lua.Numbers and Booleans"
(prop_roundtripEqual
:: (Bool, Lua.Number, T.Text, Bool, Lua.Number, Lua.Number) -> Property)
, testProperty "septuple of Text, Lua.Number and Booleans"
(prop_roundtripEqual
:: (T.Text, Bool, Lua.Number, Bool, Bool, Lua.Number, Bool) -> Property)
, testProperty "octuple of Strings and Booleans"
(prop_roundtripEqual
:: (Bool, String, Bool, Bool, String, Bool, Bool, String) -> Property)
]
]
, testGroup "Random stack values"
[ testProperty "can push/pop booleans"
(prop_stackPushingPulling :: Bool -> Property)
, testProperty "can push/pop lua integers"
(prop_stackPushingPulling :: Lua.Integer -> Property)
, testProperty "can push/pop lua numbers"
(prop_stackPushingPulling :: Lua.Number -> Property)
, testProperty "can push/pop bytestrings"
(prop_stackPushingPulling :: ByteString -> Property)
, testProperty "can push/pop lists of booleans"
(prop_stackPushingPulling :: [Bool] -> Property)
, testProperty "can push/pop lists of Lua.Integers"
(prop_stackPushingPulling :: [Lua.Integer] -> Property)
, testProperty "can push/pop lists of bytestrings"
(prop_stackPushingPulling :: [ByteString] -> Property)
, testProperty "can push/pop set of bytestrings"
(prop_stackPushingPulling :: Set ByteString -> Property)
]
]
prop_roundtripEqual :: (Eq a, Peekable a, Pushable a) => a -> Property
prop_roundtripEqual x = monadicIO $ do
y <- QCMonadic.run $ roundtrip x
assert (x == y)
roundtrip :: (Peekable a, Pushable a) => a -> IO a
roundtrip x = Lua.run @Lua.Exception $ do
push x
peek (-1)
-- | More involved check that the Peekable and Pushable instances of a
-- datatype work
prop_stackPushingPulling :: (Eq t, Pushable t, Peekable t) => t -> Property
prop_stackPushingPulling t = monadicIO $ do
-- Init Lua state
l <- QCMonadic.run newstate
-- Get an ascending list of small (1-100) positive integers
-- These are the indices at which we will push the value to be tested
-- Note that duplicate values don't matter so we don't need to guard against that
Ordered indices' <- pick arbitrary
let indices = map getPositive indices'
let nItems = (if null indices then 0 else last indices) :: Lua.Integer
-- Make sure there's enough room in the stack
assert =<< QCMonadic.run (runWith l $ checkstack (2 * fromIntegral nItems))
-- Push elements
QCMonadic.run $ forM_ [1..nItems] $ \n ->
runWith @Lua.Exception l $
if n `elem` indices
then push t
else push n
-- Check that the stack size is the same as the total number of pushed items
stackSize <- QCMonadic.run $ runWith l gettop
assert $ fromStackIndex stackSize == fromIntegral nItems
-- Peek all items
vals <- QCMonadic.run $ forM indices $
runWith @Lua.Exception l . peek . StackIndex . fromIntegral
-- Check that the stack size did not change after peeking
newStackSize <- QCMonadic.run $ runWith l gettop
assert $ stackSize == newStackSize
-- Check that we were able to peek at all pushed elements
forM_ vals $ assert . (== t)
-- Cleanup
QCMonadic.run (close l)
|