{-# 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 -- ^ optional user specified seed
    , Config -> GenParams
getGenParams :: !GenParams
        -- ^ Parameters for the generator
        --
        -- default:
        --   * 32bits long numbers;
        --   * array of 512 elements max;
        --   * string of 8192 bytes max.
        --
    , Config -> Seed
numTests     :: !Word64
        -- ^ the number of tests to perform on every property.
        --
        -- default: 100
    , Config -> Bool
listTests      :: Bool
    , Config -> [String]
testNameMatch  :: [String]
    , Config -> DisplayOption
displayOptions :: !DisplayOption
    , Config -> Bool
helpRequested  :: Bool
    }

-- | create the default configuration
--
-- see @Config@ for details
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   -- 256 bits maximum numbers
        , genMaxSizeArray :: Word
genMaxSizeArray    = Word
512  -- 512 elements
        , genMaxSizeString :: Word
genMaxSizeString   = Word
8192 -- 8K string
        }

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"
    ]