commit ec70bfd539fe2e27fd48f5f76395400287ac72d7
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Tue Oct 18 18:58:59 2022 -0500
use LSE
commit 25fff9b3ad1f2dde4cd7cf36977530b06a87d23c
Merge: 01cd3aa 1821dd6
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Tue Oct 18 18:17:45 2022 -0500
Merge branch 'staging' into seungheonoh/ply
commit 01cd3aa7a235e6fe6658246ca1026fa26dc71a83
Author: Hongrui Fang <chfanghr@gmail.com>
Date: Tue Oct 11 12:02:03 2022 +0800
update benchmark
commit a8513244892ce33cfdc9edf8cd501c4985ae8008
Author: Hongrui Fang <chfanghr@gmail.com>
Date: Tue Oct 11 11:59:22 2022 +0800
fix tests
commit 20ca40823485c2e2f78253643cf4453ac7b7ddd5
Author: Hongrui Fang <chfanghr@gmail.com>
Date: Tue Oct 11 11:57:37 2022 +0800
better import
commit a19fe49424210891bd03db71e4083fc1e0edfd98
Author: Hongrui Fang <chfanghr@gmail.com>
Date: Tue Oct 11 11:08:20 2022 +0800
update flake inputs
commit c93b21f1f9441e5c6f54525bf7c6a54757ec36cc
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Mon Oct 10 12:54:12 2022 -0500
tried to make tests pass
commit 1046ae1237299a33c58b48661bdb6d325a22147e
Merge: 2bf4e36 363bd83
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Mon Oct 10 12:18:48 2022 -0500
Merge branch 'staging' into seungheonoh/ply
commit 2bf4e3627c1b229f58078695082da85c80efd560
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Mon Oct 10 10:48:36 2022 -0500
remove junkpile
commit a1dbc9ad9e531fe0d0a0480c4aef9cf9ffa90f1d
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Mon Oct 10 10:47:25 2022 -0500
versions
commit 4542a06ac733858297d3a48c53368fad19dedc43
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Thu Oct 6 22:57:48 2022 -0500
script exporting interface
commit 6bd8c1a1d57e4bf9dc25c3068a9c8eae6bf6a19d
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Thu Oct 6 22:58:41 2022 -0500
fixed tests
commit d3ce2cf95633d336f3e621833677bd5bf10ee2c8
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Sun Oct 2 00:55:18 2022 -0500
fixed tests
commit 1ae64c9f692652b77b0506013853b2ba44267c65
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Sat Oct 1 13:28:20 2022 -0500
linker
commit db88cb75c7b74843141ad8ab4e6522b66d0dcfbc
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Sat Oct 1 01:03:50 2022 -0500
exporting scripts
commit 6389fce28e885a8a7f8669629c266f59c0edb51f
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Sat Oct 1 00:51:49 2022 -0500
made scripts parameterized on the script level
commit aea1e518a8890550bdebd0e5251da11d915c53a9
Author: Seungheon Oh <seungheon.ooh@gmail.com>
Date: Wed Sep 28 19:53:29 2022 -0500
Use `TypedScriptEnvelope` for `Agora.Bootstrap`
238 lines
7.1 KiB
Haskell
238 lines
7.1 KiB
Haskell
{- |
|
|
Module : Property.Governor
|
|
Maintainer : seungheon.ooh@gmail.com
|
|
Description: Property tests for 'Governor' related functions.
|
|
|
|
Property model and tests for 'Governor' related functions
|
|
-}
|
|
module Property.Governor (props) where
|
|
|
|
import Agora.Governor (Governor (gstOutRef), GovernorDatum (..), pisGovernorDatumValid)
|
|
import Agora.Governor.Scripts (governorPolicy)
|
|
import Agora.Proposal (
|
|
ProposalId (ProposalId),
|
|
ProposalThresholds (ProposalThresholds),
|
|
)
|
|
import Agora.Proposal.Time (
|
|
MaxTimeRangeWidth (MaxTimeRangeWidth),
|
|
ProposalTimingConfig (ProposalTimingConfig),
|
|
)
|
|
import Data.Default.Class (Default (def))
|
|
import Data.Tagged (Tagged (Tagged))
|
|
import Data.Universe (Finite (..), Universe (..))
|
|
import Plutarch.Api.V2 (PScriptContext)
|
|
import Plutarch.Builtin (pforgetData)
|
|
import Plutarch.Context (
|
|
MintingBuilder,
|
|
buildMinting',
|
|
input,
|
|
mint,
|
|
output,
|
|
script,
|
|
withDatum,
|
|
withMinting,
|
|
withRef,
|
|
withValue,
|
|
)
|
|
import PlutusLedgerApi.V1.Value (assetClassValue)
|
|
import PlutusLedgerApi.V2 (
|
|
ScriptContext (scriptContextTxInfo),
|
|
TxInInfo (txInInfoOutRef),
|
|
TxInfo (txInfoInputs, txInfoMint, txInfoOutputs),
|
|
TxOut (txOutValue),
|
|
)
|
|
import Property.Generator (genInput, genOutput)
|
|
import Sample.Shared (
|
|
govAssetClass,
|
|
govSymbol,
|
|
govValidatorHash,
|
|
governor,
|
|
gstUTXORef,
|
|
)
|
|
import Test.Tasty (TestTree)
|
|
import Test.Tasty.Plutarch.Property (classifiedPropertyNative)
|
|
import Test.Tasty.QuickCheck (
|
|
Gen,
|
|
Property,
|
|
choose,
|
|
chooseInteger,
|
|
listOf1,
|
|
testProperty,
|
|
)
|
|
|
|
data GovernorDatumCases
|
|
= ExecuteLE0
|
|
| CreateLE0
|
|
| ToVotingLE0
|
|
| VoteLE0
|
|
| CosignLE0
|
|
| Correct
|
|
deriving stock (Eq, Show)
|
|
|
|
instance Universe GovernorDatumCases where
|
|
universe =
|
|
[ ExecuteLE0
|
|
, CreateLE0
|
|
, VoteLE0
|
|
, CosignLE0
|
|
, Correct
|
|
]
|
|
|
|
instance Finite GovernorDatumCases where
|
|
universeF = universe
|
|
cardinality = Tagged 6
|
|
|
|
{- | Property that checks `governorDatumValid`.
|
|
`governorDatumValid` determines if given governor datum is valid or not. This property
|
|
ensures `governorDatumValid` is checking the datum correctly and ruling out improper datum.
|
|
-}
|
|
governorDatumValidProperty :: Property
|
|
governorDatumValidProperty =
|
|
classifiedPropertyNative gen (const []) expected classifier pisGovernorDatumValid
|
|
where
|
|
classifier :: GovernorDatum -> GovernorDatumCases
|
|
classifier
|
|
( (.proposalThresholds) ->
|
|
ProposalThresholds
|
|
execute
|
|
create
|
|
toVoting
|
|
vote
|
|
cosign
|
|
)
|
|
| execute < 0 = ExecuteLE0
|
|
| create < 0 = CreateLE0
|
|
| toVoting < 0 = ToVotingLE0
|
|
| vote < 0 = VoteLE0
|
|
| cosign < 0 = CosignLE0
|
|
| otherwise = Correct
|
|
|
|
expected :: GovernorDatum -> Maybe Bool
|
|
expected c = Just $ classifier c == Correct
|
|
|
|
gen :: GovernorDatumCases -> Gen GovernorDatum
|
|
gen c = do
|
|
thres <- genProposalThresholds c
|
|
|
|
let timing = ProposalTimingConfig 0 0 0 0
|
|
return $ GovernorDatum thres (ProposalId 0) timing (MaxTimeRangeWidth 1) 3
|
|
where
|
|
taggedInteger p = Tagged <$> chooseInteger p
|
|
genProposalThresholds :: GovernorDatumCases -> Gen ProposalThresholds
|
|
genProposalThresholds c = do
|
|
let validGT = taggedInteger (0, 1000000000)
|
|
execute <- validGT
|
|
create <- validGT
|
|
toVoting <- validGT
|
|
vote <- validGT
|
|
cosign <- validGT
|
|
le0 <- taggedInteger (-1000, -1)
|
|
|
|
case c of
|
|
ExecuteLE0 ->
|
|
-- execute < 0
|
|
return $ ProposalThresholds le0 create toVoting vote cosign
|
|
CreateLE0 ->
|
|
-- c < 0
|
|
return $ ProposalThresholds execute le0 toVoting vote cosign
|
|
ToVotingLE0 ->
|
|
return $ ProposalThresholds execute create le0 vote cosign
|
|
VoteLE0 ->
|
|
-- vote < 0
|
|
return $ ProposalThresholds execute create toVoting le0 cosign
|
|
CosignLE0 ->
|
|
return $ ProposalThresholds execute create toVoting vote le0
|
|
Correct ->
|
|
return $ ProposalThresholds execute create toVoting vote cosign
|
|
|
|
data GovernorPolicyCases
|
|
= ReferenceUTXONotSpent
|
|
| IncorrectAmountOfTokenMinted
|
|
| GovernorOutputNotFound
|
|
| GovernorPolicyCorrect
|
|
deriving stock (Eq, Show)
|
|
|
|
instance Universe GovernorPolicyCases where
|
|
universe =
|
|
[ ReferenceUTXONotSpent
|
|
, IncorrectAmountOfTokenMinted
|
|
, GovernorOutputNotFound
|
|
, GovernorPolicyCorrect
|
|
]
|
|
|
|
instance Finite GovernorPolicyCases where
|
|
universeF = universe
|
|
cardinality = Tagged 4
|
|
|
|
governorMintingProperty :: Property
|
|
governorMintingProperty =
|
|
classifiedPropertyNative gen (const []) expected classifier actual
|
|
where
|
|
{- Note:
|
|
I don't think it's easily possible to randomize orefs. We can't really pass pass `Governor` type to `actual` function.
|
|
-}
|
|
gst = assetClassValue govAssetClass 1
|
|
mintAmount x = mint . mconcat $ replicate x gst
|
|
outputToGov =
|
|
output $
|
|
mconcat
|
|
[ script govValidatorHash
|
|
, withValue gst
|
|
, withDatum govDatum
|
|
]
|
|
referencedInput = input $ withRef gstUTXORef
|
|
|
|
govDatum :: GovernorDatum
|
|
govDatum =
|
|
GovernorDatum
|
|
{ proposalThresholds = def
|
|
, nextProposalId = ProposalId 0
|
|
, proposalTimings = def
|
|
, createProposalTimeRangeMaxWidth = def
|
|
, maximumProposalsPerStake = 3
|
|
}
|
|
|
|
gen :: GovernorPolicyCases -> Gen ScriptContext
|
|
gen c = do
|
|
inputs <- fmap mconcat . listOf1 $ genInput @MintingBuilder
|
|
outputs <- fmap mconcat . listOf1 $ genOutput @MintingBuilder
|
|
toks <- choose (2, 100)
|
|
|
|
let comp =
|
|
case c of
|
|
ReferenceUTXONotSpent -> outputToGov <> mintAmount 1
|
|
IncorrectAmountOfTokenMinted -> referencedInput <> outputToGov <> mintAmount toks
|
|
GovernorOutputNotFound -> referencedInput <> mintAmount 1
|
|
GovernorPolicyCorrect -> referencedInput <> outputToGov <> mintAmount 1
|
|
|
|
return . buildMinting' $ inputs <> outputs <> comp <> withMinting govSymbol
|
|
|
|
expected :: ScriptContext -> Maybe ()
|
|
expected sc =
|
|
case classifier sc of
|
|
GovernorPolicyCorrect -> Just ()
|
|
_ -> Nothing
|
|
|
|
opaqueToUnit :: Term s (POpaque :--> PUnit)
|
|
opaqueToUnit = plam $ \_ -> pconstant ()
|
|
|
|
actual :: Term s (PScriptContext :--> PUnit)
|
|
actual = plam $ \sc -> opaqueToUnit #$ governorPolicy # pconstant governor.gstOutRef # pforgetData (pconstantData ()) # sc
|
|
|
|
classifier :: ScriptContext -> GovernorPolicyCases
|
|
classifier sc
|
|
| minted /= gst = IncorrectAmountOfTokenMinted
|
|
| refInputNotExists = ReferenceUTXONotSpent
|
|
| govOutputNotExists = GovernorOutputNotFound
|
|
| otherwise = GovernorPolicyCorrect
|
|
where
|
|
txinfo = scriptContextTxInfo sc
|
|
minted = txInfoMint txinfo
|
|
refInputNotExists = gstUTXORef `notElem` (txInInfoOutRef <$> txInfoInputs txinfo)
|
|
govOutputNotExists = gst `notElem` (txOutValue <$> txInfoOutputs txinfo)
|
|
|
|
props :: [TestTree]
|
|
props =
|
|
[ testProperty "governorDatumValid" governorDatumValidProperty
|
|
, testProperty "governorPolicy" governorMintingProperty
|
|
]
|