Skip to content

Commit

Permalink
Merge pull request #5684 from IntersectMBO/smelc/test-cli-queries
Browse files Browse the repository at this point in the history
cardano-testnet-test: Test CLI queries
  • Loading branch information
smelc authored Mar 11, 2024
2 parents 1995418 + dfb527e commit c862dfe
Show file tree
Hide file tree
Showing 9 changed files with 1,513 additions and 78 deletions.
3 changes: 2 additions & 1 deletion cardano-testnet/cardano-testnet.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -175,6 +175,7 @@ test-suite cardano-testnet-test
Cardano.Testnet.Test.Cli.Conway.Plutus
Cardano.Testnet.Test.Cli.Conway.StakeSnapshot
Cardano.Testnet.Test.Cli.KesPeriodInfo
Cardano.Testnet.Test.Cli.Queries
Cardano.Testnet.Test.Cli.QuerySlotNumber
Cardano.Testnet.Test.FoldBlocks
Cardano.Testnet.Test.Misc
Expand Down Expand Up @@ -206,7 +207,6 @@ test-suite cardano-testnet-test
, cardano-testnet
, containers
, directory
, exceptions
, filepath
, hedgehog
, hedgehog-extras
Expand All @@ -220,6 +220,7 @@ test-suite cardano-testnet-test
, text
, time
, transformers
, vector

ghc-options: -threaded -rtsopts "-with-rtsopts=-N -T"

Expand Down
68 changes: 68 additions & 0 deletions cardano-testnet/src/Testnet/Components/Query.hs
Original file line number Diff line number Diff line change
@@ -1,12 +1,14 @@
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeApplications #-}

module Testnet.Components.Query
( QueryTip
, EpochStateView
, checkDRepsNumber
, getEpochState
, queryTip
, waitUntilEpoch
Expand All @@ -19,6 +21,8 @@ module Testnet.Components.Query
) where

import Cardano.Api as Api
import Cardano.Api.Ledger (StandardCrypto)
import qualified Cardano.Api.Ledger as L
import Cardano.Api.Shelley (ShelleyLedgerEra, fromShelleyTxIn, fromShelleyTxOut)

import Cardano.CLI.Types.Output
Expand All @@ -28,6 +32,7 @@ import qualified Cardano.Ledger.UTxO as L
import Control.Exception.Safe (MonadCatch)
import Control.Monad
import Control.Monad.Trans.Resource
import Control.Monad.Trans.State.Strict (put)
import Data.Aeson
import Data.Bifunctor (bimap)
import Data.IORef
Expand All @@ -44,6 +49,7 @@ import GHC.Stack
import Lens.Micro ((^.))
import System.Directory (doesFileExist, removeFile)

import qualified Testnet.Process.Cli as P
import qualified Testnet.Process.Run as H
import Testnet.Property.Assert
import Testnet.Property.Utils (runInBackground)
Expand Down Expand Up @@ -218,3 +224,65 @@ findLargestUtxoForPaymentKey epochStateView sbe address =
. H.nothingFailM
$ findLargestUtxoWithAddress epochStateView sbe (paymentKeyInfoAddr address)


-- | @checkDRepsNumber config socket execConfig n@
-- wait for the number of DReps being @n@ for two epochs. If
-- this number is not attained before two epochs, the test is failed.
checkDRepsNumber ::
(HasCallStack, MonadIO m, MonadCatch m, MonadTest m)
=> ShelleyBasedEra ConwayEra -- ^ The era in which the test runs
-> NodeConfigFile 'In
-> SocketPath
-> H.ExecConfig
-> Int
-> m ()
checkDRepsNumber sbe configurationFile socketPath execConfig expectedDRepsNb = do
QueryTipLocalStateOutput{mEpoch} <- P.execCliStdoutToJson execConfig [ "query", "tip" ]
currentEpoch <- H.evalMaybe mEpoch
let terminationEpoch = succ . succ $ currentEpoch
void $ H.evalMaybeM $ checkDRepsNumber' sbe configurationFile socketPath terminationEpoch expectedDRepsNb

