{-# LANGUAGE DerivingVia #-}
{-# LANGUAGE KindSignatures #-}
{-# OPTIONS_GHC -Wno-unrecognised-pragmas #-}
module Cardano.TestUtils (
  
  
  PreProcessor,
  
  mkPreProcessor,
  
  preProcess,
  
  PreCondition,
  
  mkPreCondition,
  
  checkPreConditions,
  
  PostCondition,
  
  mkPostCondition,
  
  checkPostConditions,
  
  PipelinedTestCase (..),
  PipelinedTestErrors (..),
  pipelinedUnitCase,
  
  
  TxFCEKCase,
  TxFCEKInput (..),
  TxFCEKOutput (..),
  mkTxFCEKCase,
  txfCEKUnitCase,
  
  nominalPostCondition,
  nominalCaseBasic,
  
  attackCaseRegexPostCondition,
  attackCaseBasicRegex,
) where
import Control.Arrow (Kleisli (Kleisli), runKleisli, (>>>))
import Data.Foldable (fold, foldl')
import Data.Kind (Type)
import Data.List (intercalate)
import Data.Monoid (First (First), getFirst)
import Data.Text (Text)
import Plutarch.Evaluate (EvalError, applyArguments, evalScriptHuge)
import Plutarch.Internal.Term (Script)
import PlutusLedgerApi.V3 (ExBudget, ScriptContext, toData)
import Test.Tasty (TestTree)
import Test.Tasty.HUnit (assertFailure, testCase)
import Text.RE.TDFA.Text (RE, countMatches, reSource, (*=~))
newtype PreProcessor err input = PreProcessor
  { forall err input.
PreProcessor err input -> Kleisli (Either err) input input
preProcessor ::
      Kleisli
        (Either err)
        input
        input
  }
mkPreProcessor :: (input -> Either err input) -> PreProcessor err input
mkPreProcessor :: forall input err.
(input -> Either err input) -> PreProcessor err input
mkPreProcessor = Kleisli (Either err) input input -> PreProcessor err input
forall err input.
Kleisli (Either err) input input -> PreProcessor err input
PreProcessor (Kleisli (Either err) input input -> PreProcessor err input)
-> ((input -> Either err input)
    -> Kleisli (Either err) input input)
-> (input -> Either err input)
-> PreProcessor err input
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (input -> Either err input) -> Kleisli (Either err) input input
forall (m :: Type -> Type) a b. (a -> m b) -> Kleisli m a b
Kleisli
preProcess ::
  [PreProcessor err input] ->
  input ->
  Either err input
preProcess :: forall err input.
[PreProcessor err input] -> input -> Either err input
preProcess [PreProcessor err input]
pps = Kleisli (Either err) input input -> input -> Either err input
forall (m :: Type -> Type) a b. Kleisli m a b -> a -> m b
runKleisli (Kleisli (Either err) input input -> input -> Either err input)
-> Kleisli (Either err) input input -> input -> Either err input
forall a b. (a -> b) -> a -> b
$ (Kleisli (Either err) input input
 -> Kleisli (Either err) input input
 -> Kleisli (Either err) input input)
-> Kleisli (Either err) input input
-> [Kleisli (Either err) input input]
-> Kleisli (Either err) input input
forall b a. (b -> a -> b) -> b -> [a] -> b
forall (t :: Type -> Type) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl' Kleisli (Either err) input input
-> Kleisli (Either err) input input
-> Kleisli (Either err) input input
forall {k} (cat :: k -> k -> Type) (a :: k) (b :: k) (c :: k).
Category @k cat =>
cat a b -> cat b c -> cat a c
(>>>) ((input -> Either err input) -> Kleisli (Either err) input input
forall (m :: Type -> Type) a b. (a -> m b) -> Kleisli m a b
Kleisli input -> Either err input
forall a. a -> Either err a
forall (f :: Type -> Type) a. Applicative f => a -> f a
pure) ((PreProcessor err input -> Kleisli (Either err) input input)
-> [PreProcessor err input] -> [Kleisli (Either err) input input]
forall a b. (a -> b) -> [a] -> [b]
forall (f :: Type -> Type) a b. Functor f => (a -> b) -> f a -> f b
fmap PreProcessor err input -> Kleisli (Either err) input input
forall err input.
PreProcessor err input -> Kleisli (Either err) input input
preProcessor [PreProcessor err input]
pps)
newtype PreCondition err input = PreCondition
  { forall err input. PreCondition err input -> input -> First err
preCondition :: input -> First err
  }
mkPreCondition :: (input -> Maybe err) -> PreCondition err input
mkPreCondition :: forall input err. (input -> Maybe err) -> PreCondition err input
mkPreCondition input -> Maybe err
f = (input -> First err) -> PreCondition err input
forall err input. (input -> First err) -> PreCondition err input
PreCondition (Maybe err -> First err
forall a. Maybe a -> First a
First (Maybe err -> First err)
-> (input -> Maybe err) -> input -> First err
forall b c a. (b -> c) -> (a -> b) -> a -> c
. input -> Maybe err
f)
checkPreConditions ::
  [PreCondition err input] ->
  input ->
  Maybe err
checkPreConditions :: forall err input. [PreCondition err input] -> input -> Maybe err
checkPreConditions [PreCondition err input]
pcs input
args =
  First err -> Maybe err
forall a. First a -> Maybe a
getFirst ((PreCondition err input -> First err)
-> [PreCondition err input] -> First err
forall m a. Monoid m => (a -> m) -> [a] -> m
forall (t :: Type -> Type) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap ((\input -> First err
x -> input -> First err
x input
args) ((input -> First err) -> First err)
-> (PreCondition err input -> input -> First err)
-> PreCondition err input
-> First err
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PreCondition err input -> input -> First err
forall err input. PreCondition err input -> input -> First err
preCondition) [PreCondition err input]
pcs)
newtype PostCondition err output = PostCondition
  {forall err output. PostCondition err output -> output -> First err
postCondition :: output -> First err}
mkPostCondition :: (output -> Maybe err) -> PostCondition err output
mkPostCondition :: forall output err.
(output -> Maybe err) -> PostCondition err output
mkPostCondition output -> Maybe err
f = (output -> First err) -> PostCondition err output
forall err output.
(output -> First err) -> PostCondition err output
PostCondition (Maybe err -> First err
forall a. Maybe a -> First a
First (Maybe err -> First err)
-> (output -> Maybe err) -> output -> First err
forall b c a. (b -> c) -> (a -> b) -> a -> c
. output -> Maybe err
f)
checkPostConditions ::
  [PostCondition err output] ->
  output ->
  Maybe err
checkPostConditions :: forall err output.
[PostCondition err output] -> output -> Maybe err
checkPostConditions [PostCondition err output]
pcs output
args =
  First err -> Maybe err
forall a. First a -> Maybe a
getFirst ((PostCondition err output -> First err)
-> [PostCondition err output] -> First err
forall m a. Monoid m => (a -> m) -> [a] -> m
forall (t :: Type -> Type) m a.
(Foldable t, Monoid m) =>
(a -> m) -> t a -> m
foldMap ((\output -> First err
x -> output -> First err
x output
args) ((output -> First err) -> First err)
-> (PostCondition err output -> output -> First err)
-> PostCondition err output
-> First err
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PostCondition err output -> output -> First err
forall err output. PostCondition err output -> output -> First err
postCondition) [PostCondition err output]
pcs)
data PipelinedTestCase errPP errPreC errPostC input output = PipelinedTestCase
  { forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output -> String
name :: String
  
  , forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output
-> [PreProcessor errPP input]
preProcessors :: [PreProcessor errPP input]
  
  , forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output
-> [PreCondition errPreC input]
preConditions :: [PreCondition errPreC input]
  
  , forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output -> input
input :: input
  
  , forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output
-> input -> output
computation :: input -> output
  
  , forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output
-> [PostCondition errPostC output]
postConditions :: [PostCondition errPostC output]
  
  }
data PipelinedTestErrors errPP errPreC errPostC
  = PipelinedPreProcessorError errPP
  | PipelinedPreConditionError errPreC
  | PipelinedPostConditionError errPostC
pipelinedUnitCase ::
  (Show errPP, Show errPreC, Show errPostC) =>
  PipelinedTestCase errPP errPreC errPostC input output ->
  TestTree
pipelinedUnitCase :: forall errPP errPreC errPostC input output.
(Show errPP, Show errPreC, Show errPostC) =>
PipelinedTestCase errPP errPreC errPostC input output -> TestTree
pipelinedUnitCase
  PipelinedTestCase
    { $sel:name:PipelinedTestCase :: forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output -> String
name = String
name'
    , $sel:preProcessors:PipelinedTestCase :: forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output
-> [PreProcessor errPP input]
preProcessors = [PreProcessor errPP input]
preProcessors'
    , $sel:preConditions:PipelinedTestCase :: forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output
-> [PreCondition errPreC input]
preConditions = [PreCondition errPreC input]
preConditions'
    , $sel:input:PipelinedTestCase :: forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output -> input
input = input
input'
    , $sel:computation:PipelinedTestCase :: forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output
-> input -> output
computation = input -> output
computation'
    , $sel:postConditions:PipelinedTestCase :: forall errPP errPreC errPostC input output.
PipelinedTestCase errPP errPreC errPostC input output
-> [PostCondition errPostC output]
postConditions = [PostCondition errPostC output]
postConditions'
    } =
    String -> Assertion -> TestTree
testCase String
name' (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ do
      
      case [PreProcessor errPP input] -> input -> Either errPP input
forall err input.
[PreProcessor err input] -> input -> Either err input
preProcess [PreProcessor errPP input]
preProcessors' input
input' of
        Left errPP
err -> String -> Assertion
forall a. HasCallStack => String -> IO a
assertFailure (String -> Assertion) -> String -> Assertion
forall a b. (a -> b) -> a -> b
$ errPP -> String
forall a. Show a => a -> String
show errPP
err
        Right input
preProcessedArgs ->
          
          case [PreCondition errPreC input] -> input -> Maybe errPreC
forall err input. [PreCondition err input] -> input -> Maybe err
checkPreConditions [PreCondition errPreC input]
preConditions' input
preProcessedArgs of
            Just errPreC
err' -> String -> Assertion
forall a. HasCallStack => String -> IO a
assertFailure (String -> Assertion) -> String -> Assertion
forall a b. (a -> b) -> a -> b
$ errPreC -> String
forall a. Show a => a -> String
show errPreC
err'
            Maybe errPreC
Nothing ->
              
              let
                evaluationResult :: output
evaluationResult =
                  input -> output
computation' input
preProcessedArgs
               in
                
                case [PostCondition errPostC output] -> output -> Maybe errPostC
forall err output.
[PostCondition err output] -> output -> Maybe err
checkPostConditions [PostCondition errPostC output]
postConditions' output
evaluationResult of
                  Just errPostC
err'' -> String -> Assertion
forall a. HasCallStack => String -> IO a
assertFailure (String -> Assertion) -> String -> Assertion
forall a b. (a -> b) -> a -> b
$ errPostC -> String
forall a. Show a => a -> String
show errPostC
err''
                  Maybe errPostC
Nothing -> () -> Assertion
forall a. a -> IO a
forall (f :: Type -> Type) a. Applicative f => a -> f a
pure ()
data TxFCEKInput = TxFCEKInput
  { TxFCEKInput -> ScriptContext
cekScriptContext :: ScriptContext
  , TxFCEKInput -> Script
cekScript :: Script
  }
  deriving stock (TxFCEKInput -> TxFCEKInput -> Bool
(TxFCEKInput -> TxFCEKInput -> Bool)
-> (TxFCEKInput -> TxFCEKInput -> Bool) -> Eq TxFCEKInput
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: TxFCEKInput -> TxFCEKInput -> Bool
== :: TxFCEKInput -> TxFCEKInput -> Bool
$c/= :: TxFCEKInput -> TxFCEKInput -> Bool
/= :: TxFCEKInput -> TxFCEKInput -> Bool
Eq, Int -> TxFCEKInput -> ShowS
[TxFCEKInput] -> ShowS
TxFCEKInput -> String
(Int -> TxFCEKInput -> ShowS)
-> (TxFCEKInput -> String)
-> ([TxFCEKInput] -> ShowS)
-> Show TxFCEKInput
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> TxFCEKInput -> ShowS
showsPrec :: Int -> TxFCEKInput -> ShowS
$cshow :: TxFCEKInput -> String
show :: TxFCEKInput -> String
$cshowList :: [TxFCEKInput] -> ShowS
showList :: [TxFCEKInput] -> ShowS
Show)
data TxFCEKOutput = TxFCEKOutput
  { TxFCEKOutput -> Either EvalError Script
cekResult :: Either EvalError Script
  , TxFCEKOutput -> ExBudget
cekExBudget :: ExBudget
  , TxFCEKOutput -> [Text]
cekLogs :: [Text]
  }
  deriving stock (TxFCEKOutput -> TxFCEKOutput -> Bool
(TxFCEKOutput -> TxFCEKOutput -> Bool)
-> (TxFCEKOutput -> TxFCEKOutput -> Bool) -> Eq TxFCEKOutput
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: TxFCEKOutput -> TxFCEKOutput -> Bool
== :: TxFCEKOutput -> TxFCEKOutput -> Bool
$c/= :: TxFCEKOutput -> TxFCEKOutput -> Bool
/= :: TxFCEKOutput -> TxFCEKOutput -> Bool
Eq, Int -> TxFCEKOutput -> ShowS
[TxFCEKOutput] -> ShowS
TxFCEKOutput -> String
(Int -> TxFCEKOutput -> ShowS)
-> (TxFCEKOutput -> String)
-> ([TxFCEKOutput] -> ShowS)
-> Show TxFCEKOutput
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> TxFCEKOutput -> ShowS
showsPrec :: Int -> TxFCEKOutput -> ShowS
$cshow :: TxFCEKOutput -> String
show :: TxFCEKOutput -> String
$cshowList :: [TxFCEKOutput] -> ShowS
showList :: [TxFCEKOutput] -> ShowS
Show)
newtype TxFCEKCase errPP errPreC errPostC
  = TxFCEKCase
      ( PipelinedTestCase
          errPP
          errPreC
          errPostC
          TxFCEKInput
          TxFCEKOutput
      )
mkTxFCEKCase ::
  forall (errPP :: Type) (errPreC :: Type) (errPostC :: Type).
  
  String ->
  
  [PreProcessor errPP TxFCEKInput] ->
  
  [PreCondition errPreC TxFCEKInput] ->
  
  TxFCEKInput ->
  
  [PostCondition errPostC TxFCEKOutput] ->
  TxFCEKCase errPP errPreC errPostC
mkTxFCEKCase :: forall errPP errPreC errPostC.
String
-> [PreProcessor errPP TxFCEKInput]
-> [PreCondition errPreC TxFCEKInput]
-> TxFCEKInput
-> [PostCondition errPostC TxFCEKOutput]
-> TxFCEKCase errPP errPreC errPostC
mkTxFCEKCase String
name' [PreProcessor errPP TxFCEKInput]
pps [PreCondition errPreC TxFCEKInput]
preCs TxFCEKInput
input' [PostCondition errPostC TxFCEKOutput]
postCs =
  PipelinedTestCase errPP errPreC errPostC TxFCEKInput TxFCEKOutput
-> TxFCEKCase errPP errPreC errPostC
forall errPP errPreC errPostC.
PipelinedTestCase errPP errPreC errPostC TxFCEKInput TxFCEKOutput
-> TxFCEKCase errPP errPreC errPostC
TxFCEKCase (PipelinedTestCase errPP errPreC errPostC TxFCEKInput TxFCEKOutput
 -> TxFCEKCase errPP errPreC errPostC)
-> PipelinedTestCase
     errPP errPreC errPostC TxFCEKInput TxFCEKOutput
-> TxFCEKCase errPP errPreC errPostC
forall a b. (a -> b) -> a -> b
$
    PipelinedTestCase
      { $sel:name:PipelinedTestCase :: String
name = String
name'
      , $sel:preProcessors:PipelinedTestCase :: [PreProcessor errPP TxFCEKInput]
preProcessors = [PreProcessor errPP TxFCEKInput]
pps
      , $sel:preConditions:PipelinedTestCase :: [PreCondition errPreC TxFCEKInput]
preConditions = [PreCondition errPreC TxFCEKInput]
preCs
      , $sel:input:PipelinedTestCase :: TxFCEKInput
input = TxFCEKInput
input'
      , $sel:computation:PipelinedTestCase :: TxFCEKInput -> TxFCEKOutput
computation = TxFCEKInput -> TxFCEKOutput
comp
      , $sel:postConditions:PipelinedTestCase :: [PostCondition errPostC TxFCEKOutput]
postConditions = [PostCondition errPostC TxFCEKOutput]
postCs
      }
  where
    comp :: TxFCEKInput -> TxFCEKOutput
    comp :: TxFCEKInput -> TxFCEKOutput
comp (TxFCEKInput ScriptContext
sc Script
script) =
      let
        (Either EvalError Script
res, ExBudget
budget, [Text]
logs) = Script -> (Either EvalError Script, ExBudget, [Text])
evalScriptHuge (Script -> (Either EvalError Script, ExBudget, [Text]))
-> ([Data] -> Script)
-> [Data]
-> (Either EvalError Script, ExBudget, [Text])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Script -> [Data] -> Script
applyArguments Script
script ([Data] -> (Either EvalError Script, ExBudget, [Text]))
-> [Data] -> (Either EvalError Script, ExBudget, [Text])
forall a b. (a -> b) -> a -> b
$ [ScriptContext -> Data
forall a. ToData a => a -> Data
toData ScriptContext
sc]
       in
        Either EvalError Script -> ExBudget -> [Text] -> TxFCEKOutput
TxFCEKOutput Either EvalError Script
res ExBudget
budget [Text]
logs
txfCEKUnitCase ::
  forall (errPP :: Type) (errPreC :: Type) (errPostC :: Type).
  (Show errPP, Show errPreC, Show errPostC) =>
  TxFCEKCase errPP errPreC errPostC ->
  TestTree
txfCEKUnitCase :: forall errPP errPreC errPostC.
(Show errPP, Show errPreC, Show errPostC) =>
TxFCEKCase errPP errPreC errPostC -> TestTree
txfCEKUnitCase (TxFCEKCase PipelinedTestCase errPP errPreC errPostC TxFCEKInput TxFCEKOutput
pipelinedCase) = PipelinedTestCase errPP errPreC errPostC TxFCEKInput TxFCEKOutput
-> TestTree
forall errPP errPreC errPostC input output.
(Show errPP, Show errPreC, Show errPostC) =>
PipelinedTestCase errPP errPreC errPostC input output -> TestTree
pipelinedUnitCase PipelinedTestCase errPP errPreC errPostC TxFCEKInput TxFCEKOutput
pipelinedCase
nominalPostCondition ::
  ((EvalError, [Text]) -> err) -> PostCondition err TxFCEKOutput
nominalPostCondition :: forall err.
((EvalError, [Text]) -> err) -> PostCondition err TxFCEKOutput
nominalPostCondition (EvalError, [Text]) -> err
toErr =
  (TxFCEKOutput -> Maybe err) -> PostCondition err TxFCEKOutput
forall output err.
(output -> Maybe err) -> PostCondition err output
mkPostCondition ((TxFCEKOutput -> Maybe err) -> PostCondition err TxFCEKOutput)
-> (TxFCEKOutput -> Maybe err) -> PostCondition err TxFCEKOutput
forall a b. (a -> b) -> a -> b
$ \(TxFCEKOutput Either EvalError Script
eitherErrOrScript ExBudget
_ [Text]
logs) ->
    case Either EvalError Script
eitherErrOrScript of
      Left EvalError
evalErr -> err -> Maybe err
forall a. a -> Maybe a
Just (err -> Maybe err) -> err -> Maybe err
forall a b. (a -> b) -> a -> b
$ (EvalError, [Text]) -> err
toErr (EvalError
evalErr, [Text]
logs)
      Right Script
_ -> Maybe err
forall a. Maybe a
Nothing
nominalCaseBasic ::
  
  String ->
  
  ScriptContext ->
  
  Script ->
  TxFCEKCase String String String
nominalCaseBasic :: String
-> ScriptContext -> Script -> TxFCEKCase String String String
nominalCaseBasic String
name' ScriptContext
nominalCtx Script
script =
  let
    errorPrinter :: ((EvalError, [Text]) -> String)
    errorPrinter :: (EvalError, [Text]) -> String
errorPrinter (EvalError
err, [Text]
logs) =
      String
"Script failed.\nEvaluation error: \n"
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> EvalError -> String
forall a. Show a => a -> String
show EvalError
err
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> String
"\n Trace Log: \n"
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> String -> [String] -> String
forall a. [a] -> [[a]] -> [a]
intercalate String
"\n" (Text -> String
forall a. Show a => a -> String
show (Text -> String) -> [Text] -> [String]
forall (f :: Type -> Type) a b. Functor f => (a -> b) -> f a -> f b
<$> [Text]
logs)
   in
    String
-> [PreProcessor String TxFCEKInput]
-> [PreCondition String TxFCEKInput]
-> TxFCEKInput
-> [PostCondition String TxFCEKOutput]
-> TxFCEKCase String String String
forall errPP errPreC errPostC.
String
-> [PreProcessor errPP TxFCEKInput]
-> [PreCondition errPreC TxFCEKInput]
-> TxFCEKInput
-> [PostCondition errPostC TxFCEKOutput]
-> TxFCEKCase errPP errPreC errPostC
mkTxFCEKCase
      String
name'
      []
      []
      (ScriptContext -> Script -> TxFCEKInput
TxFCEKInput ScriptContext
nominalCtx Script
script)
      [((EvalError, [Text]) -> String)
-> PostCondition String TxFCEKOutput
forall err.
((EvalError, [Text]) -> err) -> PostCondition err TxFCEKOutput
nominalPostCondition (EvalError, [Text]) -> String
errorPrinter]
attackCaseRegexPostCondition ::
  
  
  ((Maybe Int, [Text]) -> err) ->
  
  
  RE ->
  PostCondition err TxFCEKOutput
attackCaseRegexPostCondition :: forall err.
((Maybe Int, [Text]) -> err)
-> RE -> PostCondition err TxFCEKOutput
attackCaseRegexPostCondition (Maybe Int, [Text]) -> err
toErr RE
expectedFailureRE =
  (TxFCEKOutput -> Maybe err) -> PostCondition err TxFCEKOutput
forall output err.
(output -> Maybe err) -> PostCondition err output
mkPostCondition ((TxFCEKOutput -> Maybe err) -> PostCondition err TxFCEKOutput)
-> (TxFCEKOutput -> Maybe err) -> PostCondition err TxFCEKOutput
forall a b. (a -> b) -> a -> b
$ \(TxFCEKOutput Either EvalError Script
eitherErrorOrScript ExBudget
_ [Text]
logs) ->
    case Either EvalError Script
eitherErrorOrScript of
      Left EvalError
_evalError ->
        let
          numMatches :: Int
numMatches = Matches Text -> Int
forall a. Matches a -> Int
countMatches ([Text] -> Text
forall m. Monoid m => [m] -> m
forall (t :: Type -> Type) m. (Foldable t, Monoid m) => t m -> m
fold [Text]
logs Text -> RE -> Matches Text
*=~ RE
expectedFailureRE)
         in
          if Int
numMatches Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
1
            then Maybe err
forall a. Maybe a
Nothing
            else err -> Maybe err
forall a. a -> Maybe a
Just (err -> Maybe err) -> err -> Maybe err
forall a b. (a -> b) -> a -> b
$ (Maybe Int, [Text]) -> err
toErr (Int -> Maybe Int
forall a. a -> Maybe a
Just Int
numMatches, [Text]
logs)
      Right Script
_ ->
        err -> Maybe err
forall a. a -> Maybe a
Just (err -> Maybe err) -> err -> Maybe err
forall a b. (a -> b) -> a -> b
$ (Maybe Int, [Text]) -> err
toErr (Maybe Int
forall a. Maybe a
Nothing, [Text]
logs)
attackCaseBasicRegex ::
  
  String ->
  
  
  RE ->
  
  ScriptContext ->
  
  Script ->
  
  PreProcessor errPP TxFCEKInput ->
  TxFCEKCase errPP String String
attackCaseBasicRegex :: forall errPP.
String
-> RE
-> ScriptContext
-> Script
-> PreProcessor errPP TxFCEKInput
-> TxFCEKCase errPP String String
attackCaseBasicRegex String
name' RE
expectedFailureRE ScriptContext
nominalCtx Script
script' PreProcessor errPP TxFCEKInput
attack =
  let
    errorPrinter :: (Maybe Int, [Text]) -> String
errorPrinter (Maybe Int
Nothing, [Text]
logs) =
      String
"Script Succeeded, but failure was expected. Logs: \n"
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> String -> [String] -> String
forall a. [a] -> [[a]] -> [a]
intercalate String
"\n    " (Text -> String
forall a. Show a => a -> String
show (Text -> String) -> [Text] -> [String]
forall (f :: Type -> Type) a b. Functor f => (a -> b) -> f a -> f b
<$> [Text]
logs)
    errorPrinter (Just Int
numMatches, [Text]
logs) =
      String
"The script execution against the attack case failed, but without "
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> String
"the expected error. The regex\n    "
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> RE -> String
reSource RE
expectedFailureRE
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> String
"\nshould have matched the logs exactly once, but it matched "
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> Int -> String
forall a. Show a => a -> String
show Int
numMatches
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> String
" times."
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> String
"\nLogs:\n    "
        String -> ShowS
forall a. Semigroup a => a -> a -> a
<> String -> [String] -> String
forall a. [a] -> [[a]] -> [a]
intercalate String
"\n    " (Text -> String
forall a. Show a => a -> String
show (Text -> String) -> [Text] -> [String]
forall (f :: Type -> Type) a b. Functor f => (a -> b) -> f a -> f b
<$> [Text]
logs)
   in
    String
-> [PreProcessor errPP TxFCEKInput]
-> [PreCondition String TxFCEKInput]
-> TxFCEKInput
-> [PostCondition String TxFCEKOutput]
-> TxFCEKCase errPP String String
forall errPP errPreC errPostC.
String
-> [PreProcessor errPP TxFCEKInput]
-> [PreCondition errPreC TxFCEKInput]
-> TxFCEKInput
-> [PostCondition errPostC TxFCEKOutput]
-> TxFCEKCase errPP errPreC errPostC
mkTxFCEKCase
      String
name'
      [PreProcessor errPP TxFCEKInput
attack]
      []
      (ScriptContext -> Script -> TxFCEKInput
TxFCEKInput ScriptContext
nominalCtx Script
script')
      [((Maybe Int, [Text]) -> String)
-> RE -> PostCondition String TxFCEKOutput
forall err.
((Maybe Int, [Text]) -> err)
-> RE -> PostCondition err TxFCEKOutput
attackCaseRegexPostCondition (Maybe Int, [Text]) -> String
errorPrinter RE
expectedFailureRE]