Skip to content
This repository has been archived by the owner on Feb 9, 2021. It is now read-only.

Commit

Permalink
Add tests for multiple validation modes
Browse files Browse the repository at this point in the history
  • Loading branch information
intricate committed Jun 11, 2019
1 parent a0a0be6 commit 0b64a31
Show file tree
Hide file tree
Showing 9 changed files with 696 additions and 8 deletions.
3 changes: 3 additions & 0 deletions cardano-ledger/cardano-ledger.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -174,6 +174,7 @@ test-suite cardano-ledger-test
Test.Cardano.Chain.Block.Gen
Test.Cardano.Chain.Block.Model
Test.Cardano.Chain.Block.Validation
Test.Cardano.Chain.Block.ValidationMode

Test.Cardano.Chain.Common.Address
Test.Cardano.Chain.Common.CBOR
Expand All @@ -192,6 +193,7 @@ test-suite cardano-ledger-test
Test.Cardano.Chain.Elaboration.Block
Test.Cardano.Chain.Elaboration.Delegation
Test.Cardano.Chain.Elaboration.Keys
Test.Cardano.Chain.Elaboration.Update
Test.Cardano.Chain.Elaboration.UTxO

Test.Cardano.Chain.Epoch.File
Expand All @@ -215,6 +217,7 @@ test-suite cardano-ledger-test
Test.Cardano.Chain.UTxO.Gen
Test.Cardano.Chain.UTxO.Json
Test.Cardano.Chain.UTxO.Model
Test.Cardano.Chain.UTxO.ValidationMode