-- | @checkDRepsNumber' config socket terminationEpoch n@
-- wait until @terminationEpoch@ for the number of DReps being @n@. If
-- this number is not attained before @terminationEpoch@, the test is failed.
-- So if you call this function, you are expecting the number of DReps to already
-- be @n@, or to be @n@ before @terminationEpoch@
checkDRepsNumber' ::
(HasCallStack, MonadIO m, MonadTest m)
=> ShelleyBasedEra ConwayEra -- ^ The era in which the test runs
-> NodeConfigFile In
-> SocketPath
-> EpochNo -- ^ The termination epoch: the constitution proposal must be found *before* this epoch
-> Int -- ^ The expected numbers of DReps. If this number is not reached until the termination epoch, this function fails the test.
-> m (Maybe [L.DRepState StandardCrypto]) -- ^ The DReps when the expected number of DReps was attained.
checkDRepsNumber' sbe nodeConfigFile socketPath maxEpoch expectedDRepsNb = do
result <- runExceptT $ foldEpochState nodeConfigFile socketPath QuickValidation maxEpoch Nothing
$ \(AnyNewEpochState actualEra newEpochState) -> do
case testEquality sbe actualEra of
Just Refl -> do
let dreps = Map.elems $ shelleyBasedEraConstraints sbe newEpochState
^. L.nesEsL
. L.esLStateL
. L.lsCertStateL
. L.certVStateL
. L.vsDRepsL
if length dreps == expectedDRepsNb then do
put $ Just dreps
pure ConditionMet
else
pure ConditionNotMet
Nothing -> do
error $ "Eras mismatch! expected: " <> show sbe <> ", actual: " <> show actualEra
case result of
Left (FoldBlocksApplyBlockError (TerminationEpochReached epochNo)) -> do
H.note_ $ unlines
[ "waitDRepsNumber: drep number did not become " <> show expectedDRepsNb <> " before termination epoch: " <> show epochNo
, "This is likely an error of this test." ]
H.failure
Left err -> do
H.note_ $ unlines
[ "waitDRepsNumber: could not reach termination epoch: " <> docToString (prettyError err)
, "This is probably an error unrelated to this test." ]
H.failure
Right (_, val) ->
return val
4 changes: 2 additions & 2 deletions cardano-testnet/src/Testnet/Process/Cli.hs
Original file line number Diff line number Diff line change
Expand Up @@ -126,11 +126,11 @@ cliNodeKeyGen tmpDir vkey skey counter = do
-- returning JSON to stdout, and needs going through a file instead, probably
-- you should add a similar function to this one.
execCliStdoutToJson :: ()
=> (Aeson.FromJSON a, MonadTest m, MonadCatch m, MonadIO m)
=> (HasCallStack, Aeson.FromJSON a, MonadTest m, MonadCatch m, MonadIO m)
=> ExecConfig -- ^ The configuration with which to call the CLI
-> [String] -- ^ The CLI command to execute
-> m a
execCliStdoutToJson execConfig cmd = do
execCliStdoutToJson execConfig cmd = GHC.withFrozenCallStack $ do
result <- execCli' execConfig cmd
H.leftFail $ Aeson.eitherDecode $ Data.String.fromString result

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,24 +19,12 @@ module Cardano.Testnet.Test.Cli.Conway.DRepRetirement

import Cardano.Api
import qualified Cardano.Api as Api
import Cardano.Api.Ledger
import qualified Cardano.Api.Ledger as L

import Cardano.CLI.Types.Output (QueryTipLocalStateOutput (..))
import qualified Cardano.Ledger.Shelley.LedgerState as L
import Cardano.Testnet

import Prelude

import Control.Monad (void)
import Control.Monad.Catch (MonadCatch)
import Control.Monad.Trans.State.Strict (put)
import Data.Data
import qualified Data.Map.Strict as Map
import qualified Data.Text as Text
import Data.Type.Equality (testEquality)
import GHC.Stack
import Lens.Micro ((^.))
import System.FilePath ((</>))

