-
Notifications
You must be signed in to change notification settings - Fork 719
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
7 changed files
with
1,443 additions
and
4 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
216 changes: 216 additions & 0 deletions
216
cardano-testnet/test/cardano-testnet-test/Cardano/Testnet/Test/Cli/Queries.hs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,216 @@ | ||
{-# LANGUAGE DataKinds #-} | ||
{-# LANGUAGE GADTs #-} | ||
{-# LANGUAGE NamedFieldPuns #-} | ||
{-# LANGUAGE NumericUnderscores #-} | ||
{-# LANGUAGE OverloadedStrings #-} | ||
{-# LANGUAGE ScopedTypeVariables #-} | ||
{-# LANGUAGE TypeApplications #-} | ||
|
||
{- HLINT ignore "Use head" -} | ||
|
||
module Cardano.Testnet.Test.Cli.Queries | ||
( hprop_cli_queries | ||
) where | ||
|
||
import Cardano.Api | ||
import qualified Cardano.Api as Api | ||
|
||
import Cardano.CLI.Types.Output (QueryTipLocalStateOutput) | ||
import Cardano.Testnet | ||
|
||
import Prelude | ||
|
||
import Control.Monad (forM_, void) | ||
import qualified Data.Aeson as Aeson | ||
import qualified Data.Text as Text | ||
import qualified Data.Vector as Vector | ||
import GHC.Stack (HasCallStack) | ||
import System.FilePath ((</>)) | ||
|
||
import Testnet.Components.Query (checkDRepsNumber, findLargestUtxoForPaymentKey, | ||
getEpochStateView) | ||
import qualified Testnet.Process.Cli as H | ||
import qualified Testnet.Process.Run as H | ||
import qualified Testnet.Property.Utils as H | ||
import Testnet.Runtime | ||
|
||
import Hedgehog | ||
import qualified Hedgehog as H | ||
import qualified Hedgehog.Extras as H | ||
import qualified Hedgehog.Extras.Stock.IO.Network.Sprocket as IO | ||
import qualified Hedgehog.Extras.Test.Golden as H | ||
|
||
-- | Test CLI queries | ||
-- Execute me with: | ||
-- @cabal test cardano-testnet-test --test-options '-p "/CliQueries/"'@ | ||
-- If you want to recreate golden files, run the comment with | ||
-- RECREATE_GOLDEN_FILES=1 as its prefix | ||
hprop_cli_queries :: Property | ||
hprop_cli_queries = H.integrationWorkspace "cli-queries" $ \tempAbsBasePath' -> do | ||
conf@Conf { tempAbsPath=tempAbsPath@(TmpAbsolutePath work) } | ||
<- mkConf tempAbsBasePath' | ||
let tempAbsPath' = unTmpAbsPath tempAbsPath | ||
tempBaseAbsPath = makeTmpBaseAbsPath tempAbsPath | ||
|
||
let sbe = ShelleyBasedEraConway | ||
era = toCardanoEra sbe | ||
cEra = AnyCardanoEra era | ||
fastTestnetOptions = cardanoDefaultTestnetOptions | ||
{ cardanoEpochLength = 100 | ||
, cardanoSlotLength = 0.1 | ||
, cardanoNodeEra = cEra | ||
} | ||
|
||
TestnetRuntime | ||
{ testnetMagic | ||
, poolNodes | ||
, wallets | ||
, configurationFile | ||
} | ||
<- cardanoTestnetDefault fastTestnetOptions conf | ||
|
||
poolNode1 <- H.headM poolNodes | ||
poolSprocket1 <- H.noteShow $ nodeSprocket $ poolRuntime poolNode1 | ||
execConfig <- H.mkExecConfig tempBaseAbsPath poolSprocket1 testnetMagic | ||
|
||
let socketName' = IO.sprocketName poolSprocket1 | ||
socketBase = IO.sprocketBase poolSprocket1 -- /tmp | ||
socketPath = socketBase </> socketName' | ||
|
||
epochStateView <- getEpochStateView (File configurationFile) (File socketPath) | ||
|
||
H.note_ $ "Sprocket: " <> show poolSprocket1 | ||
H.note_ $ "Abs path: " <> tempAbsBasePath' | ||
H.note_ $ "Socketpath: " <> socketPath | ||
H.note_ $ "Foldblocks config file: " <> configurationFile | ||
|
||
let drepVkeyFp :: Int -> FilePath | ||
drepVkeyFp n = tempAbsPath' </> "drep-keys" </> ("drep" <> show n) </> "drep.vkey" | ||
|
||
drepSKeyFp :: Int -> FilePath | ||
drepSKeyFp n = tempAbsPath' </> "drep-keys" </> ("drep" <> show n) </> "drep.skey" | ||
|
||
gov <- H.createDirectoryIfMissing $ tempAbsPath' </> "governance" | ||
|
||
-- Create DRep and register them | ||
let drepCertFile :: Int -> FilePath | ||
drepCertFile n = gov </> "drep-keys" <> "drep" <> show n <> ".regcert" | ||
forM_ [1..3] $ \n -> do | ||
H.execCli' execConfig | ||
[ "conway", "governance", "drep", "registration-certificate" | ||
, "--drep-verification-key-file", drepVkeyFp n | ||
, "--key-reg-deposit-amt", show @Int 1_000_000 | ||
, "--out-file", drepCertFile n | ||
] | ||
|
||
-- Retrieve UTxOs for registration submission | ||
txin1 <- findLargestUtxoForPaymentKey epochStateView sbe $ wallets !! 0 | ||
|
||
drepRegTxbodyFp <- H.note $ work </> "drep.registration.txbody" | ||
drepRegTxSignedFp <- H.note $ work </> "drep.registration.tx" | ||
|
||
void $ H.execCli' execConfig | ||
[ "conway", "transaction", "build" | ||
, "--change-address", Text.unpack $ paymentKeyInfoAddr $ wallets !! 0 | ||
, "--tx-in", Text.unpack $ renderTxIn txin1 | ||
, "--tx-out", Text.unpack (paymentKeyInfoAddr (wallets !! 1)) <> "+" <> show @Int 5_000_000 | ||
, "--certificate-file", drepCertFile 1 | ||
, "--certificate-file", drepCertFile 2 | ||
, "--certificate-file", drepCertFile 3 | ||
, "--witness-override", show @Int 4 | ||
, "--out-file", drepRegTxbodyFp | ||
] | ||
|
||
void $ H.execCli' execConfig | ||
[ "conway", "transaction", "sign" | ||
, "--tx-body-file", drepRegTxbodyFp | ||
, "--signing-key-file", paymentSKey $ paymentKeyInfoPair $ wallets !! 0 | ||
, "--signing-key-file", drepSKeyFp 1 | ||
, "--signing-key-file", drepSKeyFp 2 | ||
, "--signing-key-file", drepSKeyFp 3 | ||
, "--out-file", drepRegTxSignedFp | ||
] | ||
|
||
void $ H.execCli' execConfig | ||
[ "conway", "transaction", "submit" | ||
, "--tx-file", drepRegTxSignedFp | ||
] | ||
-- Finished creating DReps | ||
|
||
-- TODO: we could wait less: waiting 1 block should suffice. | ||
checkDRepsNumber sbe (Api.File configurationFile) (Api.File socketPath) execConfig 3 | ||
|
||
-- protocol-parameters to stdout | ||
protocolParametersOut <- H.execCli' execConfig | ||
[ "conway", "query", "protocol-parameters" ] | ||
H.diffVsGoldenFile | ||
protocolParametersOut | ||
"test/cardano-testnet-test/files/golden/queries/protocolParametersOut.txt" | ||
-- protocol-parameters to a file | ||
let protocolParametersOutFile = work </> "protocol-parameters-out.json" | ||
H.noteM_ $ H.execCli' execConfig [ "conway", "query", "protocol-parameters" | ||
, "--out-file", protocolParametersOutFile] | ||
H.diffFileVsGoldenFile | ||
protocolParametersOutFile | ||
"test/cardano-testnet-test/files/golden/queries/protocolParametersFileOut.json" | ||
|
||
-- tip to stdout | ||
_ :: QueryTipLocalStateOutput <- H.noteShowM $ H.execCliStdoutToJson execConfig [ "conway", "query", "tip" ] | ||
-- tip to a file | ||
let tipOutFile = work </> "tip-out.json" | ||
H.noteM_ $ H.execCli' execConfig [ "conway", "query", "tip" | ||
, "--out-file", tipOutFile] | ||
_ :: QueryTipLocalStateOutput <- H.leftFailM . H.readJsonFile $ tipOutFile | ||
|
||
-- stake-pools to stdout | ||
stakePoolsOut <- H.execCli' execConfig [ "conway", "query", "stake-pools" ] | ||
length (lines stakePoolsOut) H.=== 3 -- Because, by default, 3 stake pools are created | ||
-- Light test of the query's answer, the ids should exist: | ||
forM_ (lines stakePoolsOut) $ \stakePoolId -> do | ||
H.execCli' execConfig [ "conway", "query", "pool-state" | ||
, "--stake-pool-id", stakePoolId ] | ||
-- stake-pools to a file | ||
let stakePoolsOutFile = work </> "stake-pools-out.json" | ||
H.noteM_ $ H.execCli' execConfig [ "conway", "query", "stake-pools" | ||
, "--out-file", stakePoolsOutFile] | ||
|
||
-- query drep-state to stdout | ||
-- TODO: deserialize to a Haskell value when | ||
-- https://github.com/IntersectMBO/cardano-cli/issues/606 is tackled | ||
dreps :: Aeson.Value <- H.noteShowM $ H.execCliStdoutToJson execConfig [ "conway", "query", "drep-state", "--all-dreps"] | ||
assertArrayOfSize dreps 3 | ||
-- query drep-state to a file | ||
let drepStateOutFile = work </> "drep-state-out.json" | ||
H.noteM_ $ H.execCli' execConfig [ "conway", "query", "drep-state", "--all-dreps" | ||
, "--out-file", drepStateOutFile] | ||
_ :: Aeson.Value <- H.leftFailM . H.readJsonFile $ drepStateOutFile | ||
|
||
H.success | ||
|
||
-- | @assertArrayOfSize v n@ checks that the value is a JSON array of size @n@, | ||
-- otherwise it fails the test. | ||
assertArrayOfSize :: () | ||
=> HasCallStack | ||
=> MonadTest m | ||
=> Aeson.Value | ||
-> Int | ||
-> m () | ||
assertArrayOfSize v n = | ||
case v of | ||
Aeson.Array a -> do | ||
let actualLength = Vector.length a | ||
if actualLength == n | ||
then H.success | ||
else do | ||
H.note_ $ "Expected an array of length " <> show n <> ", but got: " <> show actualLength | ||
H.failure | ||
Aeson.Object _ -> failWrongType "object" | ||
Aeson.Bool _ -> failWrongType "bool" | ||
Aeson.Number _ -> failWrongType "number" | ||
Aeson.Null -> failWrongType "null" | ||
Aeson.String _ -> failWrongType "string" | ||
where | ||
failWrongType got = do | ||
H.note_ $ "Expected a JSON object, but received: " <> got | ||
H.failure | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.