{-# LANGUAGE OverloadedStrings #-}
module Foundation.Check.Config
( Config(..)
, Seed
, DisplayOption(..)
, defaultConfig
, parseArgs
, configHelp
) where
import Basement.Imports
import Basement.IntegralConv
import Foundation.String.Read
import Foundation.Check.Gen
type Seed = Word64
data DisplayOption =
DisplayTerminalErrorOnly
| DisplayGroupOnly
| DisplayTerminalVerbose
deriving (DisplayOption -> DisplayOption -> Bool
(DisplayOption -> DisplayOption -> Bool)
-> (DisplayOption -> DisplayOption -> Bool) -> Eq DisplayOption
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: DisplayOption -> DisplayOption -> Bool
== :: DisplayOption -> DisplayOption -> Bool
$c/= :: DisplayOption -> DisplayOption -> Bool
/= :: DisplayOption -> DisplayOption -> Bool
Eq, Eq DisplayOption
Eq DisplayOption =>
(DisplayOption -> DisplayOption -> Ordering)
-> (DisplayOption -> DisplayOption -> Bool)
-> (DisplayOption -> DisplayOption -> Bool)
-> (DisplayOption -> DisplayOption -> Bool)
-> (DisplayOption -> DisplayOption -> Bool)
-> (DisplayOption -> DisplayOption -> DisplayOption)
-> (DisplayOption -> DisplayOption -> DisplayOption)
-> Ord DisplayOption
DisplayOption -> DisplayOption -> Bool
DisplayOption -> DisplayOption -> Ordering
DisplayOption -> DisplayOption -> DisplayOption
forall a.
Eq a =>
(a -> a -> Ordering)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> a)
-> (a -> a -> a)
-> Ord a
$ccompare :: DisplayOption -> DisplayOption -> Ordering
compare :: DisplayOption -> DisplayOption -> Ordering
$c< :: DisplayOption -> DisplayOption -> Bool
< :: DisplayOption -> DisplayOption -> Bool
$c<= :: DisplayOption -> DisplayOption -> Bool
<= :: DisplayOption -> DisplayOption -> Bool
$c> :: DisplayOption -> DisplayOption -> Bool
> :: DisplayOption -> DisplayOption -> Bool
$c>= :: DisplayOption -> DisplayOption -> Bool
>= :: DisplayOption -> DisplayOption -> Bool
$cmax :: DisplayOption -> DisplayOption -> DisplayOption
max :: DisplayOption -> DisplayOption -> DisplayOption
$cmin :: DisplayOption -> DisplayOption -> DisplayOption
min :: DisplayOption -> DisplayOption -> DisplayOption
Ord, Int -> DisplayOption
DisplayOption -> Int
DisplayOption -> [DisplayOption]
DisplayOption -> DisplayOption
DisplayOption -> DisplayOption -> [DisplayOption]
DisplayOption -> DisplayOption -> DisplayOption -> [DisplayOption]
(DisplayOption -> DisplayOption)
-> (DisplayOption -> DisplayOption)
-> (Int -> DisplayOption)
-> (DisplayOption -> Int)
-> (DisplayOption -> [DisplayOption])
-> (DisplayOption -> DisplayOption -> [DisplayOption])
-> (DisplayOption -> DisplayOption -> [DisplayOption])
-> (DisplayOption
-> DisplayOption -> DisplayOption -> [DisplayOption])
-> Enum DisplayOption
forall a.
(a -> a)
-> (a -> a)
-> (Int -> a)
-> (a -> Int)
-> (a -> [a])
-> (a -> a -> [a])
-> (a -> a -> [a])
-> (a -> a -> a -> [a])
-> Enum a
$csucc :: DisplayOption -> DisplayOption
succ :: DisplayOption -> DisplayOption
$cpred :: DisplayOption -> DisplayOption
pred :: DisplayOption -> DisplayOption
$ctoEnum :: Int -> DisplayOption
toEnum :: Int -> DisplayOption
$cfromEnum :: DisplayOption -> Int
fromEnum :: DisplayOption -> Int
$cenumFrom :: DisplayOption -> [DisplayOption]
enumFrom :: DisplayOption -> [DisplayOption]
$cenumFromThen :: DisplayOption -> DisplayOption -> [DisplayOption]
enumFromThen :: DisplayOption -> DisplayOption -> [DisplayOption]
$cenumFromTo :: DisplayOption -> DisplayOption -> [DisplayOption]
enumFromTo :: DisplayOption -> DisplayOption -> [DisplayOption]
$cenumFromThenTo :: DisplayOption -> DisplayOption -> DisplayOption -> [DisplayOption]
enumFromThenTo :: DisplayOption -> DisplayOption -> DisplayOption -> [DisplayOption]
Enum, DisplayOption
DisplayOption -> DisplayOption -> Bounded DisplayOption
forall a. a -> a -> Bounded a
$cminBound :: DisplayOption
minBound :: DisplayOption
$cmaxBound :: DisplayOption
maxBound :: DisplayOption
Bounded, Int -> DisplayOption -> ShowS
[DisplayOption] -> ShowS
DisplayOption -> [Char]
(Int -> DisplayOption -> ShowS)
-> (DisplayOption -> [Char])
-> ([DisplayOption] -> ShowS)
-> Show DisplayOption
forall a.
(Int -> a -> ShowS) -> (a -> [Char]) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> DisplayOption -> ShowS
showsPrec :: Int -> DisplayOption -> ShowS
$cshow :: DisplayOption -> [Char]
show :: DisplayOption -> [Char]
$cshowList :: [DisplayOption] -> ShowS
showList :: [DisplayOption] -> ShowS
Show)
data Config = Config
{ Config -> Maybe Seed
udfSeed :: Maybe Seed
, Config -> GenParams
getGenParams :: !GenParams
, Config -> Seed
numTests :: !Word64
, Config -> Bool
listTests :: Bool
, Config -> [String]
testNameMatch :: [String]
, Config -> DisplayOption
displayOptions :: !DisplayOption
, Config -> Bool
helpRequested :: Bool
}
defaultConfig :: Config
defaultConfig :: Config
defaultConfig = Config
{ udfSeed :: Maybe Seed
udfSeed = Maybe Seed
forall a. Maybe a
Nothing
, getGenParams :: GenParams
getGenParams = GenParams
params
, numTests :: Seed
numTests = Seed
100
, listTests :: Bool
listTests = Bool
False
, testNameMatch :: [String]
testNameMatch = []
, displayOptions :: DisplayOption
displayOptions = DisplayOption
DisplayGroupOnly
, helpRequested :: Bool
helpRequested = Bool
False
}
where
params :: GenParams
params = GenParams
{ genMaxSizeIntegral :: Word
genMaxSizeIntegral = Word
32
, genMaxSizeArray :: Word
genMaxSizeArray = Word
512
, genMaxSizeString :: Word
genMaxSizeString = Word
8192
}
type ParamError = String
getInteger :: String -> String -> Either ParamError Integer
getInteger :: String -> String -> Either String Integer
getInteger String
optionName String
s =
Either String Integer
-> (Integer -> Either String Integer)
-> Maybe Integer
-> Either String Integer
forall b a. b -> (a -> b) -> Maybe a -> b
maybe (String -> Either String Integer
forall a b. a -> Either a b
Left String
errMsg) Integer -> Either String Integer
forall a b. b -> Either a b
Right (Maybe Integer -> Either String Integer)
-> Maybe Integer -> Either String Integer
forall a b. (a -> b) -> a -> b
$ String -> Maybe Integer
forall i.
(HasNegation i, IntegralUpsize Word8 i, Additive i,
Multiplicative i, IsIntegral i) =>
String -> Maybe i
readIntegral String
s
where
errMsg :: String
errMsg = String
"argument error for " String -> String -> String
forall a. Semigroup a => a -> a -> a
<> String
optionName String -> String -> String
forall a. Semigroup a => a -> a -> a
<> String
" expecting a number but got : " String -> String -> String
forall a. Semigroup a => a -> a -> a
<> String
s
parseArgs :: [String] -> Config -> Either ParamError Config
parseArgs :: [String] -> Config -> Either String Config
parseArgs [] Config
cfg = Config -> Either String Config
forall a b. b -> Either a b
Right Config
cfg
parseArgs [String
"--seed"] Config
_ = String -> Either String Config
forall a b. a -> Either a b
Left String
"option `--seed' is missing a parameter"
parseArgs (String
"--seed":String
x:[String]
xs) Config
cfg = String -> String -> Either String Integer
getInteger String
"seed" String
x Either String Integer
-> (Integer -> Either String Config) -> Either String Config
forall a b.
Either String a -> (a -> Either String b) -> Either String b
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \Integer
i -> [String] -> Config -> Either String Config
parseArgs [String]
xs (Config -> Either String Config) -> Config -> Either String Config
forall a b. (a -> b) -> a -> b
$ Config
cfg { udfSeed = Just $ integralDownsize i }
parseArgs [String
"--tests"] Config
_ = String -> Either String Config
forall a b. a -> Either a b
Left String
"option `--tests' is missing a parameter"
parseArgs (String
"--tests":String
x:[String]
xs) Config
cfg = String -> String -> Either String Integer
getInteger String
"tests" String
x Either String Integer
-> (Integer -> Either String Config) -> Either String Config
forall a b.
Either String a -> (a -> Either String b) -> Either String b
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \Integer
i -> [String] -> Config -> Either String Config
parseArgs [String]
xs (Config -> Either String Config) -> Config -> Either String Config
forall a b. (a -> b) -> a -> b
$ Config
cfg { numTests = integralDownsize i }
parseArgs (String
"--quiet":[String]
xs) Config
cfg = [String] -> Config -> Either String Config
parseArgs [String]
xs (Config -> Either String Config) -> Config -> Either String Config
forall a b. (a -> b) -> a -> b
$ Config
cfg { displayOptions = DisplayTerminalErrorOnly }
parseArgs (String
"--list-tests":[String]
xs) Config
cfg = [String] -> Config -> Either String Config
parseArgs [String]
xs (Config -> Either String Config) -> Config -> Either String Config
forall a b. (a -> b) -> a -> b
$ Config
cfg { listTests = True }
parseArgs (String
"--verbose":[String]
xs) Config
cfg = [String] -> Config -> Either String Config
parseArgs [String]
xs (Config -> Either String Config) -> Config -> Either String Config
forall a b. (a -> b) -> a -> b
$ Config
cfg { displayOptions = DisplayTerminalVerbose }
parseArgs (String
"--help":[String]
xs) Config
cfg = [String] -> Config -> Either String Config
parseArgs [String]
xs (Config -> Either String Config) -> Config -> Either String Config
forall a b. (a -> b) -> a -> b
$ Config
cfg { helpRequested = True }
parseArgs (String
x:[String]
xs) Config
cfg = [String] -> Config -> Either String Config
parseArgs [String]
xs (Config -> Either String Config) -> Config -> Either String Config
forall a b. (a -> b) -> a -> b
$ Config
cfg { testNameMatch = x : testNameMatch cfg }
configHelp :: [String]
configHelp :: [String]
configHelp =
[ String
"Usage: <program-name> [options] [test-name-match]\n"
, String
"\n"
, String
"Known options:\n"
, String
"\n"
, String
" --seed <seed>: a 64bit positive number to use as seed to generate arbitrary value.\n"
, String
" --tests <tests>: the number of tests to perform for every property tests.\n"
, String
" --quiet: print only the errors to the standard output\n"
, String
" --verbose: print every property tests to the stand output.\n"
, String
" --list-tests: print all test names.\n"
]