Test.Cardano.Chain.Update.CBOR
Test.Cardano.Chain.Update.Example
Expand Down
274 changes: 274 additions & 0 deletions cardano-ledger/test/Test/Cardano/Chain/Block/ValidationMode.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,274 @@
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE TemplateHaskell #-}
{-# LANGUAGE TypeApplications #-}

module Test.Cardano.Chain.Block.ValidationMode
( tests
)
where

import Cardano.Prelude hiding (State)
import Test.Cardano.Prelude

import Control.Lens ((^.))
import qualified Data.Bimap as BM
import qualified Data.Map.Strict as M
import qualified Data.Sequence as Seq
import qualified Data.Set as S

import Cardano.Binary (Annotated (..))
import Cardano.Chain.Block
( ABlock (..)
, AHeader (..)
, BlockValidationMode (..)
, Proof (..)
, blockProof
, initialChainValidationState
, updateBlock
)
import Cardano.Chain.Delegation as Delegation
import Cardano.Chain.UTxO (TxProof)
import Cardano.Crypto (Hash)

import Hedgehog
import qualified Hedgehog.Gen as Gen
import qualified Hedgehog.Range as Range

import qualified Cardano.Spec.Chain.STS.Block as Abstract
import qualified Cardano.Spec.Chain.STS.Rule.Chain as Abstract
import Cardano.Spec.Chain.STS.Rule.Chain (CHAIN)
import Cardano.Ledger.Spec.STS.UTXOWS (UTXOWS)
import Cardano.Ledger.Spec.STS.UTXO (UTxOEnv (..), UTxOState (..))
import Control.State.Transition
import Control.State.Transition.Generator
import qualified Ledger.Core as Abstract
import Ledger.Delegation
( ADELEGS
, DELEG
, DIState (..)
, DSEnv (..)
, DState (..)
)
import Ledger.GlobalParams (lovelaceCap)
import qualified Ledger.Update as Abstract
import qualified Ledger.UTxO as Abstract

import Test.Cardano.Chain.Block.Model (elaborateAndUpdate)
import qualified Test.Cardano.Chain.Delegation.Gen as Delegation
import Test.Cardano.Chain.Elaboration.Block (abEnvToCfg, elaborateBS, rcDCert)
import qualified Test.Cardano.Chain.Update.Gen as Update
import Test.Cardano.Chain.UTxO.Gen (genTxProof)
import Test.Cardano.Chain.UTxO.Model (elaborateInitialUTxO)
import Test.Cardano.Crypto.Gen (feedPM, genAbstractHash)
import Test.Options (TSGroup, TSProperty, withTestsTS)

--------------------------------------------------------------------------------
-- BlockValidationMode Properties
--------------------------------------------------------------------------------

-- | Property: When calling 'updateBlock' given a valid 'Block', validation
-- should pass in all 'BlockValidationMode's.
ts_prop_updateBlock_Valid :: TSProperty
ts_prop_updateBlock_Valid =
withTestsTS 100
. property
$ do
chainEnv@(_, abstractInitialUTxO, _, _) <- forAll $ initEnvGen @CHAIN
chainState <- forAll $ genInitialChainState chainEnv
abstractBlock <- forAll $
Abstract.sigGenChain
Abstract.NoGenDelegation
Abstract.NoGenUTxO
chainEnv
chainState
let config = abEnvToCfg chainEnv
cvs = either (panic . show) (\a -> a) (initialChainValidationState config)
(_, txIdMap) = elaborateInitialUTxO abstractInitialUTxO
bvmode <- forAll $ genBlockValidationMode
case elaborateAndUpdate bvmode config (cvs, txIdMap) (chainState, abstractBlock) of
Left _ -> failure
Right _ -> success

-- | Property: When calling 'updateBlock' given a 'Block' with an invalid
-- 'Proof', 'Block' validation should only pass in the 'BlockRevalidate' mode.
-- This is because this mode does not perform any validation on the 'Block'.
ts_prop_updateBlock_InvalidProof :: TSProperty
ts_prop_updateBlock_InvalidProof =
withTestsTS 100
. property
$ do
chainEnv@(_, abstractInitialUTxO, _, _) <- forAll $ initEnvGen @CHAIN
chainState <- forAll $ genInitialChainState chainEnv
abstractBlock <- forAll $
Abstract.sigGenChain
Abstract.NoGenDelegation
Abstract.NoGenUTxO
chainEnv
chainState
let config = abEnvToCfg chainEnv
cvs = either (panic . show) (\a -> a) (initialChainValidationState config)
(_, txIdMap) = elaborateInitialUTxO abstractInitialUTxO
dCert = rcDCert (abstractBlock ^. Abstract.bHeader . Abstract.bhIssuer) chainState
bvmode <- forAll $ genBlockValidationMode
let (concreteBlock, _txIdMap') = elaborateBS txIdMap config dCert cvs abstractBlock
annotateShow concreteBlock
invalidBlock <- forAll $ invalidateABlockProof concreteBlock
case updateBlock bvmode config cvs invalidBlock of
Left _ ->
if bvmode == BlockValidate
then success
else failure
Right _ ->
if bvmode == BlockRevalidate
then success
else failure

--------------------------------------------------------------------------------
-- Abstract Generators
--------------------------------------------------------------------------------

genInitialChainState
:: Environment CHAIN
-> Gen (State CHAIN)
genInitialChainState env = do
let (_slot, utxo0', _dsenv, pps') = env
utxoEnv = UTxOEnv { utxo0 = utxo0', pps = pps' }
s0 = Abstract.Slot 0
utxoSt0 = createInitialUTxOState utxoEnv
initialDelegEnv <- initEnvGen @DELEG
let initialADelegsEnv = _dSEnvAllowedDelegators initialDelegEnv
let ds = createInitialDIState (createInitialDState initialADelegsEnv)
pure $! ( s0
, (Seq.fromList . BM.keys . _dIStateDelegationMap) ds
, Abstract.genesisHash
, utxoSt0
, ds
, Abstract.emptyUPIState
)

genHash :: Gen Abstract.Hash
genHash = Abstract.Hash <$> Gen.int Range.constantBounded

genBlockValidationMode :: Gen BlockValidationMode
genBlockValidationMode = Gen.element [BlockValidate, BlockRevalidate]

--------------------------------------------------------------------------------
-- Helpers
--------------------------------------------------------------------------------

createInitialUTxOState
:: Environment UTXOWS
-> State UTXOWS
createInitialUTxOState utxoEnv =
UTxOState{ utxo = utxo0, reserves = lovelaceCap - Abstract.balance utxo0 }
where
UTxOEnv
{ utxo0
} = utxoEnv

createInitialDState
:: Environment ADELEGS
-> State ADELEGS
createInitialDState env =
DState
{ _dStateDelegationMap = BM.fromList $
map (\vkg@(Abstract.VKeyGenesis key) -> (vkg, key))
(S.toList env)
, _dStateLastDelegation = M.fromSet (const (Abstract.Slot 0)) env
}

createInitialDIState
:: State ADELEGS
-> State DELEG
createInitialDIState dState =
DIState
{ _dIStateDelegationMap = _dStateDelegationMap dState
, _dIStateLastDelegation = _dStateLastDelegation dState
, _dIStateScheduledDelegations = []
, _dIStateKeyEpochDelegations = S.empty
}

modifyAHeader
:: (AHeader ByteString -> AHeader ByteString)
-> ABlock ByteString
-> ABlock ByteString
modifyAHeader ahModifier ab =
ab { blockHeader = ahModifier (blockHeader ab) }

modifyAProof
:: (Annotated Proof ByteString -> Annotated Proof ByteString)
-> ABlock ByteString
-> ABlock ByteString
modifyAProof apModifier ab =
modifyAHeader ahModifier ab
where
ahModifier :: AHeader ByteString -> AHeader ByteString
ahModifier ah = ah { aHeaderProof = apModifier (aHeaderProof ah) }

modifyDelegationProof
:: (Hash Delegation.Payload -> Hash Delegation.Payload)
-> ABlock ByteString
-> ABlock ByteString
modifyDelegationProof dpModifier ab =
modifyAProof apModifier ab
where
apModifier :: Annotated Proof ByteString -> Annotated Proof ByteString
apModifier (Annotated p bs) = Annotated
p { proofDelegation = dpModifier (proofDelegation p) }
bs

modifyTxProof
:: (TxProof -> TxProof)
-> ABlock ByteString
-> ABlock ByteString
modifyTxProof tpModifier ab =
modifyAProof apModifier ab
where
apModifier :: Annotated Proof ByteString -> Annotated Proof ByteString
apModifier (Annotated p bs) = Annotated
p { proofUTxO = tpModifier (proofUTxO p) }
bs

invalidateABlockProof
:: ABlock ByteString
-> Gen (ABlock ByteString)
invalidateABlockProof ab =
-- 'Gen.filter' to ensure we don't generate a valid proof
Gen.filter (\x -> blockProof x /= blockProof ab) $ do
txProof <- Gen.choice
[ pure $ (proofUTxO . blockProof) ab
, feedPM genTxProof
]
sscProof <- Gen.choice
[ pure $ (proofSsc . blockProof) ab
-- SscProof only consists of a single nullary constructor.
-- No sense generating one.
]
dlgProof <- Gen.choice
[ pure $ (proofDelegation . blockProof) ab
, genAbstractHash (feedPM Delegation.genPayload)
]
updProof <- Gen.choice
[ pure $ proofUpdate (blockProof ab)
, feedPM Update.genProof
]
pure $ modifyAProof
(\(Annotated _ bs) -> Annotated
(Proof
{ proofUTxO = txProof
, proofSsc = sscProof
, proofDelegation = dlgProof
, proofUpdate = updProof
}
)
bs
)
ab

--------------------------------------------------------------------------------
-- Main Test Export
--------------------------------------------------------------------------------

tests :: TSGroup
tests = $$discoverPropArg
13 changes: 7 additions & 6 deletions cardano-ledger/test/Test/Cardano/Chain/Common/Gen.hs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ module Test.Cardano.Chain.Common.Gen
, genCompactAddress
, genCustomLovelace
, genLovelace
, genLovelaceWithRange
, genLovelacePortion
, genMerkleRoot
, genMerkleTree
Expand Down Expand Up @@ -124,14 +125,14 @@ genCompactAddress :: Gen CompactAddress
genCompactAddress = toCompactAddress <$> genAddress

genCustomLovelace :: Word64 -> Gen Lovelace
genCustomLovelace size =
mkLovelace <$> Gen.word64 (Range.linear 0 size) >>= \case
Right lovelace -> pure lovelace
Left err -> panic $ sformat build err
genCustomLovelace size = genLovelaceWithRange (Range.linear 0 size)

genLovelace :: Gen Lovelace
genLovelace =
mkLovelace <$> Gen.word64 (Range.constant 0 maxLovelaceVal) >>= \case
genLovelace = genLovelaceWithRange (Range.constant 0 maxLovelaceVal)

genLovelaceWithRange :: Range Word64 -> Gen Lovelace
genLovelaceWithRange r =
mkLovelace <$> Gen.word64 r >>= \case
Right lovelace -> pure lovelace
Left err ->
panic $ sformat ("The impossible happened in genLovelace: " . build) err
Expand Down
4 changes: 2 additions & 2 deletions cardano-ledger/test/Test/Cardano/Chain/Elaboration/Block.hs
Original file line number Diff line number Diff line change
Expand Up @@ -216,8 +216,8 @@ abEnvToCfg (_, _, dsEnv, pps) =
gPps = Update.ProtocolParameters
{ Update.ppScriptVersion = 0
, Update.ppSlotDuration = 0
, Update.ppMaxBlockSize = 748 * pps ^. maxBkSz
, Update.ppMaxHeaderSize = 95 * pps ^. maxHdrSz
, Update.ppMaxBlockSize = 832 * pps ^. maxBkSz
, Update.ppMaxHeaderSize = 569 * pps ^. maxHdrSz
, Update.ppMaxTxSize = 318 * pps ^. maxTxSz
, Update.ppMaxProposalSize = 0
, Update.ppMpcThd = LovelacePortion 0
Expand Down
1 change: 1 addition & 0 deletions cardano-ledger/test/Test/Cardano/Chain/Elaboration/UTxO.hs
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
module Test.Cardano.Chain.Elaboration.UTxO
( elaborateUTxOEnv
, elaborateUTxO
, elaborateTx
, elaborateTxWitsBS
, elaborateTxOut
)
Expand Down
45 changes: 45 additions & 0 deletions cardano-ledger/test/Test/Cardano/Chain/Elaboration/Update.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
{-# LANGUAGE OverloadedStrings #-}

module Test.Cardano.Chain.Elaboration.Update
( elaboratePParams
)
where

import Cardano.Prelude

import qualified Cardano.Chain.Common as Concrete
import qualified Cardano.Chain.Update as Concrete

import qualified Ledger.Update as Abstract

import Test.Cardano.Chain.Genesis.Dummy (dummyProtocolParameters)

elaboratePParams :: Abstract.PParams -> Concrete.ProtocolParameters
elaboratePParams pps = Concrete.ProtocolParameters
-- FIXME @intricate: Fields that I'm not sure about, I'm just using
-- dummyProtocolParameters.
{ Concrete.ppScriptVersion = fromIntegral $ Abstract._scriptVersion pps
, Concrete.ppSlotDuration = Concrete.ppSlotDuration dummyProtocolParameters
, Concrete.ppMaxBlockSize = 748 * Abstract._maxBkSz pps
, Concrete.ppMaxHeaderSize = 95 * Abstract._maxHdrSz pps
, Concrete.ppMaxTxSize = 4096 * Abstract._maxTxSz pps
, Concrete.ppMaxProposalSize = Abstract._maxPropSz pps
, Concrete.ppMpcThd = Concrete.ppMpcThd dummyProtocolParameters
, Concrete.ppHeavyDelThd = Concrete.ppHeavyDelThd dummyProtocolParameters
, Concrete.ppUpdateVoteThd = Concrete.ppUpdateVoteThd dummyProtocolParameters
, Concrete.ppUpdateProposalThd = Concrete.ppUpdateProposalThd dummyProtocolParameters
, Concrete.ppUpdateProposalTTL = Concrete.ppUpdateProposalTTL dummyProtocolParameters
, Concrete.ppSoftforkRule = Concrete.ppSoftforkRule dummyProtocolParameters
, Concrete.ppUnlockStakeEpoch = Concrete.ppUnlockStakeEpoch dummyProtocolParameters
, Concrete.ppTxFeePolicy = Concrete.TxFeePolicyTxSizeLinear
(Concrete.TxSizeLinear
(intToLovelace (Abstract._factorA pps))
(intToLovelace (Abstract._factorB pps))
)
}
where
intToLovelace :: Int -> Concrete.Lovelace
intToLovelace x =
case Concrete.mkLovelace (fromIntegral x) of
Left err -> panic $ "intToLovelace: " <> show err
Right l -> l
Loading

0 comments on commit 0b64a31

Please sign in to comment.