import Testnet.Components.Query
Expand Down Expand Up @@ -159,7 +147,7 @@ hprop_drep_retirement = H.integrationRetryWorkspace 2 "drep-retirement" $ \tempA
configFile' = Api.File configurationFile
socketPath' = Api.File socketPath

waitDRepsNumber configFile' socketPath' execConfig sizeBefore
checkDRepsNumber sbe configFile' socketPath' execConfig sizeBefore

-- Deregister first DRep
let dreprRetirementCertFile = gov </> "drep-keys" <> "drep1.retirementcert"
Expand Down Expand Up @@ -207,65 +195,5 @@ hprop_drep_retirement = H.integrationRetryWorkspace 2 "drep-retirement" $ \tempA

-- The important bit is that we pass (sizeBefore - 1) as the last argument,
-- to witness that the number of dreps indeed decreased.
waitDRepsNumber configFile' socketPath' execConfig (sizeBefore - 1)
checkDRepsNumber sbe configFile' socketPath' execConfig (sizeBefore - 1)
H.success

-- | @waitDRepsNumber config socket execConfig n@
-- wait for the number of DReps being @n@ for two epochs. If
-- this number is not attained before two epochs, the test is failed.
waitDRepsNumber ::
(HasCallStack, MonadIO m, MonadCatch m, MonadTest m)
=> NodeConfigFile 'In
-> SocketPath
-> H.ExecConfig
-> Int
-> m ()
waitDRepsNumber configurationFile socketPath execConfig expectedDRepsNb = do
QueryTipLocalStateOutput{mEpoch} <- P.execCliStdoutToJson execConfig [ "query", "tip" ]
currentEpoch <- H.evalMaybe mEpoch
let terminationEpoch = succ . succ $ currentEpoch
void $ H.evalMaybeM $ waitDRepsNumber' configurationFile socketPath terminationEpoch expectedDRepsNb

-- | @waitDRepsNumber' config socket terminationEpoch n@
-- wait until @terminationEpoch@ for the number of DReps being @n@. If
-- this number is not attained before @terminationEpoch@, the test is failed.
-- So if you call this function, you are expecting the number of DReps to already
-- be @n@, or to be @n@ before @terminationEpoch@
waitDRepsNumber' ::
(HasCallStack, MonadIO m, MonadTest m)
=> NodeConfigFile In
-> SocketPath
-> EpochNo -- ^ The termination epoch: the constitution proposal must be found *before* this epoch
-> Int -- ^ The expected numbers of DReps. If this number is not reached until the termination epoch, this function fails the test.
-> m (Maybe [L.DRepState StandardCrypto]) -- ^ The DReps when the expected number of DReps was attained.
waitDRepsNumber' nodeConfigFile socketPath maxEpoch expectedDRepsNb = do
result <- runExceptT $ foldEpochState nodeConfigFile socketPath QuickValidation maxEpoch Nothing
$ \(AnyNewEpochState actualEra newEpochState) -> do
case testEquality sbe actualEra of
Just Refl -> do
let dreps = Map.elems $ shelleyBasedEraConstraints sbe newEpochState
^. L.nesEsL
. L.esLStateL
. L.lsCertStateL
. L.certVStateL
. L.vsDRepsL
if length dreps == expectedDRepsNb then do
put $ Just dreps
pure ConditionMet
else
pure ConditionNotMet
Nothing -> do
error $ "Eras mismatch! expected: " <> show sbe <> ", actual: " <> show actualEra
case result of
Left (FoldBlocksApplyBlockError (TerminationEpochReached epochNo)) -> do
H.note_ $ unlines
[ "waitDRepsNumber: drep number did not become " <> show expectedDRepsNb <> " before termination epoch: " <> show epochNo
, "This is likely an error of this test." ]
H.failure
Left err -> do
H.note_ $ unlines
[ "waitDRepsNumber: could not reach termination epoch: " <> docToString (prettyError err)
, "This is probably an error unrelated to this test." ]
H.failure
Right (_, val) ->
return val
Loading

0 comments on commit c862dfe

Please sign in to comment.